@feynmanzhang/open-party 0.1.3-beta.0 → 0.1.3

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -1 +1 @@
1
- {"version":3,"sources":["../../src/infra/tailscale.ts","../../node_modules/hono/dist/compose.js","../../node_modules/hono/dist/http-exception.js","../../node_modules/hono/dist/request/constants.js","../../node_modules/hono/dist/utils/body.js","../../node_modules/hono/dist/utils/url.js","../../node_modules/hono/dist/request.js","../../node_modules/hono/dist/utils/html.js","../../node_modules/hono/dist/context.js","../../node_modules/hono/dist/router.js","../../node_modules/hono/dist/utils/constants.js","../../node_modules/hono/dist/hono-base.js","../../node_modules/hono/dist/router/reg-exp-router/matcher.js","../../node_modules/hono/dist/router/reg-exp-router/node.js","../../node_modules/hono/dist/router/reg-exp-router/trie.js","../../node_modules/hono/dist/router/reg-exp-router/router.js","../../node_modules/hono/dist/router/reg-exp-router/prepared-router.js","../../node_modules/hono/dist/router/reg-exp-router/index.js","../../node_modules/hono/dist/router/smart-router/router.js","../../node_modules/hono/dist/router/smart-router/index.js","../../node_modules/hono/dist/router/trie-router/node.js","../../node_modules/hono/dist/router/trie-router/router.js","../../node_modules/hono/dist/router/trie-router/index.js","../../node_modules/hono/dist/hono.js","../../node_modules/hono/dist/index.js","../../node_modules/hono/dist/middleware/cors/index.js","../../node_modules/@hono/node-server/dist/index.mjs","../../src/server/config.ts","../../src/server/message-queue.ts","../../src/server/peer-discovery.ts","../../src/server/registry.ts","../../src/server/state.ts","../../src/server/models.ts","../../src/server/routes/agent.ts","../../src/server/routes/proxy.ts","../../src/server/dashboard-html.ts","../../src/server/routes/dashboard.ts","../../src/server/index.ts","../../src/cli/setup.ts","../../src/cli/tailscale-installer.ts","../../src/cli/agent-detector.ts","../../src/cli/agent-installer.ts","../../src/cli/server-utils.ts","../../src/cli/start-server.ts","../../src/cli/stop-server.ts","../../src/cli/status.ts","../../src/cli/index.ts"],"sourcesContent":["/**\r\n * Tailscale integration for the Party Server.\r\n *\r\n * - Binary discovery and caching\r\n * - Tailnet hostname / IP resolution\r\n * - Status JSON reading (network topology)\r\n * - IP → identity whois with TTL cache\r\n * - Funnel / Serve lifecycle management\r\n */\r\n\r\nimport { execFileSync, execSync } from 'node:child_process';\r\nimport { existsSync } from 'node:fs';\r\nimport { join } from 'node:path';\r\n\r\n// ---------------------------------------------------------------------------\r\n// Helpers\r\n// ---------------------------------------------------------------------------\r\n\r\n/** Extract the first valid JSON object from possibly noisy CLI output. */\r\nfunction parsePossiblyNoisyJson(raw: string): Record<string, unknown> {\r\n const trimmed = raw.trim();\r\n const start = trimmed.indexOf('{');\r\n const end = trimmed.lastIndexOf('}');\r\n if (start >= 0 && end > start) {\r\n return JSON.parse(trimmed.slice(start, end + 1));\r\n }\r\n return JSON.parse(trimmed);\r\n}\r\n\r\n/** Run a command synchronously and return stdout. Throws on non-zero exit. */\r\nfunction runExec(cmd: string[], timeout = 5000): string {\r\n return execFileSync(cmd[0], cmd.slice(1), {\r\n timeout,\r\n encoding: 'utf-8',\r\n stdio: ['pipe', 'pipe', 'pipe'],\r\n windowsHide: true,\r\n });\r\n}\r\n\r\n// ---------------------------------------------------------------------------\r\n// Binary discovery\r\n// ---------------------------------------------------------------------------\r\n\r\nlet cachedBinary: string | null = null;\r\n\r\n/** Return true if path exists and can respond to --version. */\r\nfunction checkBinary(path: string, timeout = 3000): boolean {\r\n if (!path || !existsSync(path)) return false;\r\n try {\r\n execFileSync(path, ['--version'], { timeout, encoding: 'utf-8', stdio: 'pipe', windowsHide: true });\r\n return true;\r\n } catch {\r\n return false;\r\n }\r\n}\r\n\r\n/** Locate the tailscale binary using multiple strategies. */\r\nfunction findTailscaleBinary(): string | null {\r\n // Strategy 1 – PATH lookup\r\n try {\r\n const which = process.platform === 'win32' ? 'where' : 'which';\r\n const result = execFileSync(which, ['tailscale'], { timeout: 3000, encoding: 'utf-8', stdio: 'pipe', windowsHide: true });\r\n const fromPath = result.trim().split(/\\r?\\n/)[0];\r\n if (fromPath && checkBinary(fromPath)) return fromPath;\r\n } catch { /* not in PATH */ }\r\n\r\n // Strategy 2 – Known platform paths\r\n const knownPaths = process.platform === 'win32'\r\n ? [\r\n join(process.env.ProgramFiles || 'C:\\\\Program Files', 'Tailscale', 'tailscale.exe'),\r\n join(process.env['ProgramFiles(x86)'] || 'C:\\\\Program Files (x86)', 'Tailscale', 'tailscale.exe'),\r\n join(process.env.LOCALAPPDATA || '', 'Tailscale', 'tailscale.exe'),\r\n ]\r\n : [\r\n '/Applications/Tailscale.app/Contents/MacOS/Tailscale',\r\n '/usr/bin/tailscale',\r\n '/usr/local/bin/tailscale',\r\n ];\r\n\r\n for (const candidate of knownPaths) {\r\n if (checkBinary(candidate)) return candidate;\r\n }\r\n\r\n // Strategy 3 – find in /Applications (macOS only)\r\n if (process.platform !== 'win32') {\r\n try {\r\n const result = execSync(\r\n 'find /Applications -maxdepth 3 -name Tailscale -path \"*/Tailscale.app/Contents/MacOS/Tailscale\"',\r\n { timeout: 5000, encoding: 'utf-8' },\r\n );\r\n const first = result.trim().split('\\n')[0];\r\n if (first && checkBinary(first)) return first;\r\n } catch { /* not found */ }\r\n }\r\n\r\n return null;\r\n}\r\n\r\n/** Return a cached tailscale binary path, discovering it on the first call. */\r\nexport function getTailscaleBinary(): string {\r\n const forced = (process.env.OPENCLAW_TEST_TAILSCALE_BINARY ?? '').trim();\r\n if (forced) {\r\n cachedBinary = forced;\r\n return forced;\r\n }\r\n if (cachedBinary !== null) return cachedBinary;\r\n cachedBinary = findTailscaleBinary() ?? (process.platform === 'win32' ? 'tailscale.exe' : 'tailscale');\r\n return cachedBinary;\r\n}\r\n\r\n// ---------------------------------------------------------------------------\r\n// Status & hostname\r\n// ---------------------------------------------------------------------------\r\n\r\n/** Run `tailscale status --json` and return the parsed result. */\r\nexport function readTailscaleStatus(timeout = 5000): Record<string, unknown> {\r\n const binary = getTailscaleBinary();\r\n const stdout = runExec([binary, 'status', '--json'], timeout);\r\n return stdout.trim() ? parsePossiblyNoisyJson(stdout) : {};\r\n}\r\n\r\n/** Derive the current device's tailnet hostname or fall back to its first Tailscale IP. */\r\nexport function getTailnetHostname(binary?: string): string {\r\n const candidates = binary\r\n ? [binary]\r\n : [\r\n getTailscaleBinary(),\r\n '/Applications/Tailscale.app/Contents/MacOS/Tailscale',\r\n ];\r\n\r\n let lastErr: Error | null = null;\r\n for (const candidate of candidates) {\r\n if (candidate.startsWith('/') && !existsSync(candidate)) continue;\r\n try {\r\n const stdout = runExec([candidate, 'status', '--json'], 5000);\r\n const parsed = stdout.trim() ? parsePossiblyNoisyJson(stdout) : {};\r\n const selfInfo = (parsed.Self ?? {}) as Record<string, unknown>;\r\n\r\n const dns = selfInfo.DNSName as string | undefined;\r\n if (typeof dns === 'string' && dns) return dns.replace(/\\.$/, '');\r\n\r\n const ips = selfInfo.TailscaleIPs as string[] | undefined;\r\n if (ips && ips.length > 0) return ips[0];\r\n\r\n throw new Error('Could not determine Tailscale DNS or IP');\r\n } catch (exc) {\r\n lastErr = exc as Error;\r\n }\r\n }\r\n throw lastErr ?? new Error('Could not determine Tailscale DNS or IP');\r\n}\r\n\r\n/** Return the list of Tailscale IPs for the current device. */\r\nexport function getTailscaleIps(): string[] {\r\n const status = readTailscaleStatus();\r\n const selfInfo = status.Self as Record<string, unknown> | undefined;\r\n if (selfInfo && Array.isArray(selfInfo.TailscaleIPs)) {\r\n return selfInfo.TailscaleIPs as string[];\r\n }\r\n return [];\r\n}\r\n\r\n// ---------------------------------------------------------------------------\r\n// Whois with TTL cache\r\n// ---------------------------------------------------------------------------\r\n\r\nexport class WhoisIdentity {\r\n constructor(\r\n public readonly login: string,\r\n public readonly name?: string,\r\n ) {}\r\n}\r\n\r\ninterface CacheEntry {\r\n value: WhoisIdentity | null;\r\n expiresAt: number;\r\n}\r\n\r\nconst whoisCache = new Map<string, CacheEntry>();\r\n\r\n/** Best-effort extraction of login/name from tailscale whois JSON. */\r\nfunction parseWhoisIdentity(payload: Record<string, unknown>): WhoisIdentity | null {\r\n const userProfile =\r\n (payload.UserProfile as Record<string, unknown>) ??\r\n (payload.userProfile as Record<string, unknown>) ??\r\n (payload.User as Record<string, unknown>) ??\r\n {};\r\n\r\n const login =\r\n (userProfile.LoginName as string) ||\r\n (userProfile.Login as string) ||\r\n (userProfile.login as string) ||\r\n (payload.LoginName as string) ||\r\n (payload.login as string);\r\n\r\n if (typeof login !== 'string' || !login.trim()) return null;\r\n\r\n const rawName =\r\n (userProfile.DisplayName as string) ||\r\n (userProfile.Name as string) ||\r\n (userProfile.displayName as string) ||\r\n (payload.DisplayName as string) ||\r\n (payload.name as string);\r\n\r\n const name = typeof rawName === 'string' ? rawName.trim() : undefined;\r\n return new WhoisIdentity(login.trim(), name);\r\n}\r\n\r\n/** Resolve an IP to a WhoisIdentity via `tailscale whois --json`. */\r\nexport function readWhoisIdentity(\r\n ip: string,\r\n timeout = 5000,\r\n cacheTtl = 60,\r\n errorTtl = 5,\r\n): WhoisIdentity | null {\r\n const normalized = ip.trim();\r\n if (!normalized) return null;\r\n\r\n const now = performance.now() / 1000;\r\n const cached = whoisCache.get(normalized);\r\n if (cached) {\r\n if (cached.expiresAt > now) return cached.value;\r\n whoisCache.delete(normalized);\r\n }\r\n\r\n const binary = getTailscaleBinary();\r\n let identity: WhoisIdentity | null = null;\r\n try {\r\n const stdout = runExec([binary, 'whois', '--json', normalized], timeout);\r\n const parsed = stdout.trim() ? parsePossiblyNoisyJson(stdout) : {};\r\n identity = parseWhoisIdentity(parsed);\r\n } catch { /* lookup failed */ }\r\n\r\n const ttl = identity ? cacheTtl : errorTtl;\r\n whoisCache.set(normalized, { value: identity, expiresAt: now + ttl });\r\n return identity;\r\n}\r\n\r\n// ---------------------------------------------------------------------------\r\n// Funnel / Serve lifecycle\r\n// ---------------------------------------------------------------------------\r\n\r\nconst PERMISSION_KEYWORDS = [\r\n 'permission denied',\r\n 'access denied',\r\n 'operation not permitted',\r\n 'not permitted',\r\n 'requires root',\r\n 'must be run as root',\r\n 'must be run with sudo',\r\n 'requires sudo',\r\n 'need sudo',\r\n];\r\n\r\n/** Run cmd; on permission errors retry with sudo -n. */\r\nfunction execWithSudoFallback(cmd: string[], timeout = 15000): string {\r\n try {\r\n return runExec(cmd, timeout);\r\n } catch (exc: unknown) {\r\n const err = exc as { stderr?: string };\r\n const stderrLower = (err.stderr ?? '').toLowerCase();\r\n if (PERMISSION_KEYWORDS.some((kw) => stderrLower.includes(kw))) {\r\n try {\r\n return runExec(['sudo', '-n', ...cmd], timeout);\r\n } catch { /* sudo failed too */ }\r\n }\r\n throw exc;\r\n }\r\n}\r\n\r\n/** Enable `tailscale serve` on port (background mode). */\r\nexport function enableServe(port: number, timeout = 15000): void {\r\n const binary = getTailscaleBinary();\r\n execWithSudoFallback([binary, 'serve', '--bg', '--yes', String(port)], timeout);\r\n}\r\n\r\n/** Reset `tailscale serve`. */\r\nexport function disableServe(timeout = 15000): void {\r\n const binary = getTailscaleBinary();\r\n execWithSudoFallback([binary, 'serve', 'reset'], timeout);\r\n}\r\n\r\n/** Enable `tailscale funnel` on port (background mode). */\r\nexport function enableFunnel(port: number, timeout = 15000): void {\r\n const binary = getTailscaleBinary();\r\n execWithSudoFallback([binary, 'funnel', '--bg', '--yes', String(port)], timeout);\r\n}\r\n\r\n/** Reset `tailscale funnel`. */\r\nexport function disableFunnel(timeout = 15000): void {\r\n const binary = getTailscaleBinary();\r\n execWithSudoFallback([binary, 'funnel', 'reset'], timeout);\r\n}\r\n\r\n// ---------------------------------------------------------------------------\r\n// Network join & connection status\r\n// ---------------------------------------------------------------------------\r\n\r\n/** Join a Tailscale network using an auth key. */\r\nexport function joinTailnet(\r\n authKey: string,\r\n timeout = 30000,\r\n): { success: boolean; output: string } {\r\n const binary = getTailscaleBinary();\r\n try {\r\n const output = execWithSudoFallback(\r\n [binary, 'up', '--authkey', authKey],\r\n timeout,\r\n );\r\n return { success: true, output: output.trim() };\r\n } catch (e) {\r\n const err = e as { stderr?: string; message?: string };\r\n return { success: false, output: (err.stderr ?? err.message ?? '').trim() };\r\n }\r\n}\r\n\r\n/** Get the current Tailscale connection status. */\r\nexport function getTailscaleConnectionStatus(): {\r\n connected: boolean;\r\n tailscale_ip: string | null;\r\n hostname: string | null;\r\n error?: string;\r\n} {\r\n try {\r\n const status = readTailscaleStatus();\r\n const self = (status.Self ?? {}) as Record<string, unknown>;\r\n const online = self.Online === true;\r\n const ips = self.TailscaleIPs as string[] | undefined;\r\n const dns = (self.DNSName as string | undefined)?.replace(/\\.$/, '');\r\n return {\r\n connected: online,\r\n tailscale_ip: ips?.[0] ?? null,\r\n hostname: dns ?? null,\r\n };\r\n } catch (e) {\r\n return {\r\n connected: false,\r\n tailscale_ip: null,\r\n hostname: null,\r\n error: (e as Error).message,\r\n };\r\n }\r\n}\r\n\r\n// ---------------------------------------------------------------------------\r\n// Tri-state installation status & install instructions\r\n// ---------------------------------------------------------------------------\r\n\r\nexport type TailscaleState =\r\n | { state: 'not_installed'; platform: string }\r\n | { state: 'not_connected'; binary: string; error?: string }\r\n | { state: 'connected'; binary: string; tailscale_ip: string; hostname: string };\r\n\r\n/** Detect Tailscale installation status as one of three states. */\r\nexport function getTailscaleInstallationStatus(): TailscaleState {\r\n // Bypass cache so we always get the real on-disk state\r\n const binary = findTailscaleBinary();\r\n if (!binary) {\r\n return { state: 'not_installed', platform: process.platform };\r\n }\r\n\r\n try {\r\n const status = readTailscaleStatus();\r\n const self = (status.Self ?? {}) as Record<string, unknown>;\r\n const online = self.Online === true;\r\n\r\n if (online) {\r\n const ips = self.TailscaleIPs as string[] | undefined;\r\n const dns = (self.DNSName as string | undefined)?.replace(/\\.$/, '');\r\n return {\r\n state: 'connected',\r\n binary,\r\n tailscale_ip: ips?.[0] ?? '127.0.0.1',\r\n hostname: dns ?? null!,\r\n };\r\n }\r\n\r\n return { state: 'not_connected', binary };\r\n } catch (e) {\r\n return {\r\n state: 'not_connected',\r\n binary,\r\n error: (e as Error).message,\r\n };\r\n }\r\n}\r\n\r\n/** Clear the cached binary path so the next detection re-scans the filesystem. */\r\nexport function resetTailscaleBinaryCache(): void {\r\n cachedBinary = null;\r\n}\r\n\r\n/** Return platform-specific install instructions for Tailscale. */\r\nexport function getInstallInstructions(platform: string): {\r\n os: string;\r\n download_url: string;\r\n commands: string[];\r\n needs_sudo: boolean;\r\n} {\r\n switch (platform) {\r\n case 'linux':\r\n return {\r\n os: 'linux',\r\n download_url: 'https://tailscale.com/download/linux',\r\n commands: ['curl -fsSL https://tailscale.com/install.sh | sh'],\r\n needs_sudo: true,\r\n };\r\n case 'darwin':\r\n return {\r\n os: 'macOS',\r\n download_url: 'https://tailscale.com/download/mac',\r\n commands: ['brew install tailscale'],\r\n needs_sudo: false,\r\n };\r\n case 'win32':\r\n return {\r\n os: 'windows',\r\n download_url: 'https://tailscale.com/download/windows',\r\n commands: ['winget install Tailscale.Tailscale'],\r\n needs_sudo: false,\r\n };\r\n default:\r\n return {\r\n os: platform,\r\n download_url: 'https://tailscale.com/download/',\r\n commands: [],\r\n needs_sudo: false,\r\n };\r\n }\r\n}\r\n","// src/compose.ts\nvar compose = (middleware, onError, onNotFound) => {\n return (context, next) => {\n let index = -1;\n return dispatch(0);\n async function dispatch(i) {\n if (i <= index) {\n throw new Error(\"next() called multiple times\");\n }\n index = i;\n let res;\n let isError = false;\n let handler;\n if (middleware[i]) {\n handler = middleware[i][0][0];\n context.req.routeIndex = i;\n } else {\n handler = i === middleware.length && next || void 0;\n }\n if (handler) {\n try {\n res = await handler(context, () => dispatch(i + 1));\n } catch (err) {\n if (err instanceof Error && onError) {\n context.error = err;\n res = await onError(err, context);\n isError = true;\n } else {\n throw err;\n }\n }\n } else {\n if (context.finalized === false && onNotFound) {\n res = await onNotFound(context);\n }\n }\n if (res && (context.finalized === false || isError)) {\n context.res = res;\n }\n return context;\n }\n };\n};\nexport {\n compose\n};\n","// src/http-exception.ts\nvar HTTPException = class extends Error {\n res;\n status;\n /**\n * Creates an instance of `HTTPException`.\n * @param status - HTTP status code for the exception. Defaults to 500.\n * @param options - Additional options for the exception.\n */\n constructor(status = 500, options) {\n super(options?.message, { cause: options?.cause });\n this.res = options?.res;\n this.status = status;\n }\n /**\n * Returns the response object associated with the exception.\n * If a response object is not provided, a new response is created with the error message and status code.\n * @returns The response object.\n */\n getResponse() {\n if (this.res) {\n const newResponse = new Response(this.res.body, {\n status: this.status,\n headers: this.res.headers\n });\n return newResponse;\n }\n return new Response(this.message, {\n status: this.status\n });\n }\n};\nexport {\n HTTPException\n};\n","// src/request/constants.ts\nvar GET_MATCH_RESULT = /* @__PURE__ */ Symbol();\nexport {\n GET_MATCH_RESULT\n};\n","// src/utils/body.ts\nimport { HonoRequest } from \"../request.js\";\nvar parseBody = async (request, options = /* @__PURE__ */ Object.create(null)) => {\n const { all = false, dot = false } = options;\n const headers = request instanceof HonoRequest ? request.raw.headers : request.headers;\n const contentType = headers.get(\"Content-Type\");\n if (contentType?.startsWith(\"multipart/form-data\") || contentType?.startsWith(\"application/x-www-form-urlencoded\")) {\n return parseFormData(request, { all, dot });\n }\n return {};\n};\nasync function parseFormData(request, options) {\n const formData = await request.formData();\n if (formData) {\n return convertFormDataToBodyData(formData, options);\n }\n return {};\n}\nfunction convertFormDataToBodyData(formData, options) {\n const form = /* @__PURE__ */ Object.create(null);\n formData.forEach((value, key) => {\n const shouldParseAllValues = options.all || key.endsWith(\"[]\");\n if (!shouldParseAllValues) {\n form[key] = value;\n } else {\n handleParsingAllValues(form, key, value);\n }\n });\n if (options.dot) {\n Object.entries(form).forEach(([key, value]) => {\n const shouldParseDotValues = key.includes(\".\");\n if (shouldParseDotValues) {\n handleParsingNestedValues(form, key, value);\n delete form[key];\n }\n });\n }\n return form;\n}\nvar handleParsingAllValues = (form, key, value) => {\n if (form[key] !== void 0) {\n if (Array.isArray(form[key])) {\n ;\n form[key].push(value);\n } else {\n form[key] = [form[key], value];\n }\n } else {\n if (!key.endsWith(\"[]\")) {\n form[key] = value;\n } else {\n form[key] = [value];\n }\n }\n};\nvar handleParsingNestedValues = (form, key, value) => {\n if (/(?:^|\\.)__proto__\\./.test(key)) {\n return;\n }\n let nestedForm = form;\n const keys = key.split(\".\");\n keys.forEach((key2, index) => {\n if (index === keys.length - 1) {\n nestedForm[key2] = value;\n } else {\n if (!nestedForm[key2] || typeof nestedForm[key2] !== \"object\" || Array.isArray(nestedForm[key2]) || nestedForm[key2] instanceof File) {\n nestedForm[key2] = /* @__PURE__ */ Object.create(null);\n }\n nestedForm = nestedForm[key2];\n }\n });\n};\nexport {\n parseBody\n};\n","// src/utils/url.ts\nvar splitPath = (path) => {\n const paths = path.split(\"/\");\n if (paths[0] === \"\") {\n paths.shift();\n }\n return paths;\n};\nvar splitRoutingPath = (routePath) => {\n const { groups, path } = extractGroupsFromPath(routePath);\n const paths = splitPath(path);\n return replaceGroupMarks(paths, groups);\n};\nvar extractGroupsFromPath = (path) => {\n const groups = [];\n path = path.replace(/\\{[^}]+\\}/g, (match, index) => {\n const mark = `@${index}`;\n groups.push([mark, match]);\n return mark;\n });\n return { groups, path };\n};\nvar replaceGroupMarks = (paths, groups) => {\n for (let i = groups.length - 1; i >= 0; i--) {\n const [mark] = groups[i];\n for (let j = paths.length - 1; j >= 0; j--) {\n if (paths[j].includes(mark)) {\n paths[j] = paths[j].replace(mark, groups[i][1]);\n break;\n }\n }\n }\n return paths;\n};\nvar patternCache = {};\nvar getPattern = (label, next) => {\n if (label === \"*\") {\n return \"*\";\n }\n const match = label.match(/^\\:([^\\{\\}]+)(?:\\{(.+)\\})?$/);\n if (match) {\n const cacheKey = `${label}#${next}`;\n if (!patternCache[cacheKey]) {\n if (match[2]) {\n patternCache[cacheKey] = next && next[0] !== \":\" && next[0] !== \"*\" ? [cacheKey, match[1], new RegExp(`^${match[2]}(?=/${next})`)] : [label, match[1], new RegExp(`^${match[2]}$`)];\n } else {\n patternCache[cacheKey] = [label, match[1], true];\n }\n }\n return patternCache[cacheKey];\n }\n return null;\n};\nvar tryDecode = (str, decoder) => {\n try {\n return decoder(str);\n } catch {\n return str.replace(/(?:%[0-9A-Fa-f]{2})+/g, (match) => {\n try {\n return decoder(match);\n } catch {\n return match;\n }\n });\n }\n};\nvar tryDecodeURI = (str) => tryDecode(str, decodeURI);\nvar getPath = (request) => {\n const url = request.url;\n const start = url.indexOf(\"/\", url.indexOf(\":\") + 4);\n let i = start;\n for (; i < url.length; i++) {\n const charCode = url.charCodeAt(i);\n if (charCode === 37) {\n const queryIndex = url.indexOf(\"?\", i);\n const hashIndex = url.indexOf(\"#\", i);\n const end = queryIndex === -1 ? hashIndex === -1 ? void 0 : hashIndex : hashIndex === -1 ? queryIndex : Math.min(queryIndex, hashIndex);\n const path = url.slice(start, end);\n return tryDecodeURI(path.includes(\"%25\") ? path.replace(/%25/g, \"%2525\") : path);\n } else if (charCode === 63 || charCode === 35) {\n break;\n }\n }\n return url.slice(start, i);\n};\nvar getQueryStrings = (url) => {\n const queryIndex = url.indexOf(\"?\", 8);\n return queryIndex === -1 ? \"\" : \"?\" + url.slice(queryIndex + 1);\n};\nvar getPathNoStrict = (request) => {\n const result = getPath(request);\n return result.length > 1 && result.at(-1) === \"/\" ? result.slice(0, -1) : result;\n};\nvar mergePath = (base, sub, ...rest) => {\n if (rest.length) {\n sub = mergePath(sub, ...rest);\n }\n return `${base?.[0] === \"/\" ? \"\" : \"/\"}${base}${sub === \"/\" ? \"\" : `${base?.at(-1) === \"/\" ? \"\" : \"/\"}${sub?.[0] === \"/\" ? sub.slice(1) : sub}`}`;\n};\nvar checkOptionalParameter = (path) => {\n if (path.charCodeAt(path.length - 1) !== 63 || !path.includes(\":\")) {\n return null;\n }\n const segments = path.split(\"/\");\n const results = [];\n let basePath = \"\";\n segments.forEach((segment) => {\n if (segment !== \"\" && !/\\:/.test(segment)) {\n basePath += \"/\" + segment;\n } else if (/\\:/.test(segment)) {\n if (/\\?/.test(segment)) {\n if (results.length === 0 && basePath === \"\") {\n results.push(\"/\");\n } else {\n results.push(basePath);\n }\n const optionalSegment = segment.replace(\"?\", \"\");\n basePath += \"/\" + optionalSegment;\n results.push(basePath);\n } else {\n basePath += \"/\" + segment;\n }\n }\n });\n return results.filter((v, i, a) => a.indexOf(v) === i);\n};\nvar _decodeURI = (value) => {\n if (!/[%+]/.test(value)) {\n return value;\n }\n if (value.indexOf(\"+\") !== -1) {\n value = value.replace(/\\+/g, \" \");\n }\n return value.indexOf(\"%\") !== -1 ? tryDecode(value, decodeURIComponent_) : value;\n};\nvar _getQueryParam = (url, key, multiple) => {\n let encoded;\n if (!multiple && key && !/[%+]/.test(key)) {\n let keyIndex2 = url.indexOf(\"?\", 8);\n if (keyIndex2 === -1) {\n return void 0;\n }\n if (!url.startsWith(key, keyIndex2 + 1)) {\n keyIndex2 = url.indexOf(`&${key}`, keyIndex2 + 1);\n }\n while (keyIndex2 !== -1) {\n const trailingKeyCode = url.charCodeAt(keyIndex2 + key.length + 1);\n if (trailingKeyCode === 61) {\n const valueIndex = keyIndex2 + key.length + 2;\n const endIndex = url.indexOf(\"&\", valueIndex);\n return _decodeURI(url.slice(valueIndex, endIndex === -1 ? void 0 : endIndex));\n } else if (trailingKeyCode == 38 || isNaN(trailingKeyCode)) {\n return \"\";\n }\n keyIndex2 = url.indexOf(`&${key}`, keyIndex2 + 1);\n }\n encoded = /[%+]/.test(url);\n if (!encoded) {\n return void 0;\n }\n }\n const results = {};\n encoded ??= /[%+]/.test(url);\n let keyIndex = url.indexOf(\"?\", 8);\n while (keyIndex !== -1) {\n const nextKeyIndex = url.indexOf(\"&\", keyIndex + 1);\n let valueIndex = url.indexOf(\"=\", keyIndex);\n if (valueIndex > nextKeyIndex && nextKeyIndex !== -1) {\n valueIndex = -1;\n }\n let name = url.slice(\n keyIndex + 1,\n valueIndex === -1 ? nextKeyIndex === -1 ? void 0 : nextKeyIndex : valueIndex\n );\n if (encoded) {\n name = _decodeURI(name);\n }\n keyIndex = nextKeyIndex;\n if (name === \"\") {\n continue;\n }\n let value;\n if (valueIndex === -1) {\n value = \"\";\n } else {\n value = url.slice(valueIndex + 1, nextKeyIndex === -1 ? void 0 : nextKeyIndex);\n if (encoded) {\n value = _decodeURI(value);\n }\n }\n if (multiple) {\n if (!(results[name] && Array.isArray(results[name]))) {\n results[name] = [];\n }\n ;\n results[name].push(value);\n } else {\n results[name] ??= value;\n }\n }\n return key ? results[key] : results;\n};\nvar getQueryParam = _getQueryParam;\nvar getQueryParams = (url, key) => {\n return _getQueryParam(url, key, true);\n};\nvar decodeURIComponent_ = decodeURIComponent;\nexport {\n checkOptionalParameter,\n decodeURIComponent_,\n getPath,\n getPathNoStrict,\n getPattern,\n getQueryParam,\n getQueryParams,\n getQueryStrings,\n mergePath,\n splitPath,\n splitRoutingPath,\n tryDecode,\n tryDecodeURI\n};\n","// src/request.ts\nimport { HTTPException } from \"./http-exception.js\";\nimport { GET_MATCH_RESULT } from \"./request/constants.js\";\nimport { parseBody } from \"./utils/body.js\";\nimport { decodeURIComponent_, getQueryParam, getQueryParams, tryDecode } from \"./utils/url.js\";\nvar tryDecodeURIComponent = (str) => tryDecode(str, decodeURIComponent_);\nvar HonoRequest = class {\n /**\n * `.raw` can get the raw Request object.\n *\n * @see {@link https://hono.dev/docs/api/request#raw}\n *\n * @example\n * ```ts\n * // For Cloudflare Workers\n * app.post('/', async (c) => {\n * const metadata = c.req.raw.cf?.hostMetadata?\n * ...\n * })\n * ```\n */\n raw;\n #validatedData;\n // Short name of validatedData\n #matchResult;\n routeIndex = 0;\n /**\n * `.path` can get the pathname of the request.\n *\n * @see {@link https://hono.dev/docs/api/request#path}\n *\n * @example\n * ```ts\n * app.get('/about/me', (c) => {\n * const pathname = c.req.path // `/about/me`\n * })\n * ```\n */\n path;\n bodyCache = {};\n constructor(request, path = \"/\", matchResult = [[]]) {\n this.raw = request;\n this.path = path;\n this.#matchResult = matchResult;\n this.#validatedData = {};\n }\n param(key) {\n return key ? this.#getDecodedParam(key) : this.#getAllDecodedParams();\n }\n #getDecodedParam(key) {\n const paramKey = this.#matchResult[0][this.routeIndex][1][key];\n const param = this.#getParamValue(paramKey);\n return param && /\\%/.test(param) ? tryDecodeURIComponent(param) : param;\n }\n #getAllDecodedParams() {\n const decoded = {};\n const keys = Object.keys(this.#matchResult[0][this.routeIndex][1]);\n for (const key of keys) {\n const value = this.#getParamValue(this.#matchResult[0][this.routeIndex][1][key]);\n if (value !== void 0) {\n decoded[key] = /\\%/.test(value) ? tryDecodeURIComponent(value) : value;\n }\n }\n return decoded;\n }\n #getParamValue(paramKey) {\n return this.#matchResult[1] ? this.#matchResult[1][paramKey] : paramKey;\n }\n query(key) {\n return getQueryParam(this.url, key);\n }\n queries(key) {\n return getQueryParams(this.url, key);\n }\n header(name) {\n if (name) {\n return this.raw.headers.get(name) ?? void 0;\n }\n const headerData = {};\n this.raw.headers.forEach((value, key) => {\n headerData[key] = value;\n });\n return headerData;\n }\n async parseBody(options) {\n return parseBody(this, options);\n }\n #cachedBody = (key) => {\n const { bodyCache, raw } = this;\n const cachedBody = bodyCache[key];\n if (cachedBody) {\n return cachedBody;\n }\n const anyCachedKey = Object.keys(bodyCache)[0];\n if (anyCachedKey) {\n return bodyCache[anyCachedKey].then((body) => {\n if (anyCachedKey === \"json\") {\n body = JSON.stringify(body);\n }\n return new Response(body)[key]();\n });\n }\n return bodyCache[key] = raw[key]();\n };\n /**\n * `.json()` can parse Request body of type `application/json`\n *\n * @see {@link https://hono.dev/docs/api/request#json}\n *\n * @example\n * ```ts\n * app.post('/entry', async (c) => {\n * const body = await c.req.json()\n * })\n * ```\n */\n json() {\n return this.#cachedBody(\"text\").then((text) => JSON.parse(text));\n }\n /**\n * `.text()` can parse Request body of type `text/plain`\n *\n * @see {@link https://hono.dev/docs/api/request#text}\n *\n * @example\n * ```ts\n * app.post('/entry', async (c) => {\n * const body = await c.req.text()\n * })\n * ```\n */\n text() {\n return this.#cachedBody(\"text\");\n }\n /**\n * `.arrayBuffer()` parse Request body as an `ArrayBuffer`\n *\n * @see {@link https://hono.dev/docs/api/request#arraybuffer}\n *\n * @example\n * ```ts\n * app.post('/entry', async (c) => {\n * const body = await c.req.arrayBuffer()\n * })\n * ```\n */\n arrayBuffer() {\n return this.#cachedBody(\"arrayBuffer\");\n }\n /**\n * `.bytes()` parses the request body as a `Uint8Array`.\n *\n * @see {@link https://hono.dev/docs/api/request#bytes}\n *\n * @example\n * ```ts\n * app.post('/entry', async (c) => {\n * const body = await c.req.bytes()\n * })\n * ```\n */\n bytes() {\n return this.#cachedBody(\"arrayBuffer\").then((buffer) => new Uint8Array(buffer));\n }\n /**\n * Parses the request body as a `Blob`.\n * @example\n * ```ts\n * app.post('/entry', async (c) => {\n * const body = await c.req.blob();\n * });\n * ```\n * @see https://hono.dev/docs/api/request#blob\n */\n blob() {\n return this.#cachedBody(\"blob\");\n }\n /**\n * Parses the request body as `FormData`.\n * @example\n * ```ts\n * app.post('/entry', async (c) => {\n * const body = await c.req.formData();\n * });\n * ```\n * @see https://hono.dev/docs/api/request#formdata\n */\n formData() {\n return this.#cachedBody(\"formData\");\n }\n /**\n * Adds validated data to the request.\n *\n * @param target - The target of the validation.\n * @param data - The validated data to add.\n */\n addValidatedData(target, data) {\n this.#validatedData[target] = data;\n }\n valid(target) {\n return this.#validatedData[target];\n }\n /**\n * `.url()` can get the request url strings.\n *\n * @see {@link https://hono.dev/docs/api/request#url}\n *\n * @example\n * ```ts\n * app.get('/about/me', (c) => {\n * const url = c.req.url // `http://localhost:8787/about/me`\n * ...\n * })\n * ```\n */\n get url() {\n return this.raw.url;\n }\n /**\n * `.method()` can get the method name of the request.\n *\n * @see {@link https://hono.dev/docs/api/request#method}\n *\n * @example\n * ```ts\n * app.get('/about/me', (c) => {\n * const method = c.req.method // `GET`\n * })\n * ```\n */\n get method() {\n return this.raw.method;\n }\n get [GET_MATCH_RESULT]() {\n return this.#matchResult;\n }\n /**\n * `.matchedRoutes()` can return a matched route in the handler\n *\n * @deprecated\n *\n * Use matchedRoutes helper defined in \"hono/route\" instead.\n *\n * @see {@link https://hono.dev/docs/api/request#matchedroutes}\n *\n * @example\n * ```ts\n * app.use('*', async function logger(c, next) {\n * await next()\n * c.req.matchedRoutes.forEach(({ handler, method, path }, i) => {\n * const name = handler.name || (handler.length < 2 ? '[handler]' : '[middleware]')\n * console.log(\n * method,\n * ' ',\n * path,\n * ' '.repeat(Math.max(10 - path.length, 0)),\n * name,\n * i === c.req.routeIndex ? '<- respond from here' : ''\n * )\n * })\n * })\n * ```\n */\n get matchedRoutes() {\n return this.#matchResult[0].map(([[, route]]) => route);\n }\n /**\n * `routePath()` can retrieve the path registered within the handler\n *\n * @deprecated\n *\n * Use routePath helper defined in \"hono/route\" instead.\n *\n * @see {@link https://hono.dev/docs/api/request#routepath}\n *\n * @example\n * ```ts\n * app.get('/posts/:id', (c) => {\n * return c.json({ path: c.req.routePath })\n * })\n * ```\n */\n get routePath() {\n return this.#matchResult[0].map(([[, route]]) => route)[this.routeIndex].path;\n }\n};\nvar cloneRawRequest = async (req) => {\n if (!req.raw.bodyUsed) {\n return req.raw.clone();\n }\n const cacheKey = Object.keys(req.bodyCache)[0];\n if (!cacheKey) {\n throw new HTTPException(500, {\n message: \"Cannot clone request: body was already consumed and not cached. Please use HonoRequest methods (e.g., req.json(), req.text()) instead of consuming req.raw directly.\"\n });\n }\n const requestInit = {\n body: await req[cacheKey](),\n cache: req.raw.cache,\n credentials: req.raw.credentials,\n headers: req.header(),\n integrity: req.raw.integrity,\n keepalive: req.raw.keepalive,\n method: req.method,\n mode: req.raw.mode,\n redirect: req.raw.redirect,\n referrer: req.raw.referrer,\n referrerPolicy: req.raw.referrerPolicy,\n signal: req.raw.signal\n };\n return new Request(req.url, requestInit);\n};\nexport {\n HonoRequest,\n cloneRawRequest\n};\n","// src/utils/html.ts\nvar HtmlEscapedCallbackPhase = {\n Stringify: 1,\n BeforeStream: 2,\n Stream: 3\n};\nvar raw = (value, callbacks) => {\n const escapedString = new String(value);\n escapedString.isEscaped = true;\n escapedString.callbacks = callbacks;\n return escapedString;\n};\nvar escapeRe = /[&<>'\"]/;\nvar stringBufferToString = async (buffer, callbacks) => {\n let str = \"\";\n callbacks ||= [];\n const resolvedBuffer = await Promise.all(buffer);\n for (let i = resolvedBuffer.length - 1; ; i--) {\n str += resolvedBuffer[i];\n i--;\n if (i < 0) {\n break;\n }\n let r = resolvedBuffer[i];\n if (typeof r === \"object\") {\n callbacks.push(...r.callbacks || []);\n }\n const isEscaped = r.isEscaped;\n r = await (typeof r === \"object\" ? r.toString() : r);\n if (typeof r === \"object\") {\n callbacks.push(...r.callbacks || []);\n }\n if (r.isEscaped ?? isEscaped) {\n str += r;\n } else {\n const buf = [str];\n escapeToBuffer(r, buf);\n str = buf[0];\n }\n }\n return raw(str, callbacks);\n};\nvar escapeToBuffer = (str, buffer) => {\n const match = str.search(escapeRe);\n if (match === -1) {\n buffer[0] += str;\n return;\n }\n let escape;\n let index;\n let lastIndex = 0;\n for (index = match; index < str.length; index++) {\n switch (str.charCodeAt(index)) {\n case 34:\n escape = \"&quot;\";\n break;\n case 39:\n escape = \"&#39;\";\n break;\n case 38:\n escape = \"&amp;\";\n break;\n case 60:\n escape = \"&lt;\";\n break;\n case 62:\n escape = \"&gt;\";\n break;\n default:\n continue;\n }\n buffer[0] += str.substring(lastIndex, index) + escape;\n lastIndex = index + 1;\n }\n buffer[0] += str.substring(lastIndex, index);\n};\nvar resolveCallbackSync = (str) => {\n const callbacks = str.callbacks;\n if (!callbacks?.length) {\n return str;\n }\n const buffer = [str];\n const context = {};\n callbacks.forEach((c) => c({ phase: HtmlEscapedCallbackPhase.Stringify, buffer, context }));\n return buffer[0];\n};\nvar resolveCallback = async (str, phase, preserveCallbacks, context, buffer) => {\n if (typeof str === \"object\" && !(str instanceof String)) {\n if (!(str instanceof Promise)) {\n str = str.toString();\n }\n if (str instanceof Promise) {\n str = await str;\n }\n }\n const callbacks = str.callbacks;\n if (!callbacks?.length) {\n return Promise.resolve(str);\n }\n if (buffer) {\n buffer[0] += str;\n } else {\n buffer = [str];\n }\n const resStr = Promise.all(callbacks.map((c) => c({ phase, buffer, context }))).then(\n (res) => Promise.all(\n res.filter(Boolean).map((str2) => resolveCallback(str2, phase, false, context, buffer))\n ).then(() => buffer[0])\n );\n if (preserveCallbacks) {\n return raw(await resStr, callbacks);\n } else {\n return resStr;\n }\n};\nexport {\n HtmlEscapedCallbackPhase,\n escapeToBuffer,\n raw,\n resolveCallback,\n resolveCallbackSync,\n stringBufferToString\n};\n","// src/context.ts\nimport { HonoRequest } from \"./request.js\";\nimport { HtmlEscapedCallbackPhase, resolveCallback } from \"./utils/html.js\";\nvar TEXT_PLAIN = \"text/plain; charset=UTF-8\";\nvar setDefaultContentType = (contentType, headers) => {\n return {\n \"Content-Type\": contentType,\n ...headers\n };\n};\nvar createResponseInstance = (body, init) => new Response(body, init);\nvar Context = class {\n #rawRequest;\n #req;\n /**\n * `.env` can get bindings (environment variables, secrets, KV namespaces, D1 database, R2 bucket etc.) in Cloudflare Workers.\n *\n * @see {@link https://hono.dev/docs/api/context#env}\n *\n * @example\n * ```ts\n * // Environment object for Cloudflare Workers\n * app.get('*', async c => {\n * const counter = c.env.COUNTER\n * })\n * ```\n */\n env = {};\n #var;\n finalized = false;\n /**\n * `.error` can get the error object from the middleware if the Handler throws an error.\n *\n * @see {@link https://hono.dev/docs/api/context#error}\n *\n * @example\n * ```ts\n * app.use('*', async (c, next) => {\n * await next()\n * if (c.error) {\n * // do something...\n * }\n * })\n * ```\n */\n error;\n #status;\n #executionCtx;\n #res;\n #layout;\n #renderer;\n #notFoundHandler;\n #preparedHeaders;\n #matchResult;\n #path;\n /**\n * Creates an instance of the Context class.\n *\n * @param req - The Request object.\n * @param options - Optional configuration options for the context.\n */\n constructor(req, options) {\n this.#rawRequest = req;\n if (options) {\n this.#executionCtx = options.executionCtx;\n this.env = options.env;\n this.#notFoundHandler = options.notFoundHandler;\n this.#path = options.path;\n this.#matchResult = options.matchResult;\n }\n }\n /**\n * `.req` is the instance of {@link HonoRequest}.\n */\n get req() {\n this.#req ??= new HonoRequest(this.#rawRequest, this.#path, this.#matchResult);\n return this.#req;\n }\n /**\n * @see {@link https://hono.dev/docs/api/context#event}\n * The FetchEvent associated with the current request.\n *\n * @throws Will throw an error if the context does not have a FetchEvent.\n */\n get event() {\n if (this.#executionCtx && \"respondWith\" in this.#executionCtx) {\n return this.#executionCtx;\n } else {\n throw Error(\"This context has no FetchEvent\");\n }\n }\n /**\n * @see {@link https://hono.dev/docs/api/context#executionctx}\n * The ExecutionContext associated with the current request.\n *\n * @throws Will throw an error if the context does not have an ExecutionContext.\n */\n get executionCtx() {\n if (this.#executionCtx) {\n return this.#executionCtx;\n } else {\n throw Error(\"This context has no ExecutionContext\");\n }\n }\n /**\n * @see {@link https://hono.dev/docs/api/context#res}\n * The Response object for the current request.\n */\n get res() {\n return this.#res ||= createResponseInstance(null, {\n headers: this.#preparedHeaders ??= new Headers()\n });\n }\n /**\n * Sets the Response object for the current request.\n *\n * @param _res - The Response object to set.\n */\n set res(_res) {\n if (this.#res && _res) {\n _res = createResponseInstance(_res.body, _res);\n for (const [k, v] of this.#res.headers.entries()) {\n if (k === \"content-type\") {\n continue;\n }\n if (k === \"set-cookie\") {\n const cookies = this.#res.headers.getSetCookie();\n _res.headers.delete(\"set-cookie\");\n for (const cookie of cookies) {\n _res.headers.append(\"set-cookie\", cookie);\n }\n } else {\n _res.headers.set(k, v);\n }\n }\n }\n this.#res = _res;\n this.finalized = true;\n }\n /**\n * `.render()` can create a response within a layout.\n *\n * @see {@link https://hono.dev/docs/api/context#render-setrenderer}\n *\n * @example\n * ```ts\n * app.get('/', (c) => {\n * return c.render('Hello!')\n * })\n * ```\n */\n render = (...args) => {\n this.#renderer ??= (content) => this.html(content);\n return this.#renderer(...args);\n };\n /**\n * Sets the layout for the response.\n *\n * @param layout - The layout to set.\n * @returns The layout function.\n */\n setLayout = (layout) => this.#layout = layout;\n /**\n * Gets the current layout for the response.\n *\n * @returns The current layout function.\n */\n getLayout = () => this.#layout;\n /**\n * `.setRenderer()` can set the layout in the custom middleware.\n *\n * @see {@link https://hono.dev/docs/api/context#render-setrenderer}\n *\n * @example\n * ```tsx\n * app.use('*', async (c, next) => {\n * c.setRenderer((content) => {\n * return c.html(\n * <html>\n * <body>\n * <p>{content}</p>\n * </body>\n * </html>\n * )\n * })\n * await next()\n * })\n * ```\n */\n setRenderer = (renderer) => {\n this.#renderer = renderer;\n };\n /**\n * `.header()` can set headers.\n *\n * @see {@link https://hono.dev/docs/api/context#header}\n *\n * @example\n * ```ts\n * app.get('/welcome', (c) => {\n * // Set headers\n * c.header('X-Message', 'Hello!')\n * c.header('Content-Type', 'text/plain')\n *\n * return c.body('Thank you for coming')\n * })\n * ```\n */\n header = (name, value, options) => {\n if (this.finalized) {\n this.#res = createResponseInstance(this.#res.body, this.#res);\n }\n const headers = this.#res ? this.#res.headers : this.#preparedHeaders ??= new Headers();\n if (value === void 0) {\n headers.delete(name);\n } else if (options?.append) {\n headers.append(name, value);\n } else {\n headers.set(name, value);\n }\n };\n status = (status) => {\n this.#status = status;\n };\n /**\n * `.set()` can set the value specified by the key.\n *\n * @see {@link https://hono.dev/docs/api/context#set-get}\n *\n * @example\n * ```ts\n * app.use('*', async (c, next) => {\n * c.set('message', 'Hono is hot!!')\n * await next()\n * })\n * ```\n */\n set = (key, value) => {\n this.#var ??= /* @__PURE__ */ new Map();\n this.#var.set(key, value);\n };\n /**\n * `.get()` can use the value specified by the key.\n *\n * @see {@link https://hono.dev/docs/api/context#set-get}\n *\n * @example\n * ```ts\n * app.get('/', (c) => {\n * const message = c.get('message')\n * return c.text(`The message is \"${message}\"`)\n * })\n * ```\n */\n get = (key) => {\n return this.#var ? this.#var.get(key) : void 0;\n };\n /**\n * `.var` can access the value of a variable.\n *\n * @see {@link https://hono.dev/docs/api/context#var}\n *\n * @example\n * ```ts\n * const result = c.var.client.oneMethod()\n * ```\n */\n // c.var.propName is a read-only\n get var() {\n if (!this.#var) {\n return {};\n }\n return Object.fromEntries(this.#var);\n }\n #newResponse(data, arg, headers) {\n const responseHeaders = this.#res ? new Headers(this.#res.headers) : this.#preparedHeaders ?? new Headers();\n if (typeof arg === \"object\" && \"headers\" in arg) {\n const argHeaders = arg.headers instanceof Headers ? arg.headers : new Headers(arg.headers);\n for (const [key, value] of argHeaders) {\n if (key.toLowerCase() === \"set-cookie\") {\n responseHeaders.append(key, value);\n } else {\n responseHeaders.set(key, value);\n }\n }\n }\n if (headers) {\n for (const [k, v] of Object.entries(headers)) {\n if (typeof v === \"string\") {\n responseHeaders.set(k, v);\n } else {\n responseHeaders.delete(k);\n for (const v2 of v) {\n responseHeaders.append(k, v2);\n }\n }\n }\n }\n const status = typeof arg === \"number\" ? arg : arg?.status ?? this.#status;\n return createResponseInstance(data, { status, headers: responseHeaders });\n }\n newResponse = (...args) => this.#newResponse(...args);\n /**\n * `.body()` can return the HTTP response.\n * You can set headers with `.header()` and set HTTP status code with `.status`.\n * This can also be set in `.text()`, `.json()` and so on.\n *\n * @see {@link https://hono.dev/docs/api/context#body}\n *\n * @example\n * ```ts\n * app.get('/welcome', (c) => {\n * // Set headers\n * c.header('X-Message', 'Hello!')\n * c.header('Content-Type', 'text/plain')\n * // Set HTTP status code\n * c.status(201)\n *\n * // Return the response body\n * return c.body('Thank you for coming')\n * })\n * ```\n */\n body = (data, arg, headers) => this.#newResponse(data, arg, headers);\n /**\n * `.text()` can render text as `Content-Type:text/plain`.\n *\n * @see {@link https://hono.dev/docs/api/context#text}\n *\n * @example\n * ```ts\n * app.get('/say', (c) => {\n * return c.text('Hello!')\n * })\n * ```\n */\n text = (text, arg, headers) => {\n return !this.#preparedHeaders && !this.#status && !arg && !headers && !this.finalized ? new Response(text) : this.#newResponse(\n text,\n arg,\n setDefaultContentType(TEXT_PLAIN, headers)\n );\n };\n /**\n * `.json()` can render JSON as `Content-Type:application/json`.\n *\n * @see {@link https://hono.dev/docs/api/context#json}\n *\n * @example\n * ```ts\n * app.get('/api', (c) => {\n * return c.json({ message: 'Hello!' })\n * })\n * ```\n */\n json = (object, arg, headers) => {\n return this.#newResponse(\n JSON.stringify(object),\n arg,\n setDefaultContentType(\"application/json\", headers)\n );\n };\n html = (html, arg, headers) => {\n const res = (html2) => this.#newResponse(html2, arg, setDefaultContentType(\"text/html; charset=UTF-8\", headers));\n return typeof html === \"object\" ? resolveCallback(html, HtmlEscapedCallbackPhase.Stringify, false, {}).then(res) : res(html);\n };\n /**\n * `.redirect()` can Redirect, default status code is 302.\n *\n * @see {@link https://hono.dev/docs/api/context#redirect}\n *\n * @example\n * ```ts\n * app.get('/redirect', (c) => {\n * return c.redirect('/')\n * })\n * app.get('/redirect-permanently', (c) => {\n * return c.redirect('/', 301)\n * })\n * ```\n */\n redirect = (location, status) => {\n const locationString = String(location);\n this.header(\n \"Location\",\n // Multibyes should be encoded\n // eslint-disable-next-line no-control-regex\n !/[^\\x00-\\xFF]/.test(locationString) ? locationString : encodeURI(locationString)\n );\n return this.newResponse(null, status ?? 302);\n };\n /**\n * `.notFound()` can return the Not Found Response.\n *\n * @see {@link https://hono.dev/docs/api/context#notfound}\n *\n * @example\n * ```ts\n * app.get('/notfound', (c) => {\n * return c.notFound()\n * })\n * ```\n */\n notFound = () => {\n this.#notFoundHandler ??= () => createResponseInstance();\n return this.#notFoundHandler(this);\n };\n};\nexport {\n Context,\n TEXT_PLAIN\n};\n","// src/router.ts\nvar METHOD_NAME_ALL = \"ALL\";\nvar METHOD_NAME_ALL_LOWERCASE = \"all\";\nvar METHODS = [\"get\", \"post\", \"put\", \"delete\", \"options\", \"patch\"];\nvar MESSAGE_MATCHER_IS_ALREADY_BUILT = \"Can not add a route since the matcher is already built.\";\nvar UnsupportedPathError = class extends Error {\n};\nexport {\n MESSAGE_MATCHER_IS_ALREADY_BUILT,\n METHODS,\n METHOD_NAME_ALL,\n METHOD_NAME_ALL_LOWERCASE,\n UnsupportedPathError\n};\n","// src/utils/constants.ts\nvar COMPOSED_HANDLER = \"__COMPOSED_HANDLER\";\nexport {\n COMPOSED_HANDLER\n};\n","// src/hono-base.ts\nimport { compose } from \"./compose.js\";\nimport { Context } from \"./context.js\";\nimport { METHODS, METHOD_NAME_ALL, METHOD_NAME_ALL_LOWERCASE } from \"./router.js\";\nimport { COMPOSED_HANDLER } from \"./utils/constants.js\";\nimport { getPath, getPathNoStrict, mergePath } from \"./utils/url.js\";\nvar notFoundHandler = (c) => {\n return c.text(\"404 Not Found\", 404);\n};\nvar errorHandler = (err, c) => {\n if (\"getResponse\" in err) {\n const res = err.getResponse();\n return c.newResponse(res.body, res);\n }\n console.error(err);\n return c.text(\"Internal Server Error\", 500);\n};\nvar Hono = class _Hono {\n get;\n post;\n put;\n delete;\n options;\n patch;\n all;\n on;\n use;\n /*\n This class is like an abstract class and does not have a router.\n To use it, inherit the class and implement router in the constructor.\n */\n router;\n getPath;\n // Cannot use `#` because it requires visibility at JavaScript runtime.\n _basePath = \"/\";\n #path = \"/\";\n routes = [];\n constructor(options = {}) {\n const allMethods = [...METHODS, METHOD_NAME_ALL_LOWERCASE];\n allMethods.forEach((method) => {\n this[method] = (args1, ...args) => {\n if (typeof args1 === \"string\") {\n this.#path = args1;\n } else {\n this.#addRoute(method, this.#path, args1);\n }\n args.forEach((handler) => {\n this.#addRoute(method, this.#path, handler);\n });\n return this;\n };\n });\n this.on = (method, path, ...handlers) => {\n for (const p of [path].flat()) {\n this.#path = p;\n for (const m of [method].flat()) {\n handlers.map((handler) => {\n this.#addRoute(m.toUpperCase(), this.#path, handler);\n });\n }\n }\n return this;\n };\n this.use = (arg1, ...handlers) => {\n if (typeof arg1 === \"string\") {\n this.#path = arg1;\n } else {\n this.#path = \"*\";\n handlers.unshift(arg1);\n }\n handlers.forEach((handler) => {\n this.#addRoute(METHOD_NAME_ALL, this.#path, handler);\n });\n return this;\n };\n const { strict, ...optionsWithoutStrict } = options;\n Object.assign(this, optionsWithoutStrict);\n this.getPath = strict ?? true ? options.getPath ?? getPath : getPathNoStrict;\n }\n #clone() {\n const clone = new _Hono({\n router: this.router,\n getPath: this.getPath\n });\n clone.errorHandler = this.errorHandler;\n clone.#notFoundHandler = this.#notFoundHandler;\n clone.routes = this.routes;\n return clone;\n }\n #notFoundHandler = notFoundHandler;\n // Cannot use `#` because it requires visibility at JavaScript runtime.\n errorHandler = errorHandler;\n /**\n * `.route()` allows grouping other Hono instance in routes.\n *\n * @see {@link https://hono.dev/docs/api/routing#grouping}\n *\n * @param {string} path - base Path\n * @param {Hono} app - other Hono instance\n * @returns {Hono} routed Hono instance\n *\n * @example\n * ```ts\n * const app = new Hono()\n * const app2 = new Hono()\n *\n * app2.get(\"/user\", (c) => c.text(\"user\"))\n * app.route(\"/api\", app2) // GET /api/user\n * ```\n */\n route(path, app) {\n const subApp = this.basePath(path);\n app.routes.map((r) => {\n let handler;\n if (app.errorHandler === errorHandler) {\n handler = r.handler;\n } else {\n handler = async (c, next) => (await compose([], app.errorHandler)(c, () => r.handler(c, next))).res;\n handler[COMPOSED_HANDLER] = r.handler;\n }\n subApp.#addRoute(r.method, r.path, handler, r.basePath);\n });\n return this;\n }\n /**\n * `.basePath()` allows base paths to be specified.\n *\n * @see {@link https://hono.dev/docs/api/routing#base-path}\n *\n * @param {string} path - base Path\n * @returns {Hono} changed Hono instance\n *\n * @example\n * ```ts\n * const api = new Hono().basePath('/api')\n * ```\n */\n basePath(path) {\n const subApp = this.#clone();\n subApp._basePath = mergePath(this._basePath, path);\n return subApp;\n }\n /**\n * `.onError()` handles an error and returns a customized Response.\n *\n * @see {@link https://hono.dev/docs/api/hono#error-handling}\n *\n * @param {ErrorHandler} handler - request Handler for error\n * @returns {Hono} changed Hono instance\n *\n * @example\n * ```ts\n * app.onError((err, c) => {\n * console.error(`${err}`)\n * return c.text('Custom Error Message', 500)\n * })\n * ```\n */\n onError = (handler) => {\n this.errorHandler = handler;\n return this;\n };\n /**\n * `.notFound()` allows you to customize a Not Found Response.\n *\n * @see {@link https://hono.dev/docs/api/hono#not-found}\n *\n * @param {NotFoundHandler} handler - request handler for not-found\n * @returns {Hono} changed Hono instance\n *\n * @example\n * ```ts\n * app.notFound((c) => {\n * return c.text('Custom 404 Message', 404)\n * })\n * ```\n */\n notFound = (handler) => {\n this.#notFoundHandler = handler;\n return this;\n };\n /**\n * `.mount()` allows you to mount applications built with other frameworks into your Hono application.\n *\n * @see {@link https://hono.dev/docs/api/hono#mount}\n *\n * @param {string} path - base Path\n * @param {Function} applicationHandler - other Request Handler\n * @param {MountOptions} [options] - options of `.mount()`\n * @returns {Hono} mounted Hono instance\n *\n * @example\n * ```ts\n * import { Router as IttyRouter } from 'itty-router'\n * import { Hono } from 'hono'\n * // Create itty-router application\n * const ittyRouter = IttyRouter()\n * // GET /itty-router/hello\n * ittyRouter.get('/hello', () => new Response('Hello from itty-router'))\n *\n * const app = new Hono()\n * app.mount('/itty-router', ittyRouter.handle)\n * ```\n *\n * @example\n * ```ts\n * const app = new Hono()\n * // Send the request to another application without modification.\n * app.mount('/app', anotherApp, {\n * replaceRequest: (req) => req,\n * })\n * ```\n */\n mount(path, applicationHandler, options) {\n let replaceRequest;\n let optionHandler;\n if (options) {\n if (typeof options === \"function\") {\n optionHandler = options;\n } else {\n optionHandler = options.optionHandler;\n if (options.replaceRequest === false) {\n replaceRequest = (request) => request;\n } else {\n replaceRequest = options.replaceRequest;\n }\n }\n }\n const getOptions = optionHandler ? (c) => {\n const options2 = optionHandler(c);\n return Array.isArray(options2) ? options2 : [options2];\n } : (c) => {\n let executionContext = void 0;\n try {\n executionContext = c.executionCtx;\n } catch {\n }\n return [c.env, executionContext];\n };\n replaceRequest ||= (() => {\n const mergedPath = mergePath(this._basePath, path);\n const pathPrefixLength = mergedPath === \"/\" ? 0 : mergedPath.length;\n return (request) => {\n const url = new URL(request.url);\n url.pathname = this.getPath(request).slice(pathPrefixLength) || \"/\";\n return new Request(url, request);\n };\n })();\n const handler = async (c, next) => {\n const res = await applicationHandler(replaceRequest(c.req.raw), ...getOptions(c));\n if (res) {\n return res;\n }\n await next();\n };\n this.#addRoute(METHOD_NAME_ALL, mergePath(path, \"*\"), handler);\n return this;\n }\n #addRoute(method, path, handler, baseRoutePath) {\n method = method.toUpperCase();\n path = mergePath(this._basePath, path);\n const r = {\n basePath: baseRoutePath !== void 0 ? mergePath(this._basePath, baseRoutePath) : this._basePath,\n path,\n method,\n handler\n };\n this.router.add(method, path, [handler, r]);\n this.routes.push(r);\n }\n #handleError(err, c) {\n if (err instanceof Error) {\n return this.errorHandler(err, c);\n }\n throw err;\n }\n #dispatch(request, executionCtx, env, method) {\n if (method === \"HEAD\") {\n return (async () => new Response(null, await this.#dispatch(request, executionCtx, env, \"GET\")))();\n }\n const path = this.getPath(request, { env });\n const matchResult = this.router.match(method, path);\n const c = new Context(request, {\n path,\n matchResult,\n env,\n executionCtx,\n notFoundHandler: this.#notFoundHandler\n });\n if (matchResult[0].length === 1) {\n let res;\n try {\n res = matchResult[0][0][0][0](c, async () => {\n c.res = await this.#notFoundHandler(c);\n });\n } catch (err) {\n return this.#handleError(err, c);\n }\n return res instanceof Promise ? res.then(\n (resolved) => resolved || (c.finalized ? c.res : this.#notFoundHandler(c))\n ).catch((err) => this.#handleError(err, c)) : res ?? this.#notFoundHandler(c);\n }\n const composed = compose(matchResult[0], this.errorHandler, this.#notFoundHandler);\n return (async () => {\n try {\n const context = await composed(c);\n if (!context.finalized) {\n throw new Error(\n \"Context is not finalized. Did you forget to return a Response object or `await next()`?\"\n );\n }\n return context.res;\n } catch (err) {\n return this.#handleError(err, c);\n }\n })();\n }\n /**\n * `.fetch()` will be entry point of your app.\n *\n * @see {@link https://hono.dev/docs/api/hono#fetch}\n *\n * @param {Request} request - request Object of request\n * @param {Env} Env - env Object\n * @param {ExecutionContext} - context of execution\n * @returns {Response | Promise<Response>} response of request\n *\n */\n fetch = (request, ...rest) => {\n return this.#dispatch(request, rest[1], rest[0], request.method);\n };\n /**\n * `.request()` is a useful method for testing.\n * You can pass a URL or pathname to send a GET request.\n * app will return a Response object.\n * ```ts\n * test('GET /hello is ok', async () => {\n * const res = await app.request('/hello')\n * expect(res.status).toBe(200)\n * })\n * ```\n * @see https://hono.dev/docs/api/hono#request\n */\n request = (input, requestInit, Env, executionCtx) => {\n if (input instanceof Request) {\n return this.fetch(requestInit ? new Request(input, requestInit) : input, Env, executionCtx);\n }\n input = input.toString();\n return this.fetch(\n new Request(\n /^https?:\\/\\//.test(input) ? input : `http://localhost${mergePath(\"/\", input)}`,\n requestInit\n ),\n Env,\n executionCtx\n );\n };\n /**\n * `.fire()` automatically adds a global fetch event listener.\n * This can be useful for environments that adhere to the Service Worker API, such as non-ES module Cloudflare Workers.\n * @deprecated\n * Use `fire` from `hono/service-worker` instead.\n * ```ts\n * import { Hono } from 'hono'\n * import { fire } from 'hono/service-worker'\n *\n * const app = new Hono()\n * // ...\n * fire(app)\n * ```\n * @see https://hono.dev/docs/api/hono#fire\n * @see https://developer.mozilla.org/en-US/docs/Web/API/Service_Worker_API\n * @see https://developers.cloudflare.com/workers/reference/migrate-to-module-workers/\n */\n fire = () => {\n addEventListener(\"fetch\", (event) => {\n event.respondWith(this.#dispatch(event.request, event, void 0, event.request.method));\n });\n };\n};\nexport {\n Hono as HonoBase\n};\n","// src/router/reg-exp-router/matcher.ts\nimport { METHOD_NAME_ALL } from \"../../router.js\";\nvar emptyParam = [];\nfunction match(method, path) {\n const matchers = this.buildAllMatchers();\n const match2 = ((method2, path2) => {\n const matcher = matchers[method2] || matchers[METHOD_NAME_ALL];\n const staticMatch = matcher[2][path2];\n if (staticMatch) {\n return staticMatch;\n }\n const match3 = path2.match(matcher[0]);\n if (!match3) {\n return [[], emptyParam];\n }\n const index = match3.indexOf(\"\", 1);\n return [matcher[1][index], match3];\n });\n this.match = match2;\n return match2(method, path);\n}\nexport {\n emptyParam,\n match\n};\n","// src/router/reg-exp-router/node.ts\nvar LABEL_REG_EXP_STR = \"[^/]+\";\nvar ONLY_WILDCARD_REG_EXP_STR = \".*\";\nvar TAIL_WILDCARD_REG_EXP_STR = \"(?:|/.*)\";\nvar PATH_ERROR = /* @__PURE__ */ Symbol();\nvar regExpMetaChars = new Set(\".\\\\+*[^]$()\");\nfunction compareKey(a, b) {\n if (a.length === 1) {\n return b.length === 1 ? a < b ? -1 : 1 : -1;\n }\n if (b.length === 1) {\n return 1;\n }\n if (a === ONLY_WILDCARD_REG_EXP_STR || a === TAIL_WILDCARD_REG_EXP_STR) {\n return 1;\n } else if (b === ONLY_WILDCARD_REG_EXP_STR || b === TAIL_WILDCARD_REG_EXP_STR) {\n return -1;\n }\n if (a === LABEL_REG_EXP_STR) {\n return 1;\n } else if (b === LABEL_REG_EXP_STR) {\n return -1;\n }\n return a.length === b.length ? a < b ? -1 : 1 : b.length - a.length;\n}\nvar Node = class _Node {\n #index;\n #varIndex;\n #children = /* @__PURE__ */ Object.create(null);\n insert(tokens, index, paramMap, context, pathErrorCheckOnly) {\n if (tokens.length === 0) {\n if (this.#index !== void 0) {\n throw PATH_ERROR;\n }\n if (pathErrorCheckOnly) {\n return;\n }\n this.#index = index;\n return;\n }\n const [token, ...restTokens] = tokens;\n const pattern = token === \"*\" ? restTokens.length === 0 ? [\"\", \"\", ONLY_WILDCARD_REG_EXP_STR] : [\"\", \"\", LABEL_REG_EXP_STR] : token === \"/*\" ? [\"\", \"\", TAIL_WILDCARD_REG_EXP_STR] : token.match(/^\\:([^\\{\\}]+)(?:\\{(.+)\\})?$/);\n let node;\n if (pattern) {\n const name = pattern[1];\n let regexpStr = pattern[2] || LABEL_REG_EXP_STR;\n if (name && pattern[2]) {\n if (regexpStr === \".*\") {\n throw PATH_ERROR;\n }\n regexpStr = regexpStr.replace(/^\\((?!\\?:)(?=[^)]+\\)$)/, \"(?:\");\n if (/\\((?!\\?:)/.test(regexpStr)) {\n throw PATH_ERROR;\n }\n }\n node = this.#children[regexpStr];\n if (!node) {\n if (Object.keys(this.#children).some(\n (k) => k !== ONLY_WILDCARD_REG_EXP_STR && k !== TAIL_WILDCARD_REG_EXP_STR\n )) {\n throw PATH_ERROR;\n }\n if (pathErrorCheckOnly) {\n return;\n }\n node = this.#children[regexpStr] = new _Node();\n if (name !== \"\") {\n node.#varIndex = context.varIndex++;\n }\n }\n if (!pathErrorCheckOnly && name !== \"\") {\n paramMap.push([name, node.#varIndex]);\n }\n } else {\n node = this.#children[token];\n if (!node) {\n if (Object.keys(this.#children).some(\n (k) => k.length > 1 && k !== ONLY_WILDCARD_REG_EXP_STR && k !== TAIL_WILDCARD_REG_EXP_STR\n )) {\n throw PATH_ERROR;\n }\n if (pathErrorCheckOnly) {\n return;\n }\n node = this.#children[token] = new _Node();\n }\n }\n node.insert(restTokens, index, paramMap, context, pathErrorCheckOnly);\n }\n buildRegExpStr() {\n const childKeys = Object.keys(this.#children).sort(compareKey);\n const strList = childKeys.map((k) => {\n const c = this.#children[k];\n return (typeof c.#varIndex === \"number\" ? `(${k})@${c.#varIndex}` : regExpMetaChars.has(k) ? `\\\\${k}` : k) + c.buildRegExpStr();\n });\n if (typeof this.#index === \"number\") {\n strList.unshift(`#${this.#index}`);\n }\n if (strList.length === 0) {\n return \"\";\n }\n if (strList.length === 1) {\n return strList[0];\n }\n return \"(?:\" + strList.join(\"|\") + \")\";\n }\n};\nexport {\n Node,\n PATH_ERROR\n};\n","// src/router/reg-exp-router/trie.ts\nimport { Node } from \"./node.js\";\nvar Trie = class {\n #context = { varIndex: 0 };\n #root = new Node();\n insert(path, index, pathErrorCheckOnly) {\n const paramAssoc = [];\n const groups = [];\n for (let i = 0; ; ) {\n let replaced = false;\n path = path.replace(/\\{[^}]+\\}/g, (m) => {\n const mark = `@\\\\${i}`;\n groups[i] = [mark, m];\n i++;\n replaced = true;\n return mark;\n });\n if (!replaced) {\n break;\n }\n }\n const tokens = path.match(/(?::[^\\/]+)|(?:\\/\\*$)|./g) || [];\n for (let i = groups.length - 1; i >= 0; i--) {\n const [mark] = groups[i];\n for (let j = tokens.length - 1; j >= 0; j--) {\n if (tokens[j].indexOf(mark) !== -1) {\n tokens[j] = tokens[j].replace(mark, groups[i][1]);\n break;\n }\n }\n }\n this.#root.insert(tokens, index, paramAssoc, this.#context, pathErrorCheckOnly);\n return paramAssoc;\n }\n buildRegExp() {\n let regexp = this.#root.buildRegExpStr();\n if (regexp === \"\") {\n return [/^$/, [], []];\n }\n let captureIndex = 0;\n const indexReplacementMap = [];\n const paramReplacementMap = [];\n regexp = regexp.replace(/#(\\d+)|@(\\d+)|\\.\\*\\$/g, (_, handlerIndex, paramIndex) => {\n if (handlerIndex !== void 0) {\n indexReplacementMap[++captureIndex] = Number(handlerIndex);\n return \"$()\";\n }\n if (paramIndex !== void 0) {\n paramReplacementMap[Number(paramIndex)] = ++captureIndex;\n return \"\";\n }\n return \"\";\n });\n return [new RegExp(`^${regexp}`), indexReplacementMap, paramReplacementMap];\n }\n};\nexport {\n Trie\n};\n","// src/router/reg-exp-router/router.ts\nimport {\n MESSAGE_MATCHER_IS_ALREADY_BUILT,\n METHOD_NAME_ALL,\n UnsupportedPathError\n} from \"../../router.js\";\nimport { checkOptionalParameter } from \"../../utils/url.js\";\nimport { match, emptyParam } from \"./matcher.js\";\nimport { PATH_ERROR } from \"./node.js\";\nimport { Trie } from \"./trie.js\";\nvar nullMatcher = [/^$/, [], /* @__PURE__ */ Object.create(null)];\nvar wildcardRegExpCache = /* @__PURE__ */ Object.create(null);\nfunction buildWildcardRegExp(path) {\n return wildcardRegExpCache[path] ??= new RegExp(\n path === \"*\" ? \"\" : `^${path.replace(\n /\\/\\*$|([.\\\\+*[^\\]$()])/g,\n (_, metaChar) => metaChar ? `\\\\${metaChar}` : \"(?:|/.*)\"\n )}$`\n );\n}\nfunction clearWildcardRegExpCache() {\n wildcardRegExpCache = /* @__PURE__ */ Object.create(null);\n}\nfunction buildMatcherFromPreprocessedRoutes(routes) {\n const trie = new Trie();\n const handlerData = [];\n if (routes.length === 0) {\n return nullMatcher;\n }\n const routesWithStaticPathFlag = routes.map(\n (route) => [!/\\*|\\/:/.test(route[0]), ...route]\n ).sort(\n ([isStaticA, pathA], [isStaticB, pathB]) => isStaticA ? 1 : isStaticB ? -1 : pathA.length - pathB.length\n );\n const staticMap = /* @__PURE__ */ Object.create(null);\n for (let i = 0, j = -1, len = routesWithStaticPathFlag.length; i < len; i++) {\n const [pathErrorCheckOnly, path, handlers] = routesWithStaticPathFlag[i];\n if (pathErrorCheckOnly) {\n staticMap[path] = [handlers.map(([h]) => [h, /* @__PURE__ */ Object.create(null)]), emptyParam];\n } else {\n j++;\n }\n let paramAssoc;\n try {\n paramAssoc = trie.insert(path, j, pathErrorCheckOnly);\n } catch (e) {\n throw e === PATH_ERROR ? new UnsupportedPathError(path) : e;\n }\n if (pathErrorCheckOnly) {\n continue;\n }\n handlerData[j] = handlers.map(([h, paramCount]) => {\n const paramIndexMap = /* @__PURE__ */ Object.create(null);\n paramCount -= 1;\n for (; paramCount >= 0; paramCount--) {\n const [key, value] = paramAssoc[paramCount];\n paramIndexMap[key] = value;\n }\n return [h, paramIndexMap];\n });\n }\n const [regexp, indexReplacementMap, paramReplacementMap] = trie.buildRegExp();\n for (let i = 0, len = handlerData.length; i < len; i++) {\n for (let j = 0, len2 = handlerData[i].length; j < len2; j++) {\n const map = handlerData[i][j]?.[1];\n if (!map) {\n continue;\n }\n const keys = Object.keys(map);\n for (let k = 0, len3 = keys.length; k < len3; k++) {\n map[keys[k]] = paramReplacementMap[map[keys[k]]];\n }\n }\n }\n const handlerMap = [];\n for (const i in indexReplacementMap) {\n handlerMap[i] = handlerData[indexReplacementMap[i]];\n }\n return [regexp, handlerMap, staticMap];\n}\nfunction findMiddleware(middleware, path) {\n if (!middleware) {\n return void 0;\n }\n for (const k of Object.keys(middleware).sort((a, b) => b.length - a.length)) {\n if (buildWildcardRegExp(k).test(path)) {\n return [...middleware[k]];\n }\n }\n return void 0;\n}\nvar RegExpRouter = class {\n name = \"RegExpRouter\";\n #middleware;\n #routes;\n constructor() {\n this.#middleware = { [METHOD_NAME_ALL]: /* @__PURE__ */ Object.create(null) };\n this.#routes = { [METHOD_NAME_ALL]: /* @__PURE__ */ Object.create(null) };\n }\n add(method, path, handler) {\n const middleware = this.#middleware;\n const routes = this.#routes;\n if (!middleware || !routes) {\n throw new Error(MESSAGE_MATCHER_IS_ALREADY_BUILT);\n }\n if (!middleware[method]) {\n ;\n [middleware, routes].forEach((handlerMap) => {\n handlerMap[method] = /* @__PURE__ */ Object.create(null);\n Object.keys(handlerMap[METHOD_NAME_ALL]).forEach((p) => {\n handlerMap[method][p] = [...handlerMap[METHOD_NAME_ALL][p]];\n });\n });\n }\n if (path === \"/*\") {\n path = \"*\";\n }\n const paramCount = (path.match(/\\/:/g) || []).length;\n if (/\\*$/.test(path)) {\n const re = buildWildcardRegExp(path);\n if (method === METHOD_NAME_ALL) {\n Object.keys(middleware).forEach((m) => {\n middleware[m][path] ||= findMiddleware(middleware[m], path) || findMiddleware(middleware[METHOD_NAME_ALL], path) || [];\n });\n } else {\n middleware[method][path] ||= findMiddleware(middleware[method], path) || findMiddleware(middleware[METHOD_NAME_ALL], path) || [];\n }\n Object.keys(middleware).forEach((m) => {\n if (method === METHOD_NAME_ALL || method === m) {\n Object.keys(middleware[m]).forEach((p) => {\n re.test(p) && middleware[m][p].push([handler, paramCount]);\n });\n }\n });\n Object.keys(routes).forEach((m) => {\n if (method === METHOD_NAME_ALL || method === m) {\n Object.keys(routes[m]).forEach(\n (p) => re.test(p) && routes[m][p].push([handler, paramCount])\n );\n }\n });\n return;\n }\n const paths = checkOptionalParameter(path) || [path];\n for (let i = 0, len = paths.length; i < len; i++) {\n const path2 = paths[i];\n Object.keys(routes).forEach((m) => {\n if (method === METHOD_NAME_ALL || method === m) {\n routes[m][path2] ||= [\n ...findMiddleware(middleware[m], path2) || findMiddleware(middleware[METHOD_NAME_ALL], path2) || []\n ];\n routes[m][path2].push([handler, paramCount - len + i + 1]);\n }\n });\n }\n }\n match = match;\n buildAllMatchers() {\n const matchers = /* @__PURE__ */ Object.create(null);\n Object.keys(this.#routes).concat(Object.keys(this.#middleware)).forEach((method) => {\n matchers[method] ||= this.#buildMatcher(method);\n });\n this.#middleware = this.#routes = void 0;\n clearWildcardRegExpCache();\n return matchers;\n }\n #buildMatcher(method) {\n const routes = [];\n let hasOwnRoute = method === METHOD_NAME_ALL;\n [this.#middleware, this.#routes].forEach((r) => {\n const ownRoute = r[method] ? Object.keys(r[method]).map((path) => [path, r[method][path]]) : [];\n if (ownRoute.length !== 0) {\n hasOwnRoute ||= true;\n routes.push(...ownRoute);\n } else if (method !== METHOD_NAME_ALL) {\n routes.push(\n ...Object.keys(r[METHOD_NAME_ALL]).map((path) => [path, r[METHOD_NAME_ALL][path]])\n );\n }\n });\n if (!hasOwnRoute) {\n return null;\n } else {\n return buildMatcherFromPreprocessedRoutes(routes);\n }\n }\n};\nexport {\n RegExpRouter\n};\n","// src/router/reg-exp-router/prepared-router.ts\nimport { METHOD_NAME_ALL } from \"../../router.js\";\nimport { match, emptyParam } from \"./matcher.js\";\nimport { RegExpRouter } from \"./router.js\";\nvar PreparedRegExpRouter = class {\n name = \"PreparedRegExpRouter\";\n #matchers;\n #relocateMap;\n constructor(matchers, relocateMap) {\n this.#matchers = matchers;\n this.#relocateMap = relocateMap;\n }\n #addWildcard(method, handlerData) {\n const matcher = this.#matchers[method];\n matcher[1].forEach((list) => list && list.push(handlerData));\n Object.values(matcher[2]).forEach((list) => list[0].push(handlerData));\n }\n #addPath(method, path, handler, indexes, map) {\n const matcher = this.#matchers[method];\n if (!map) {\n matcher[2][path][0].push([handler, {}]);\n } else {\n indexes.forEach((index) => {\n if (typeof index === \"number\") {\n matcher[1][index].push([handler, map]);\n } else {\n ;\n matcher[2][index || path][0].push([handler, map]);\n }\n });\n }\n }\n add(method, path, handler) {\n if (!this.#matchers[method]) {\n const all = this.#matchers[METHOD_NAME_ALL];\n const staticMap = {};\n for (const key in all[2]) {\n staticMap[key] = [all[2][key][0].slice(), emptyParam];\n }\n this.#matchers[method] = [\n all[0],\n all[1].map((list) => Array.isArray(list) ? list.slice() : 0),\n staticMap\n ];\n }\n if (path === \"/*\" || path === \"*\") {\n const handlerData = [handler, {}];\n if (method === METHOD_NAME_ALL) {\n for (const m in this.#matchers) {\n this.#addWildcard(m, handlerData);\n }\n } else {\n this.#addWildcard(method, handlerData);\n }\n return;\n }\n const data = this.#relocateMap[path];\n if (!data) {\n throw new Error(`Path ${path} is not registered`);\n }\n for (const [indexes, map] of data) {\n if (method === METHOD_NAME_ALL) {\n for (const m in this.#matchers) {\n this.#addPath(m, path, handler, indexes, map);\n }\n } else {\n this.#addPath(method, path, handler, indexes, map);\n }\n }\n }\n buildAllMatchers() {\n return this.#matchers;\n }\n match = match;\n};\nvar buildInitParams = ({ paths }) => {\n const RegExpRouterWithMatcherExport = class extends RegExpRouter {\n buildAndExportAllMatchers() {\n return this.buildAllMatchers();\n }\n };\n const router = new RegExpRouterWithMatcherExport();\n for (const path of paths) {\n router.add(METHOD_NAME_ALL, path, path);\n }\n const matchers = router.buildAndExportAllMatchers();\n const all = matchers[METHOD_NAME_ALL];\n const relocateMap = {};\n for (const path of paths) {\n if (path === \"/*\" || path === \"*\") {\n continue;\n }\n all[1].forEach((list, i) => {\n list.forEach(([p, map]) => {\n if (p === path) {\n if (relocateMap[path]) {\n relocateMap[path][0][1] = {\n ...relocateMap[path][0][1],\n ...map\n };\n } else {\n relocateMap[path] = [[[], map]];\n }\n if (relocateMap[path][0][0].findIndex((j) => j === i) === -1) {\n relocateMap[path][0][0].push(i);\n }\n }\n });\n });\n for (const path2 in all[2]) {\n all[2][path2][0].forEach(([p]) => {\n if (p === path) {\n relocateMap[path] ||= [[[]]];\n const value = path2 === path ? \"\" : path2;\n if (relocateMap[path][0][0].findIndex((v) => v === value) === -1) {\n relocateMap[path][0][0].push(value);\n }\n }\n });\n }\n }\n for (let i = 0, len = all[1].length; i < len; i++) {\n all[1][i] = all[1][i] ? [] : 0;\n }\n for (const path in all[2]) {\n all[2][path][0] = [];\n }\n return [matchers, relocateMap];\n};\nvar serializeInitParams = ([matchers, relocateMap]) => {\n const matchersStr = JSON.stringify(\n matchers,\n (_, value) => value instanceof RegExp ? `##${value.toString()}##` : value\n ).replace(/\"##(.+?)##\"/g, (_, str) => str.replace(/\\\\\\\\/g, \"\\\\\"));\n const relocateMapStr = JSON.stringify(relocateMap);\n return `[${matchersStr},${relocateMapStr}]`;\n};\nexport {\n PreparedRegExpRouter,\n buildInitParams,\n serializeInitParams\n};\n","// src/router/reg-exp-router/index.ts\nimport { RegExpRouter } from \"./router.js\";\nimport { PreparedRegExpRouter, buildInitParams, serializeInitParams } from \"./prepared-router.js\";\nexport {\n PreparedRegExpRouter,\n RegExpRouter,\n buildInitParams,\n serializeInitParams\n};\n","// src/router/smart-router/router.ts\nimport { MESSAGE_MATCHER_IS_ALREADY_BUILT, UnsupportedPathError } from \"../../router.js\";\nvar SmartRouter = class {\n name = \"SmartRouter\";\n #routers = [];\n #routes = [];\n constructor(init) {\n this.#routers = init.routers;\n }\n add(method, path, handler) {\n if (!this.#routes) {\n throw new Error(MESSAGE_MATCHER_IS_ALREADY_BUILT);\n }\n this.#routes.push([method, path, handler]);\n }\n match(method, path) {\n if (!this.#routes) {\n throw new Error(\"Fatal error\");\n }\n const routers = this.#routers;\n const routes = this.#routes;\n const len = routers.length;\n let i = 0;\n let res;\n for (; i < len; i++) {\n const router = routers[i];\n try {\n for (let i2 = 0, len2 = routes.length; i2 < len2; i2++) {\n router.add(...routes[i2]);\n }\n res = router.match(method, path);\n } catch (e) {\n if (e instanceof UnsupportedPathError) {\n continue;\n }\n throw e;\n }\n this.match = router.match.bind(router);\n this.#routers = [router];\n this.#routes = void 0;\n break;\n }\n if (i === len) {\n throw new Error(\"Fatal error\");\n }\n this.name = `SmartRouter + ${this.activeRouter.name}`;\n return res;\n }\n get activeRouter() {\n if (this.#routes || this.#routers.length !== 1) {\n throw new Error(\"No active router has been determined yet.\");\n }\n return this.#routers[0];\n }\n};\nexport {\n SmartRouter\n};\n","// src/router/smart-router/index.ts\nimport { SmartRouter } from \"./router.js\";\nexport {\n SmartRouter\n};\n","// src/router/trie-router/node.ts\nimport { METHOD_NAME_ALL } from \"../../router.js\";\nimport { getPattern, splitPath, splitRoutingPath } from \"../../utils/url.js\";\nvar emptyParams = /* @__PURE__ */ Object.create(null);\nvar hasChildren = (children) => {\n for (const _ in children) {\n return true;\n }\n return false;\n};\nvar Node = class _Node {\n #methods;\n #children;\n #patterns;\n #order = 0;\n #params = emptyParams;\n constructor(method, handler, children) {\n this.#children = children || /* @__PURE__ */ Object.create(null);\n this.#methods = [];\n if (method && handler) {\n const m = /* @__PURE__ */ Object.create(null);\n m[method] = { handler, possibleKeys: [], score: 0 };\n this.#methods = [m];\n }\n this.#patterns = [];\n }\n insert(method, path, handler) {\n this.#order = ++this.#order;\n let curNode = this;\n const parts = splitRoutingPath(path);\n const possibleKeys = [];\n for (let i = 0, len = parts.length; i < len; i++) {\n const p = parts[i];\n const nextP = parts[i + 1];\n const pattern = getPattern(p, nextP);\n const key = Array.isArray(pattern) ? pattern[0] : p;\n if (key in curNode.#children) {\n curNode = curNode.#children[key];\n if (pattern) {\n possibleKeys.push(pattern[1]);\n }\n continue;\n }\n curNode.#children[key] = new _Node();\n if (pattern) {\n curNode.#patterns.push(pattern);\n possibleKeys.push(pattern[1]);\n }\n curNode = curNode.#children[key];\n }\n curNode.#methods.push({\n [method]: {\n handler,\n possibleKeys: possibleKeys.filter((v, i, a) => a.indexOf(v) === i),\n score: this.#order\n }\n });\n return curNode;\n }\n #pushHandlerSets(handlerSets, node, method, nodeParams, params) {\n for (let i = 0, len = node.#methods.length; i < len; i++) {\n const m = node.#methods[i];\n const handlerSet = m[method] || m[METHOD_NAME_ALL];\n const processedSet = {};\n if (handlerSet !== void 0) {\n handlerSet.params = /* @__PURE__ */ Object.create(null);\n handlerSets.push(handlerSet);\n if (nodeParams !== emptyParams || params && params !== emptyParams) {\n for (let i2 = 0, len2 = handlerSet.possibleKeys.length; i2 < len2; i2++) {\n const key = handlerSet.possibleKeys[i2];\n const processed = processedSet[handlerSet.score];\n handlerSet.params[key] = params?.[key] && !processed ? params[key] : nodeParams[key] ?? params?.[key];\n processedSet[handlerSet.score] = true;\n }\n }\n }\n }\n }\n search(method, path) {\n const handlerSets = [];\n this.#params = emptyParams;\n const curNode = this;\n let curNodes = [curNode];\n const parts = splitPath(path);\n const curNodesQueue = [];\n const len = parts.length;\n let partOffsets = null;\n for (let i = 0; i < len; i++) {\n const part = parts[i];\n const isLast = i === len - 1;\n const tempNodes = [];\n for (let j = 0, len2 = curNodes.length; j < len2; j++) {\n const node = curNodes[j];\n const nextNode = node.#children[part];\n if (nextNode) {\n nextNode.#params = node.#params;\n if (isLast) {\n if (nextNode.#children[\"*\"]) {\n this.#pushHandlerSets(handlerSets, nextNode.#children[\"*\"], method, node.#params);\n }\n this.#pushHandlerSets(handlerSets, nextNode, method, node.#params);\n } else {\n tempNodes.push(nextNode);\n }\n }\n for (let k = 0, len3 = node.#patterns.length; k < len3; k++) {\n const pattern = node.#patterns[k];\n const params = node.#params === emptyParams ? {} : { ...node.#params };\n if (pattern === \"*\") {\n const astNode = node.#children[\"*\"];\n if (astNode) {\n this.#pushHandlerSets(handlerSets, astNode, method, node.#params);\n astNode.#params = params;\n tempNodes.push(astNode);\n }\n continue;\n }\n const [key, name, matcher] = pattern;\n if (!part && !(matcher instanceof RegExp)) {\n continue;\n }\n const child = node.#children[key];\n if (matcher instanceof RegExp) {\n if (partOffsets === null) {\n partOffsets = new Array(len);\n let offset = path[0] === \"/\" ? 1 : 0;\n for (let p = 0; p < len; p++) {\n partOffsets[p] = offset;\n offset += parts[p].length + 1;\n }\n }\n const restPathString = path.substring(partOffsets[i]);\n const m = matcher.exec(restPathString);\n if (m) {\n params[name] = m[0];\n this.#pushHandlerSets(handlerSets, child, method, node.#params, params);\n if (hasChildren(child.#children)) {\n child.#params = params;\n const componentCount = m[0].match(/\\//)?.length ?? 0;\n const targetCurNodes = curNodesQueue[componentCount] ||= [];\n targetCurNodes.push(child);\n }\n continue;\n }\n }\n if (matcher === true || matcher.test(part)) {\n params[name] = part;\n if (isLast) {\n this.#pushHandlerSets(handlerSets, child, method, params, node.#params);\n if (child.#children[\"*\"]) {\n this.#pushHandlerSets(\n handlerSets,\n child.#children[\"*\"],\n method,\n params,\n node.#params\n );\n }\n } else {\n child.#params = params;\n tempNodes.push(child);\n }\n }\n }\n }\n const shifted = curNodesQueue.shift();\n curNodes = shifted ? tempNodes.concat(shifted) : tempNodes;\n }\n if (handlerSets.length > 1) {\n handlerSets.sort((a, b) => {\n return a.score - b.score;\n });\n }\n return [handlerSets.map(({ handler, params }) => [handler, params])];\n }\n};\nexport {\n Node\n};\n","// src/router/trie-router/router.ts\nimport { checkOptionalParameter } from \"../../utils/url.js\";\nimport { Node } from \"./node.js\";\nvar TrieRouter = class {\n name = \"TrieRouter\";\n #node;\n constructor() {\n this.#node = new Node();\n }\n add(method, path, handler) {\n const results = checkOptionalParameter(path);\n if (results) {\n for (let i = 0, len = results.length; i < len; i++) {\n this.#node.insert(method, results[i], handler);\n }\n return;\n }\n this.#node.insert(method, path, handler);\n }\n match(method, path) {\n return this.#node.search(method, path);\n }\n};\nexport {\n TrieRouter\n};\n","// src/router/trie-router/index.ts\nimport { TrieRouter } from \"./router.js\";\nexport {\n TrieRouter\n};\n","// src/hono.ts\nimport { HonoBase } from \"./hono-base.js\";\nimport { RegExpRouter } from \"./router/reg-exp-router/index.js\";\nimport { SmartRouter } from \"./router/smart-router/index.js\";\nimport { TrieRouter } from \"./router/trie-router/index.js\";\nvar Hono = class extends HonoBase {\n /**\n * Creates an instance of the Hono class.\n *\n * @param options - Optional configuration options for the Hono instance.\n */\n constructor(options = {}) {\n super(options);\n this.router = options.router ?? new SmartRouter({\n routers: [new RegExpRouter(), new TrieRouter()]\n });\n }\n};\nexport {\n Hono\n};\n","// src/index.ts\nimport { Hono } from \"./hono.js\";\nimport { Context } from \"./context.js\";\nexport {\n Context,\n Hono\n};\n","// src/middleware/cors/index.ts\nvar cors = (options) => {\n const opts = {\n origin: \"*\",\n allowMethods: [\"GET\", \"HEAD\", \"PUT\", \"POST\", \"DELETE\", \"PATCH\"],\n allowHeaders: [],\n exposeHeaders: [],\n ...options\n };\n const findAllowOrigin = ((optsOrigin) => {\n if (typeof optsOrigin === \"string\") {\n if (optsOrigin === \"*\") {\n if (opts.credentials) {\n return (origin) => origin || null;\n }\n return () => optsOrigin;\n } else {\n return (origin) => optsOrigin === origin ? origin : null;\n }\n } else if (typeof optsOrigin === \"function\") {\n return optsOrigin;\n } else {\n return (origin) => optsOrigin.includes(origin) ? origin : null;\n }\n })(opts.origin);\n const findAllowMethods = ((optsAllowMethods) => {\n if (typeof optsAllowMethods === \"function\") {\n return optsAllowMethods;\n } else if (Array.isArray(optsAllowMethods)) {\n return () => optsAllowMethods;\n } else {\n return () => [];\n }\n })(opts.allowMethods);\n return async function cors2(c, next) {\n function set(key, value) {\n c.res.headers.set(key, value);\n }\n const allowOrigin = await findAllowOrigin(c.req.header(\"origin\") || \"\", c);\n if (allowOrigin) {\n set(\"Access-Control-Allow-Origin\", allowOrigin);\n }\n if (opts.credentials) {\n set(\"Access-Control-Allow-Credentials\", \"true\");\n }\n if (opts.exposeHeaders?.length) {\n set(\"Access-Control-Expose-Headers\", opts.exposeHeaders.join(\",\"));\n }\n if (c.req.method === \"OPTIONS\") {\n if (opts.origin !== \"*\" || opts.credentials) {\n set(\"Vary\", \"Origin\");\n }\n if (opts.maxAge != null) {\n set(\"Access-Control-Max-Age\", opts.maxAge.toString());\n }\n const allowMethods = await findAllowMethods(c.req.header(\"origin\") || \"\", c);\n if (allowMethods.length) {\n set(\"Access-Control-Allow-Methods\", allowMethods.join(\",\"));\n }\n let headers = opts.allowHeaders;\n if (!headers?.length) {\n const requestHeaders = c.req.header(\"Access-Control-Request-Headers\");\n if (requestHeaders) {\n headers = requestHeaders.split(/\\s*,\\s*/);\n }\n }\n if (headers?.length) {\n set(\"Access-Control-Allow-Headers\", headers.join(\",\"));\n c.res.headers.append(\"Vary\", \"Access-Control-Request-Headers\");\n }\n c.res.headers.delete(\"Content-Length\");\n c.res.headers.delete(\"Content-Type\");\n return new Response(null, {\n headers: c.res.headers,\n status: 204,\n statusText: \"No Content\"\n });\n }\n await next();\n if (opts.origin !== \"*\" || opts.credentials) {\n c.header(\"Vary\", \"Origin\", { append: true });\n }\n };\n};\nexport {\n cors\n};\n","// src/server.ts\nimport { createServer as createServerHTTP } from \"http\";\n\n// src/listener.ts\nimport { Http2ServerRequest as Http2ServerRequest2, constants as h2constants } from \"http2\";\n\n// src/request.ts\nimport { Http2ServerRequest } from \"http2\";\nimport { Readable } from \"stream\";\nvar RequestError = class extends Error {\n constructor(message, options) {\n super(message, options);\n this.name = \"RequestError\";\n }\n};\nvar toRequestError = (e) => {\n if (e instanceof RequestError) {\n return e;\n }\n return new RequestError(e.message, { cause: e });\n};\nvar GlobalRequest = global.Request;\nvar Request = class extends GlobalRequest {\n constructor(input, options) {\n if (typeof input === \"object\" && getRequestCache in input) {\n input = input[getRequestCache]();\n }\n if (typeof options?.body?.getReader !== \"undefined\") {\n ;\n options.duplex ??= \"half\";\n }\n super(input, options);\n }\n};\nvar newHeadersFromIncoming = (incoming) => {\n const headerRecord = [];\n const rawHeaders = incoming.rawHeaders;\n for (let i = 0; i < rawHeaders.length; i += 2) {\n const { [i]: key, [i + 1]: value } = rawHeaders;\n if (key.charCodeAt(0) !== /*:*/\n 58) {\n headerRecord.push([key, value]);\n }\n }\n return new Headers(headerRecord);\n};\nvar wrapBodyStream = Symbol(\"wrapBodyStream\");\nvar newRequestFromIncoming = (method, url, headers, incoming, abortController) => {\n const init = {\n method,\n headers,\n signal: abortController.signal\n };\n if (method === \"TRACE\") {\n init.method = \"GET\";\n const req = new Request(url, init);\n Object.defineProperty(req, \"method\", {\n get() {\n return \"TRACE\";\n }\n });\n return req;\n }\n if (!(method === \"GET\" || method === \"HEAD\")) {\n if (\"rawBody\" in incoming && incoming.rawBody instanceof Buffer) {\n init.body = new ReadableStream({\n start(controller) {\n controller.enqueue(incoming.rawBody);\n controller.close();\n }\n });\n } else if (incoming[wrapBodyStream]) {\n let reader;\n init.body = new ReadableStream({\n async pull(controller) {\n try {\n reader ||= Readable.toWeb(incoming).getReader();\n const { done, value } = await reader.read();\n if (done) {\n controller.close();\n } else {\n controller.enqueue(value);\n }\n } catch (error) {\n controller.error(error);\n }\n }\n });\n } else {\n init.body = Readable.toWeb(incoming);\n }\n }\n return new Request(url, init);\n};\nvar getRequestCache = Symbol(\"getRequestCache\");\nvar requestCache = Symbol(\"requestCache\");\nvar incomingKey = Symbol(\"incomingKey\");\nvar urlKey = Symbol(\"urlKey\");\nvar headersKey = Symbol(\"headersKey\");\nvar abortControllerKey = Symbol(\"abortControllerKey\");\nvar getAbortController = Symbol(\"getAbortController\");\nvar requestPrototype = {\n get method() {\n return this[incomingKey].method || \"GET\";\n },\n get url() {\n return this[urlKey];\n },\n get headers() {\n return this[headersKey] ||= newHeadersFromIncoming(this[incomingKey]);\n },\n [getAbortController]() {\n this[getRequestCache]();\n return this[abortControllerKey];\n },\n [getRequestCache]() {\n this[abortControllerKey] ||= new AbortController();\n return this[requestCache] ||= newRequestFromIncoming(\n this.method,\n this[urlKey],\n this.headers,\n this[incomingKey],\n this[abortControllerKey]\n );\n }\n};\n[\n \"body\",\n \"bodyUsed\",\n \"cache\",\n \"credentials\",\n \"destination\",\n \"integrity\",\n \"mode\",\n \"redirect\",\n \"referrer\",\n \"referrerPolicy\",\n \"signal\",\n \"keepalive\"\n].forEach((k) => {\n Object.defineProperty(requestPrototype, k, {\n get() {\n return this[getRequestCache]()[k];\n }\n });\n});\n[\"arrayBuffer\", \"blob\", \"clone\", \"formData\", \"json\", \"text\"].forEach((k) => {\n Object.defineProperty(requestPrototype, k, {\n value: function() {\n return this[getRequestCache]()[k]();\n }\n });\n});\nObject.defineProperty(requestPrototype, Symbol.for(\"nodejs.util.inspect.custom\"), {\n value: function(depth, options, inspectFn) {\n const props = {\n method: this.method,\n url: this.url,\n headers: this.headers,\n nativeRequest: this[requestCache]\n };\n return `Request (lightweight) ${inspectFn(props, { ...options, depth: depth == null ? null : depth - 1 })}`;\n }\n});\nObject.setPrototypeOf(requestPrototype, Request.prototype);\nvar newRequest = (incoming, defaultHostname) => {\n const req = Object.create(requestPrototype);\n req[incomingKey] = incoming;\n const incomingUrl = incoming.url || \"\";\n if (incomingUrl[0] !== \"/\" && // short-circuit for performance. most requests are relative URL.\n (incomingUrl.startsWith(\"http://\") || incomingUrl.startsWith(\"https://\"))) {\n if (incoming instanceof Http2ServerRequest) {\n throw new RequestError(\"Absolute URL for :path is not allowed in HTTP/2\");\n }\n try {\n const url2 = new URL(incomingUrl);\n req[urlKey] = url2.href;\n } catch (e) {\n throw new RequestError(\"Invalid absolute URL\", { cause: e });\n }\n return req;\n }\n const host = (incoming instanceof Http2ServerRequest ? incoming.authority : incoming.headers.host) || defaultHostname;\n if (!host) {\n throw new RequestError(\"Missing host header\");\n }\n let scheme;\n if (incoming instanceof Http2ServerRequest) {\n scheme = incoming.scheme;\n if (!(scheme === \"http\" || scheme === \"https\")) {\n throw new RequestError(\"Unsupported scheme\");\n }\n } else {\n scheme = incoming.socket && incoming.socket.encrypted ? \"https\" : \"http\";\n }\n const url = new URL(`${scheme}://${host}${incomingUrl}`);\n if (url.hostname.length !== host.length && url.hostname !== host.replace(/:\\d+$/, \"\")) {\n throw new RequestError(\"Invalid host header\");\n }\n req[urlKey] = url.href;\n return req;\n};\n\n// src/response.ts\nvar responseCache = Symbol(\"responseCache\");\nvar getResponseCache = Symbol(\"getResponseCache\");\nvar cacheKey = Symbol(\"cache\");\nvar GlobalResponse = global.Response;\nvar Response2 = class _Response {\n #body;\n #init;\n [getResponseCache]() {\n delete this[cacheKey];\n return this[responseCache] ||= new GlobalResponse(this.#body, this.#init);\n }\n constructor(body, init) {\n let headers;\n this.#body = body;\n if (init instanceof _Response) {\n const cachedGlobalResponse = init[responseCache];\n if (cachedGlobalResponse) {\n this.#init = cachedGlobalResponse;\n this[getResponseCache]();\n return;\n } else {\n this.#init = init.#init;\n headers = new Headers(init.#init.headers);\n }\n } else {\n this.#init = init;\n }\n if (typeof body === \"string\" || typeof body?.getReader !== \"undefined\" || body instanceof Blob || body instanceof Uint8Array) {\n ;\n this[cacheKey] = [init?.status || 200, body, headers || init?.headers];\n }\n }\n get headers() {\n const cache = this[cacheKey];\n if (cache) {\n if (!(cache[2] instanceof Headers)) {\n cache[2] = new Headers(\n cache[2] || { \"content-type\": \"text/plain; charset=UTF-8\" }\n );\n }\n return cache[2];\n }\n return this[getResponseCache]().headers;\n }\n get status() {\n return this[cacheKey]?.[0] ?? this[getResponseCache]().status;\n }\n get ok() {\n const status = this.status;\n return status >= 200 && status < 300;\n }\n};\n[\"body\", \"bodyUsed\", \"redirected\", \"statusText\", \"trailers\", \"type\", \"url\"].forEach((k) => {\n Object.defineProperty(Response2.prototype, k, {\n get() {\n return this[getResponseCache]()[k];\n }\n });\n});\n[\"arrayBuffer\", \"blob\", \"clone\", \"formData\", \"json\", \"text\"].forEach((k) => {\n Object.defineProperty(Response2.prototype, k, {\n value: function() {\n return this[getResponseCache]()[k]();\n }\n });\n});\nObject.defineProperty(Response2.prototype, Symbol.for(\"nodejs.util.inspect.custom\"), {\n value: function(depth, options, inspectFn) {\n const props = {\n status: this.status,\n headers: this.headers,\n ok: this.ok,\n nativeResponse: this[responseCache]\n };\n return `Response (lightweight) ${inspectFn(props, { ...options, depth: depth == null ? null : depth - 1 })}`;\n }\n});\nObject.setPrototypeOf(Response2, GlobalResponse);\nObject.setPrototypeOf(Response2.prototype, GlobalResponse.prototype);\n\n// src/utils.ts\nasync function readWithoutBlocking(readPromise) {\n return Promise.race([readPromise, Promise.resolve().then(() => Promise.resolve(void 0))]);\n}\nfunction writeFromReadableStreamDefaultReader(reader, writable, currentReadPromise) {\n const cancel = (error) => {\n reader.cancel(error).catch(() => {\n });\n };\n writable.on(\"close\", cancel);\n writable.on(\"error\", cancel);\n (currentReadPromise ?? reader.read()).then(flow, handleStreamError);\n return reader.closed.finally(() => {\n writable.off(\"close\", cancel);\n writable.off(\"error\", cancel);\n });\n function handleStreamError(error) {\n if (error) {\n writable.destroy(error);\n }\n }\n function onDrain() {\n reader.read().then(flow, handleStreamError);\n }\n function flow({ done, value }) {\n try {\n if (done) {\n writable.end();\n } else if (!writable.write(value)) {\n writable.once(\"drain\", onDrain);\n } else {\n return reader.read().then(flow, handleStreamError);\n }\n } catch (e) {\n handleStreamError(e);\n }\n }\n}\nfunction writeFromReadableStream(stream, writable) {\n if (stream.locked) {\n throw new TypeError(\"ReadableStream is locked.\");\n } else if (writable.destroyed) {\n return;\n }\n return writeFromReadableStreamDefaultReader(stream.getReader(), writable);\n}\nvar buildOutgoingHttpHeaders = (headers) => {\n const res = {};\n if (!(headers instanceof Headers)) {\n headers = new Headers(headers ?? void 0);\n }\n const cookies = [];\n for (const [k, v] of headers) {\n if (k === \"set-cookie\") {\n cookies.push(v);\n } else {\n res[k] = v;\n }\n }\n if (cookies.length > 0) {\n res[\"set-cookie\"] = cookies;\n }\n res[\"content-type\"] ??= \"text/plain; charset=UTF-8\";\n return res;\n};\n\n// src/utils/response/constants.ts\nvar X_ALREADY_SENT = \"x-hono-already-sent\";\n\n// src/globals.ts\nimport crypto from \"crypto\";\nif (typeof global.crypto === \"undefined\") {\n global.crypto = crypto;\n}\n\n// src/listener.ts\nvar outgoingEnded = Symbol(\"outgoingEnded\");\nvar incomingDraining = Symbol(\"incomingDraining\");\nvar DRAIN_TIMEOUT_MS = 500;\nvar MAX_DRAIN_BYTES = 64 * 1024 * 1024;\nvar drainIncoming = (incoming) => {\n const incomingWithDrainState = incoming;\n if (incoming.destroyed || incomingWithDrainState[incomingDraining]) {\n return;\n }\n incomingWithDrainState[incomingDraining] = true;\n if (incoming instanceof Http2ServerRequest2) {\n try {\n ;\n incoming.stream?.close?.(h2constants.NGHTTP2_NO_ERROR);\n } catch {\n }\n return;\n }\n let bytesRead = 0;\n const cleanup = () => {\n clearTimeout(timer);\n incoming.off(\"data\", onData);\n incoming.off(\"end\", cleanup);\n incoming.off(\"error\", cleanup);\n };\n const forceClose = () => {\n cleanup();\n const socket = incoming.socket;\n if (socket && !socket.destroyed) {\n socket.destroySoon();\n }\n };\n const timer = setTimeout(forceClose, DRAIN_TIMEOUT_MS);\n timer.unref?.();\n const onData = (chunk) => {\n bytesRead += chunk.length;\n if (bytesRead > MAX_DRAIN_BYTES) {\n forceClose();\n }\n };\n incoming.on(\"data\", onData);\n incoming.on(\"end\", cleanup);\n incoming.on(\"error\", cleanup);\n incoming.resume();\n};\nvar handleRequestError = () => new Response(null, {\n status: 400\n});\nvar handleFetchError = (e) => new Response(null, {\n status: e instanceof Error && (e.name === \"TimeoutError\" || e.constructor.name === \"TimeoutError\") ? 504 : 500\n});\nvar handleResponseError = (e, outgoing) => {\n const err = e instanceof Error ? e : new Error(\"unknown error\", { cause: e });\n if (err.code === \"ERR_STREAM_PREMATURE_CLOSE\") {\n console.info(\"The user aborted a request.\");\n } else {\n console.error(e);\n if (!outgoing.headersSent) {\n outgoing.writeHead(500, { \"Content-Type\": \"text/plain\" });\n }\n outgoing.end(`Error: ${err.message}`);\n outgoing.destroy(err);\n }\n};\nvar flushHeaders = (outgoing) => {\n if (\"flushHeaders\" in outgoing && outgoing.writable) {\n outgoing.flushHeaders();\n }\n};\nvar responseViaCache = async (res, outgoing) => {\n let [status, body, header] = res[cacheKey];\n let hasContentLength = false;\n if (!header) {\n header = { \"content-type\": \"text/plain; charset=UTF-8\" };\n } else if (header instanceof Headers) {\n hasContentLength = header.has(\"content-length\");\n header = buildOutgoingHttpHeaders(header);\n } else if (Array.isArray(header)) {\n const headerObj = new Headers(header);\n hasContentLength = headerObj.has(\"content-length\");\n header = buildOutgoingHttpHeaders(headerObj);\n } else {\n for (const key in header) {\n if (key.length === 14 && key.toLowerCase() === \"content-length\") {\n hasContentLength = true;\n break;\n }\n }\n }\n if (!hasContentLength) {\n if (typeof body === \"string\") {\n header[\"Content-Length\"] = Buffer.byteLength(body);\n } else if (body instanceof Uint8Array) {\n header[\"Content-Length\"] = body.byteLength;\n } else if (body instanceof Blob) {\n header[\"Content-Length\"] = body.size;\n }\n }\n outgoing.writeHead(status, header);\n if (typeof body === \"string\" || body instanceof Uint8Array) {\n outgoing.end(body);\n } else if (body instanceof Blob) {\n outgoing.end(new Uint8Array(await body.arrayBuffer()));\n } else {\n flushHeaders(outgoing);\n await writeFromReadableStream(body, outgoing)?.catch(\n (e) => handleResponseError(e, outgoing)\n );\n }\n ;\n outgoing[outgoingEnded]?.();\n};\nvar isPromise = (res) => typeof res.then === \"function\";\nvar responseViaResponseObject = async (res, outgoing, options = {}) => {\n if (isPromise(res)) {\n if (options.errorHandler) {\n try {\n res = await res;\n } catch (err) {\n const errRes = await options.errorHandler(err);\n if (!errRes) {\n return;\n }\n res = errRes;\n }\n } else {\n res = await res.catch(handleFetchError);\n }\n }\n if (cacheKey in res) {\n return responseViaCache(res, outgoing);\n }\n const resHeaderRecord = buildOutgoingHttpHeaders(res.headers);\n if (res.body) {\n const reader = res.body.getReader();\n const values = [];\n let done = false;\n let currentReadPromise = void 0;\n if (resHeaderRecord[\"transfer-encoding\"] !== \"chunked\") {\n let maxReadCount = 2;\n for (let i = 0; i < maxReadCount; i++) {\n currentReadPromise ||= reader.read();\n const chunk = await readWithoutBlocking(currentReadPromise).catch((e) => {\n console.error(e);\n done = true;\n });\n if (!chunk) {\n if (i === 1) {\n await new Promise((resolve) => setTimeout(resolve));\n maxReadCount = 3;\n continue;\n }\n break;\n }\n currentReadPromise = void 0;\n if (chunk.value) {\n values.push(chunk.value);\n }\n if (chunk.done) {\n done = true;\n break;\n }\n }\n if (done && !(\"content-length\" in resHeaderRecord)) {\n resHeaderRecord[\"content-length\"] = values.reduce((acc, value) => acc + value.length, 0);\n }\n }\n outgoing.writeHead(res.status, resHeaderRecord);\n values.forEach((value) => {\n ;\n outgoing.write(value);\n });\n if (done) {\n outgoing.end();\n } else {\n if (values.length === 0) {\n flushHeaders(outgoing);\n }\n await writeFromReadableStreamDefaultReader(reader, outgoing, currentReadPromise);\n }\n } else if (resHeaderRecord[X_ALREADY_SENT]) {\n } else {\n outgoing.writeHead(res.status, resHeaderRecord);\n outgoing.end();\n }\n ;\n outgoing[outgoingEnded]?.();\n};\nvar getRequestListener = (fetchCallback, options = {}) => {\n const autoCleanupIncoming = options.autoCleanupIncoming ?? true;\n if (options.overrideGlobalObjects !== false && global.Request !== Request) {\n Object.defineProperty(global, \"Request\", {\n value: Request\n });\n Object.defineProperty(global, \"Response\", {\n value: Response2\n });\n }\n return async (incoming, outgoing) => {\n let res, req;\n try {\n req = newRequest(incoming, options.hostname);\n let incomingEnded = !autoCleanupIncoming || incoming.method === \"GET\" || incoming.method === \"HEAD\";\n if (!incomingEnded) {\n ;\n incoming[wrapBodyStream] = true;\n incoming.on(\"end\", () => {\n incomingEnded = true;\n });\n if (incoming instanceof Http2ServerRequest2) {\n ;\n outgoing[outgoingEnded] = () => {\n if (!incomingEnded) {\n setTimeout(() => {\n if (!incomingEnded) {\n setTimeout(() => {\n drainIncoming(incoming);\n });\n }\n });\n }\n };\n }\n outgoing.on(\"finish\", () => {\n if (!incomingEnded) {\n drainIncoming(incoming);\n }\n });\n }\n outgoing.on(\"close\", () => {\n const abortController = req[abortControllerKey];\n if (abortController) {\n if (incoming.errored) {\n req[abortControllerKey].abort(incoming.errored.toString());\n } else if (!outgoing.writableFinished) {\n req[abortControllerKey].abort(\"Client connection prematurely closed.\");\n }\n }\n if (!incomingEnded) {\n setTimeout(() => {\n if (!incomingEnded) {\n setTimeout(() => {\n drainIncoming(incoming);\n });\n }\n });\n }\n });\n res = fetchCallback(req, { incoming, outgoing });\n if (cacheKey in res) {\n return responseViaCache(res, outgoing);\n }\n } catch (e) {\n if (!res) {\n if (options.errorHandler) {\n res = await options.errorHandler(req ? e : toRequestError(e));\n if (!res) {\n return;\n }\n } else if (!req) {\n res = handleRequestError();\n } else {\n res = handleFetchError(e);\n }\n } else {\n return handleResponseError(e, outgoing);\n }\n }\n try {\n return await responseViaResponseObject(res, outgoing, options);\n } catch (e) {\n return handleResponseError(e, outgoing);\n }\n };\n};\n\n// src/server.ts\nvar createAdaptorServer = (options) => {\n const fetchCallback = options.fetch;\n const requestListener = getRequestListener(fetchCallback, {\n hostname: options.hostname,\n overrideGlobalObjects: options.overrideGlobalObjects,\n autoCleanupIncoming: options.autoCleanupIncoming\n });\n const createServer = options.createServer || createServerHTTP;\n const server = createServer(options.serverOptions || {}, requestListener);\n return server;\n};\nvar serve = (options, listeningListener) => {\n const server = createAdaptorServer(options);\n server.listen(options?.port ?? 3e3, options.hostname, () => {\n const serverInfo = server.address();\n listeningListener && listeningListener(serverInfo);\n });\n return server;\n};\nexport {\n RequestError,\n createAdaptorServer,\n getRequestListener,\n serve\n};\n","// Server\nexport const PARTY_PORT = parseInt(process.env.PARTY_PORT || '8000', 10);\n\n// Agent lifecycle\nexport const HEARTBEAT_TIMEOUT = parseFloat(process.env.HEARTBEAT_TIMEOUT || '30');\nexport const CLEANUP_INTERVAL = parseFloat(process.env.CLEANUP_INTERVAL || '10');\n\n// Peer discovery\nexport const DISCOVERY_INTERVAL = parseFloat(process.env.DISCOVERY_INTERVAL || '20');\nexport const REMOTE_STALE_FACTOR = parseInt(process.env.REMOTE_STALE_FACTOR || '3', 10);\nexport const PROBE_TIMEOUT = parseFloat(process.env.PROBE_TIMEOUT || '5');\n","import type { MessageEnvelope, HistoryEntry } from './models.js';\n\nconst HISTORY_CAP = 200; // max history entries per agent\n\n/** Per-agent message queue stored in memory. */\nexport class MessageQueue {\n private _queues = new Map<string, MessageEnvelope[]>();\n private _history = new Map<string, HistoryEntry[]>();\n\n /** Enqueue a message for agentId. Returns the queue length after enqueue. */\n enqueue(agentId: string, envelope: MessageEnvelope): number {\n let q = this._queues.get(agentId);\n if (!q) {\n q = [];\n this._queues.set(agentId, q);\n }\n q.push(envelope);\n return q.length;\n }\n\n /** Pop up to maxCount messages for agentId. */\n dequeue(agentId: string, maxCount = 50): MessageEnvelope[] {\n const q = this._queues.get(agentId);\n if (!q) return [];\n return q.splice(0, Math.min(maxCount, q.length));\n }\n\n /** Return the number of pending messages for agentId. */\n pendingCount(agentId: string): number {\n return this._queues.get(agentId)?.length ?? 0;\n }\n\n /** Clean up queue when agent is removed. */\n removeAgent(agentId: string): void {\n this._queues.delete(agentId);\n }\n\n /** Record a message in an agent's history. */\n logToHistory(agentId: string, direction: 'sent' | 'received', envelope: MessageEnvelope): void {\n let h = this._history.get(agentId);\n if (!h) {\n h = [];\n this._history.set(agentId, h);\n }\n h.push({\n direction,\n sender_id: envelope.sender_id,\n recipient_id: envelope.recipient_id,\n summary: envelope.summary,\n content: envelope.content,\n timestamp: envelope.timestamp ?? Date.now() / 1000,\n });\n if (h.length > HISTORY_CAP) {\n this._history.set(agentId, h.slice(-HISTORY_CAP));\n }\n }\n\n /** Get recent N history entries for an agent. */\n getHistory(agentId: string, limit = 20): HistoryEntry[] {\n const h = this._history.get(agentId);\n if (!h) return [];\n return h.slice(-limit);\n }\n\n /** Clean up history when agent is removed. */\n removeAgentHistory(agentId: string): void {\n this._history.delete(agentId);\n }\n}\n","/**\r\n * Peer discovery and remote agent cache for cross-host communication.\r\n *\r\n * Pull-based discovery loop:\r\n * 1. Read Tailscale peers via readTailscaleStatus()\r\n * 2. Probe each peer's /proxy/health to identify Party Servers\r\n * 3. Fetch agent lists from healthy peers via /proxy/list_agents\r\n * 4. Maintain a remote agent cache with per-peer reachability tracking\r\n */\r\n\r\nimport type { AgentInfo } from './models.js';\r\nimport { DISCOVERY_INTERVAL, PARTY_PORT, PROBE_TIMEOUT, REMOTE_STALE_FACTOR } from './config.js';\r\nimport { readTailscaleStatus } from '../infra/tailscale.js';\r\n\r\n// ---------------------------------------------------------------------------\r\n// Peer state machine constants\r\n// ---------------------------------------------------------------------------\r\n\r\nconst UNKNOWN = 'UNKNOWN';\r\nconst PARTY_SERVER = 'PARTY_SERVER';\r\nconst DEGRADED = 'DEGRADED';\r\nconst SUSPECT = 'SUSPECT';\r\nconst DOWN = 'DOWN';\r\nconst NOT_SERVER = 'NOT_SERVER';\r\nconst MAYBE = 'MAYBE';\r\n\r\nconst MAYBE_MAX_RETRIES = 3;\r\nconst BACKOFF_BASE = 60;\r\nconst BACKOFF_CAP = 900;\r\nconst FAILURE_DEGRADED = 1;\r\nconst FAILURE_SUSPECT = 2;\r\nconst FAILURE_DOWN = 3;\r\n\r\n// ---------------------------------------------------------------------------\r\n// Data types\r\n// ---------------------------------------------------------------------------\r\n\r\ninterface PeerState {\r\n ip: string;\r\n status: string;\r\n consecutiveFailures: number;\r\n lastProbeAt: number;\r\n backoffUntil: number | null;\r\n maybeRetries: number;\r\n}\r\n\r\ninterface RemoteAgentEntry {\r\n agentInfo: AgentInfo;\r\n sourcePeerIp: string;\r\n lastSyncedAt: number;\r\n reachable: boolean;\r\n}\r\n\r\n// ---------------------------------------------------------------------------\r\n// Error classification for native fetch\r\n// ---------------------------------------------------------------------------\r\n\r\n/** Classify a fetch error as refused (false), timeout/uncertain (null). */\r\nfunction classifyFetchError(error: unknown): boolean | null {\r\n if (error instanceof TypeError && /fetch failed/i.test(error.message)) {\r\n const cause = (error as NodeJS.ErrnoException).cause as { code?: string } | undefined;\r\n if (cause?.code === 'ECONNREFUSED') return false;\r\n return null; // timeout or other network error = uncertain\r\n }\r\n if (error instanceof DOMException && error.name === 'AbortError') return null;\r\n return null;\r\n}\r\n\r\n// ---------------------------------------------------------------------------\r\n// PeerDiscovery\r\n// ---------------------------------------------------------------------------\r\n\r\nexport class PeerDiscovery {\r\n private _selfIp: string;\r\n private _peers = new Map<string, PeerState>();\r\n private _remoteAgents = new Map<string, RemoteAgentEntry>();\r\n\r\n constructor(selfIp: string) {\r\n this._selfIp = selfIp;\r\n }\r\n\r\n // ------------------------------------------------------------------\r\n // Public query API (used by router)\r\n // ------------------------------------------------------------------\r\n\r\n /** Return the peer IP that owns agentId, or undefined. */\r\n getPeerForAgent(agentId: string): string | undefined {\r\n return this._remoteAgents.get(agentId)?.sourcePeerIp;\r\n }\r\n\r\n /** Return true if the peer is in a serving state. */\r\n isPeerReachable(peerIp: string): boolean {\r\n const ps = this._peers.get(peerIp);\r\n return ps !== undefined && (ps.status === PARTY_SERVER || ps.status === DEGRADED || ps.status === SUSPECT);\r\n }\r\n\r\n /** Return all remote agents (including unreachable ones). */\r\n getAllRemoteAgents(): AgentInfo[] {\r\n return Array.from(this._remoteAgents.values(), (e) => e.agentInfo);\r\n }\r\n\r\n /** Return only reachable remote agents. */\r\n getReachableRemoteAgents(): AgentInfo[] {\r\n return Array.from(this._remoteAgents.values())\r\n .filter((e) => e.reachable)\r\n .map((e) => e.agentInfo);\r\n }\r\n\r\n /** Return all known peer states for dashboard consumption. */\r\n getPeerStates(): Array<{\r\n ip: string;\r\n status: string;\r\n consecutiveFailures: number;\r\n lastProbeAt: number;\r\n backoffUntil: number | null;\r\n }> {\r\n return Array.from(this._peers.values()).map((ps) => ({\r\n ip: ps.ip,\r\n status: ps.status,\r\n consecutiveFailures: ps.consecutiveFailures,\r\n lastProbeAt: ps.lastProbeAt,\r\n backoffUntil: ps.backoffUntil,\r\n }));\r\n }\r\n\r\n /** Return remote agent entries with peer mapping info. */\r\n getRemoteAgentEntries(): Array<{\r\n agentInfo: AgentInfo;\r\n sourcePeerIp: string;\r\n lastSyncedAt: number;\r\n reachable: boolean;\r\n }> {\r\n return Array.from(this._remoteAgents.values()).map((e) => ({\r\n agentInfo: e.agentInfo,\r\n sourcePeerIp: e.sourcePeerIp,\r\n lastSyncedAt: e.lastSyncedAt,\r\n reachable: e.reachable,\r\n }));\r\n }\r\n\r\n // ------------------------------------------------------------------\r\n // Main discovery loop\r\n // ------------------------------------------------------------------\r\n\r\n async runLoop(): Promise<void> {\r\n while (true) {\r\n try {\r\n await this.discoveryCycle();\r\n } catch (e) {\r\n console.error('[Discovery] Cycle failed:', e);\r\n }\r\n await sleep(DISCOVERY_INTERVAL * 1000);\r\n }\r\n }\r\n\r\n private async discoveryCycle(): Promise<void> {\r\n const peerIps = this.getTailscalePeers();\r\n\r\n // Add new peers as UNKNOWN\r\n for (const ip of peerIps) {\r\n if (!this._peers.has(ip)) {\r\n this._peers.set(ip, { ip, status: UNKNOWN, consecutiveFailures: 0, lastProbeAt: 0, backoffUntil: null, maybeRetries: 0 });\r\n }\r\n }\r\n\r\n const now = performance.now() / 1000;\r\n\r\n // Process each peer\r\n for (const ip of peerIps) {\r\n const ps = this._peers.get(ip)!;\r\n\r\n // Skip NOT_SERVER peers still in backoff\r\n if (ps.status === NOT_SERVER) {\r\n if (ps.backoffUntil !== null && now < ps.backoffUntil) continue;\r\n ps.status = UNKNOWN;\r\n }\r\n\r\n // Skip MAYBE peers that exhausted retries\r\n if (ps.status === MAYBE && ps.maybeRetries >= MAYBE_MAX_RETRIES) {\r\n this.transition(ps, NOT_SERVER);\r\n continue;\r\n }\r\n\r\n await this.probePeer(ps);\r\n }\r\n\r\n this.evictDownAgents();\r\n this.evictStaleAgents();\r\n }\r\n\r\n // ------------------------------------------------------------------\r\n // Tailscale peer enumeration\r\n // ------------------------------------------------------------------\r\n\r\n private getTailscalePeers(): string[] {\r\n let status: Record<string, unknown>;\r\n try {\r\n status = readTailscaleStatus();\r\n } catch {\r\n console.warn('[Discovery] Failed to read Tailscale status');\r\n return [];\r\n }\r\n\r\n const peersRaw = status.Peer as Record<string, Record<string, unknown>> | undefined;\r\n if (!peersRaw || typeof peersRaw !== 'object') return [];\r\n\r\n const ips: string[] = [];\r\n for (const peer of Object.values(peersRaw)) {\r\n if (typeof peer !== 'object' || !peer) continue;\r\n if (!peer.Online) continue;\r\n if (peer.ExitNode) continue;\r\n const tailscaleIps = peer.TailscaleIPs as string[] | undefined;\r\n if (!tailscaleIps) continue;\r\n for (const ip of tailscaleIps) {\r\n if (ip.includes('.') && !ip.includes(':')) {\r\n if (ip !== this._selfIp) ips.push(ip);\r\n break;\r\n }\r\n }\r\n }\r\n return ips;\r\n }\r\n\r\n // ------------------------------------------------------------------\r\n // Peer probing\r\n // ------------------------------------------------------------------\r\n\r\n private async probePeer(ps: PeerState): Promise<void> {\r\n ps.lastProbeAt = performance.now() / 1000;\r\n const healthy = await this.checkHealth(ps.ip);\r\n\r\n if (healthy === null) {\r\n this.handleProbeFailure(ps, true);\r\n } else if (healthy) {\r\n await this.handleProbeSuccess(ps);\r\n } else {\r\n this.handleProbeFailure(ps, false);\r\n }\r\n }\r\n\r\n private async checkHealth(ip: string): Promise<boolean | null> {\r\n const url = `http://${ip}:${PARTY_PORT}/proxy/health`;\r\n try {\r\n const controller = new AbortController();\r\n const timer = setTimeout(() => controller.abort(), PROBE_TIMEOUT * 1000);\r\n const resp = await fetch(url, { signal: controller.signal });\r\n clearTimeout(timer);\r\n const data = (await resp.json()) as Record<string, unknown>;\r\n return resp.status === 200 && 'status' in data;\r\n } catch (e) {\r\n return classifyFetchError(e);\r\n }\r\n }\r\n\r\n // ------------------------------------------------------------------\r\n // State machine transitions\r\n // ------------------------------------------------------------------\r\n\r\n private async handleProbeSuccess(ps: PeerState): Promise<void> {\r\n ps.consecutiveFailures = 0;\r\n ps.maybeRetries = 0;\r\n ps.backoffUntil = null;\r\n\r\n if (ps.status !== PARTY_SERVER) {\r\n this.transition(ps, PARTY_SERVER);\r\n }\r\n\r\n await this.syncAgents(ps.ip);\r\n }\r\n\r\n private handleProbeFailure(ps: PeerState, timeout: boolean): void {\r\n if (ps.status === UNKNOWN || ps.status === MAYBE) {\r\n if (timeout) {\r\n ps.maybeRetries++;\r\n if (ps.maybeRetries >= MAYBE_MAX_RETRIES) {\r\n this.transition(ps, NOT_SERVER);\r\n } else {\r\n this.transition(ps, MAYBE);\r\n }\r\n } else {\r\n this.transition(ps, NOT_SERVER);\r\n }\r\n } else if (ps.status === PARTY_SERVER) {\r\n ps.consecutiveFailures++;\r\n if (ps.consecutiveFailures >= FAILURE_DOWN) {\r\n this.transition(ps, DOWN);\r\n } else if (ps.consecutiveFailures >= FAILURE_SUSPECT) {\r\n this.transition(ps, SUSPECT);\r\n } else {\r\n this.transition(ps, DEGRADED);\r\n }\r\n } else if (ps.status === DEGRADED || ps.status === SUSPECT) {\r\n ps.consecutiveFailures++;\r\n if (ps.consecutiveFailures >= FAILURE_DOWN) {\r\n this.transition(ps, DOWN);\r\n } else if (ps.status === DEGRADED && ps.consecutiveFailures >= FAILURE_SUSPECT) {\r\n this.transition(ps, SUSPECT);\r\n }\r\n }\r\n // DOWN → nothing changes\r\n }\r\n\r\n private transition(ps: PeerState, newStatus: string): void {\r\n const old = ps.status;\r\n ps.status = newStatus;\r\n if (old !== newStatus) {\r\n console.log(`[Discovery] Peer ${ps.ip}: ${old} -> ${newStatus}`);\r\n }\r\n\r\n if (newStatus === NOT_SERVER) {\r\n const retries = ps.maybeRetries > 0 ? ps.maybeRetries : 1;\r\n const delay = Math.min(BACKOFF_BASE * Math.pow(2, retries - 1), BACKOFF_CAP);\r\n ps.backoffUntil = performance.now() / 1000 + delay;\r\n }\r\n\r\n if (newStatus === DOWN) {\r\n for (const entry of this._remoteAgents.values()) {\r\n if (entry.sourcePeerIp === ps.ip) {\r\n entry.reachable = false;\r\n }\r\n }\r\n }\r\n }\r\n\r\n // ------------------------------------------------------------------\r\n // Agent synchronization\r\n // ------------------------------------------------------------------\r\n\r\n private async syncAgents(peerIp: string): Promise<void> {\r\n const url = `http://${peerIp}:${PARTY_PORT}/proxy/list_agents`;\r\n try {\r\n const controller = new AbortController();\r\n const timer = setTimeout(() => controller.abort(), PROBE_TIMEOUT * 1000);\r\n const resp = await fetch(url, { signal: controller.signal });\r\n clearTimeout(timer);\r\n const data = (await resp.json()) as { agents?: AgentInfo[] };\r\n const agentsRaw = data.agents ?? [];\r\n const now = Date.now() / 1000;\r\n\r\n const seenIds = new Set<string>();\r\n for (const a of agentsRaw) {\r\n seenIds.add(a.agent_id);\r\n this._remoteAgents.set(a.agent_id, {\r\n agentInfo: a,\r\n sourcePeerIp: peerIp,\r\n lastSyncedAt: now,\r\n reachable: true,\r\n });\r\n }\r\n\r\n // Remove agents from this peer no longer present\r\n for (const [aid, entry] of this._remoteAgents) {\r\n if (entry.sourcePeerIp === peerIp && !seenIds.has(aid)) {\r\n this._remoteAgents.delete(aid);\r\n }\r\n }\r\n } catch {\r\n console.warn(`[Discovery] Failed to sync agents from ${peerIp}`);\r\n }\r\n }\r\n\r\n // ------------------------------------------------------------------\r\n // Cleanup\r\n // ------------------------------------------------------------------\r\n\r\n private evictDownAgents(): void {\r\n const downPeers = new Set<string>();\r\n for (const [ip, ps] of this._peers) {\r\n if (ps.status === DOWN) downPeers.add(ip);\r\n }\r\n if (!downPeers.size) return;\r\n\r\n for (const [aid, entry] of this._remoteAgents) {\r\n if (downPeers.has(entry.sourcePeerIp)) {\r\n this._remoteAgents.delete(aid);\r\n }\r\n }\r\n }\r\n\r\n private evictStaleAgents(): void {\r\n const threshold = DISCOVERY_INTERVAL * REMOTE_STALE_FACTOR;\r\n const now = Date.now() / 1000;\r\n\r\n for (const [aid, entry] of this._remoteAgents) {\r\n if (now - entry.lastSyncedAt > threshold) {\r\n this._remoteAgents.delete(aid);\r\n }\r\n }\r\n }\r\n}\r\n\r\n// ---------------------------------------------------------------------------\r\n// Utility\r\n// ---------------------------------------------------------------------------\r\n\r\nfunction sleep(ms: number): Promise<void> {\r\n return new Promise((resolve) => setTimeout(resolve, ms));\r\n}\r\n","import type { AgentInfo, RegistrationRequest } from './models.js';\n\n/** In-memory registry of agents connected to this server. */\nexport class AgentRegistry {\n private _agents = new Map<string, AgentInfo>();\n private _selfIp: string;\n\n constructor(selfIp: string) {\n this._selfIp = selfIp;\n }\n\n register(req: RegistrationRequest): AgentInfo {\n const now = Date.now() / 1000;\n const info: AgentInfo = {\n agent_id: req.agent_id,\n display_name: req.display_name,\n host_ip: this._selfIp,\n registered_at: now,\n last_heartbeat: now,\n metadata: req.metadata ?? {},\n };\n this._agents.set(req.agent_id, info);\n return info;\n }\n\n remove(agentId: string): boolean {\n return this._agents.delete(agentId);\n }\n\n heartbeat(agentId: string): AgentInfo {\n const info = this._agents.get(agentId);\n if (!info) throw new Error(`Agent '${agentId}' not registered`);\n info.last_heartbeat = Date.now() / 1000;\n return info;\n }\n\n get(agentId: string): AgentInfo | undefined {\n return this._agents.get(agentId);\n }\n\n listAll(): AgentInfo[] {\n return Array.from(this._agents.values());\n }\n\n /** Remove agents whose last heartbeat is older than timeout seconds. */\n cleanupStale(timeout: number): string[] {\n const now = Date.now() / 1000;\n const stale: string[] = [];\n for (const [aid, info] of this._agents) {\n if (now - info.last_heartbeat > timeout) {\n stale.push(aid);\n }\n }\n for (const aid of stale) {\n this._agents.delete(aid);\n }\n return stale;\n }\n}\n","/**\r\n * Shared state singletons for the Party Server.\r\n *\r\n * All mutable state (registry, messageQueue, discovery) is owned here.\r\n */\r\nimport { getTailscaleIps } from '../infra/tailscale.js';\r\nimport { MessageQueue } from './message-queue.js';\r\nimport { PeerDiscovery } from './peer-discovery.js';\r\nimport { AgentRegistry } from './registry.js';\r\n\r\nfunction resolveSelfIp(): string {\r\n try {\r\n const ips = getTailscaleIps();\r\n for (const ip of ips) {\r\n if (ip.includes('.') && !ip.includes(':')) return ip;\r\n }\r\n return ips.length > 0 ? ips[0] : '127.0.0.1';\r\n } catch {\r\n return '127.0.0.1';\r\n }\r\n}\r\n\r\nlet _selfIp = resolveSelfIp();\r\n\r\nexport function getSelfIp(): string {\r\n return _selfIp;\r\n}\r\n\r\nexport function refreshSelfIp(): string {\r\n _selfIp = resolveSelfIp();\r\n return _selfIp;\r\n}\r\n\r\nexport const STARTED_AT = Date.now();\r\nexport const registry = new AgentRegistry(getSelfIp());\r\nexport const messageQueue = new MessageQueue();\r\nexport const discovery = new PeerDiscovery(getSelfIp());\r\n","/** Agent registration request. */\nexport interface RegistrationRequest {\n agent_id: string;\n display_name?: string;\n metadata?: Record<string, unknown>;\n}\n\n/** Heartbeat request. */\nexport interface HeartbeatRequest {\n agent_id: string;\n}\n\n/** Registered agent info (internal — includes host_ip for routing). */\nexport interface AgentInfo {\n agent_id: string;\n display_name?: string;\n host_ip: string;\n registered_at: number;\n last_heartbeat: number;\n metadata: Record<string, unknown>;\n}\n\n/** Agent info safe to return to agents (no host_ip). */\nexport interface PublicAgentInfo {\n agent_id: string;\n display_name?: string;\n registered_at: number;\n last_heartbeat: number;\n metadata: Record<string, unknown>;\n}\n\n/** Strip host_ip from AgentInfo for agent-facing responses. */\nexport function sanitizeAgentInfo(info: AgentInfo): PublicAgentInfo {\n const { host_ip: _, ...publicInfo } = info;\n return publicInfo;\n}\n\n/** Strip host_ip from an array of AgentInfo. */\nexport function sanitizeAgentList(agents: AgentInfo[]): PublicAgentInfo[] {\n return agents.map(sanitizeAgentInfo);\n}\n\n/** Agent list response. */\nexport interface AgentListResponse {\n agents: AgentInfo[];\n count: number;\n}\n\n/** Remove request. */\nexport interface RemoveRequest {\n agent_id: string;\n}\n\n/** Message envelope. */\nexport interface MessageEnvelope {\n sender_id: string;\n recipient_id?: string; // undefined = broadcast\n group_id?: string;\n summary?: string; // short summary for terminal display\n content: string;\n timestamp?: number;\n}\n\n/** A single entry in an agent's message history. */\nexport interface HistoryEntry {\n direction: 'sent' | 'received';\n sender_id: string;\n recipient_id?: string;\n summary?: string;\n content: string;\n timestamp: number;\n}\n\n/** Send result. */\nexport interface SendResult {\n status: 'delivered_locally' | 'forwarded' | 'agent_not_found' | 'error' | 'peer_unreachable';\n target?: string;\n error?: string;\n}\n","import { Hono } from 'hono';\nimport type { MessageEnvelope, RegistrationRequest, HeartbeatRequest, RemoveRequest, SendResult } from '../models.js';\nimport { sanitizeAgentInfo, sanitizeAgentList } from '../models.js';\nimport { PARTY_PORT } from '../config.js';\n\nconst agentRoutes = new Hono();\n\nasync function forwardToPeer(peerIp: string, envelope: MessageEnvelope): Promise<SendResult> {\n const url = `http://${peerIp}:${PARTY_PORT}/proxy/receive`;\n const payload: Record<string, string> = { sender_id: envelope.sender_id, content: envelope.content };\n if (envelope.recipient_id) payload.recipient_id = envelope.recipient_id;\n if (envelope.group_id) payload.group_id = envelope.group_id;\n try {\n await fetch(url, {\n method: 'POST',\n headers: { 'Content-Type': 'application/json' },\n body: JSON.stringify(payload),\n });\n return { status: 'forwarded', target: peerIp };\n } catch (e) {\n return { status: 'error', target: peerIp, error: (e as Error).message };\n }\n}\n\nagentRoutes.post('/register', async (c) => {\n const { registry } = await import('../state.js');\n const req = await c.req.json<RegistrationRequest>();\n const info = registry.register(req);\n return c.json(sanitizeAgentInfo(info));\n});\n\nagentRoutes.post('/remove', async (c) => {\n const { registry } = await import('../state.js');\n const req = await c.req.json<RemoveRequest>();\n const removed = registry.remove(req.agent_id);\n return c.json({ status: removed ? 'removed' : 'not_found' });\n});\n\nagentRoutes.post('/heartbeat', async (c) => {\n const { registry } = await import('../state.js');\n const req = await c.req.json<HeartbeatRequest>();\n const info = registry.heartbeat(req.agent_id);\n return c.json({ status: 'ok', last_heartbeat: info.last_heartbeat });\n});\n\nagentRoutes.get('/list', async (c) => {\n const { registry, discovery } = await import('../state.js');\n const localAgents = registry.listAll();\n const remoteAgents = discovery.getReachableRemoteAgents();\n const allAgents = localAgents.concat(remoteAgents);\n return c.json({ agents: sanitizeAgentList(allAgents), count: allAgents.length });\n});\n\nagentRoutes.post('/send', async (c) => {\n const { registry, messageQueue, discovery } = await import('../state.js');\n const envelope = await c.req.json<MessageEnvelope>();\n const recipient = envelope.recipient_id;\n\n if (!recipient) {\n return c.json({ status: 'error' as const, error: 'recipient_id is required' });\n }\n\n // Check if recipient is local\n if (registry.get(recipient)) {\n const stamped = { ...envelope, timestamp: envelope.timestamp ?? Date.now() / 1000 };\n const count = messageQueue.enqueue(recipient, stamped);\n messageQueue.logToHistory(envelope.sender_id, 'sent', stamped);\n messageQueue.logToHistory(recipient, 'received', stamped);\n return c.json({ status: 'delivered_locally' as const, target: recipient });\n }\n\n // Check remote agent cache\n const peerIp = discovery.getPeerForAgent(recipient);\n if (peerIp) {\n if (discovery.isPeerReachable(peerIp)) {\n const result = await forwardToPeer(peerIp, envelope);\n if (result.status === 'forwarded') {\n messageQueue.logToHistory(envelope.sender_id, 'sent', { ...envelope, timestamp: envelope.timestamp ?? Date.now() / 1000 });\n }\n return c.json(result);\n } else {\n return c.json({ status: 'peer_unreachable' as const, target: peerIp });\n }\n }\n\n return c.json({ status: 'agent_not_found' as const, target: recipient });\n});\n\nagentRoutes.get('/messages/:agent_id', async (c) => {\n const { messageQueue } = await import('../state.js');\n const agentId = c.req.param('agent_id');\n const maxCount = parseInt(c.req.query('max_count') || '50', 10);\n const msgs = messageQueue.dequeue(agentId, maxCount);\n return c.json({ messages: msgs, count: msgs.length });\n});\n\n/** Get message history for an agent. */\nagentRoutes.get('/history/:agent_id', async (c) => {\n const { messageQueue } = await import('../state.js');\n const agentId = c.req.param('agent_id');\n const limit = parseInt(c.req.query('limit') || '20', 10);\n const history = messageQueue.getHistory(agentId, limit);\n return c.json({ history, count: history.length });\n});\n\nexport { agentRoutes };\n","import { Hono } from 'hono';\r\nimport type { MessageEnvelope, SendResult } from '../models.js';\r\nimport { PARTY_PORT } from '../config.js';\r\nimport { getTailnetHostname } from '../../infra/tailscale.js';\r\nimport { registry, messageQueue, getSelfIp } from '../state.js';\r\n\r\nconst proxyRoutes = new Hono();\r\n\r\nproxyRoutes.get('/health', async (c) => {\r\n let hostname = '127.0.0.1';\r\n try {\r\n hostname = getTailnetHostname();\r\n } catch { /* ignore */ }\r\n\r\n return c.json({\r\n status: 'ok',\r\n tailscale_ip: getSelfIp(),\r\n hostname,\r\n agent_count: registry.listAll().length,\r\n });\r\n});\r\n\r\nproxyRoutes.get('/list_agents', async (c) => {\r\n const agents = registry.listAll();\r\n return c.json({ agents, count: agents.length });\r\n});\r\n\r\nproxyRoutes.post('/receive', async (c) => {\r\n const envelope = await c.req.json<MessageEnvelope>();\r\n const stamped = { ...envelope, timestamp: envelope.timestamp ?? Date.now() / 1000 };\r\n\r\n if (envelope.recipient_id) {\r\n messageQueue.enqueue(envelope.recipient_id, stamped);\r\n messageQueue.logToHistory(envelope.recipient_id, 'received', stamped);\r\n console.log(`[收到消息] ${envelope.sender_id} -> ${envelope.recipient_id}: ${envelope.content}`);\r\n } else {\r\n // Broadcast\r\n for (const agent of registry.listAll()) {\r\n messageQueue.enqueue(agent.agent_id, stamped);\r\n messageQueue.logToHistory(agent.agent_id, 'received', stamped);\r\n }\r\n console.log(`[广播消息] ${envelope.sender_id} -> all: ${envelope.content}`);\r\n }\r\n return c.json({ status: 'received' });\r\n});\r\n\r\nproxyRoutes.post('/send/:target_ip', async (c) => {\r\n const targetIp = c.req.param('target_ip');\r\n const envelope = await c.req.json<MessageEnvelope>();\r\n\r\n const senderId = envelope.sender_id || getSelfIp();\r\n const url = `http://${targetIp}:${PARTY_PORT}/proxy/receive`;\r\n const payload: Record<string, string> = { sender_id: senderId, content: envelope.content };\r\n if (envelope.recipient_id) payload.recipient_id = envelope.recipient_id;\r\n if (envelope.group_id) payload.group_id = envelope.group_id;\r\n\r\n try {\r\n await fetch(url, {\r\n method: 'POST',\r\n headers: { 'Content-Type': 'application/json' },\r\n body: JSON.stringify(payload),\r\n });\r\n console.log(`[发送成功] 目标 ${targetIp}`);\r\n return c.json({ status: 'forwarded' as const, target: targetIp });\r\n } catch (e) {\r\n console.log(`[发送失败] 目标 ${targetIp}, 错误: ${(e as Error).message}`);\r\n return c.json({ status: 'error' as const, target: targetIp, error: (e as Error).message });\r\n }\r\n});\r\n\r\nexport { proxyRoutes };\r\n","export const DASHBOARD_HTML = `<!DOCTYPE html>\r\n<html lang=\"zh-CN\">\r\n<head>\r\n<meta charset=\"UTF-8\">\r\n<meta name=\"viewport\" content=\"width=device-width, initial-scale=1.0\">\r\n<title>OPEN PARTY // Dashboard</title>\r\n<style>\r\n*,*::before,*::after{box-sizing:border-box;margin:0;padding:0}\r\n:root{\r\n --bg:#0a0a0f;--card:#12121a;--border:#1e1e2e;--border-bright:#2a2a3e;\r\n --text:#e0e0e8;--muted:#6a6a8a;\r\n --cyan:#00fff0;--magenta:#ff00ff;--green:#00ff88;--red:#ff3366;--yellow:#ffaa00;--orange:#ff8800;\r\n --font-mono:'JetBrains Mono','Fira Code','Cascadia Code','Consolas','Courier New',monospace;\r\n --font-sans:'Inter','Segoe UI',system-ui,-apple-system,sans-serif;\r\n}\r\nhtml{font-size:14px}\r\nbody{\r\n background:var(--bg);color:var(--text);font-family:var(--font-sans);\r\n min-height:100vh;overflow-x:hidden;position:relative;\r\n}\r\n/* Grid background */\r\nbody::before{\r\n content:'';position:fixed;inset:0;z-index:0;pointer-events:none;\r\n background-image:\r\n linear-gradient(rgba(0,255,240,0.03) 1px,transparent 1px),\r\n linear-gradient(90deg,rgba(0,255,240,0.03) 1px,transparent 1px);\r\n background-size:40px 40px;\r\n}\r\n/* Scanline overlay */\r\nbody::after{\r\n content:'';position:fixed;inset:0;z-index:9999;pointer-events:none;\r\n background:repeating-linear-gradient(\r\n 0deg,transparent,transparent 2px,rgba(0,0,0,0.08) 2px,rgba(0,0,0,0.08) 4px\r\n );\r\n}\r\n#app{position:relative;z-index:1;max-width:1400px;margin:0 auto;padding:0 20px 40px}\r\n\r\n/* ===== Header ===== */\r\n.header{\r\n display:flex;align-items:center;justify-content:space-between;\r\n padding:16px 0;border-bottom:1px solid var(--border);\r\n margin-bottom:20px;flex-wrap:wrap;gap:12px;\r\n}\r\n.header-title{\r\n font-family:var(--font-mono);font-size:1.4rem;font-weight:700;\r\n color:var(--cyan);letter-spacing:3px;\r\n text-shadow:0 0 10px rgba(0,255,240,0.5),0 0 30px rgba(0,255,240,0.2);\r\n}\r\n.header-title span{color:var(--muted);font-weight:400;font-size:0.9rem;margin-left:8px;letter-spacing:1px}\r\n.header-center{display:flex;align-items:center;gap:16px;flex-wrap:wrap}\r\n.header-status{display:flex;align-items:center;gap:6px;font-family:var(--font-mono);font-size:0.85rem}\r\n.status-dot{\r\n width:8px;height:8px;border-radius:50%;background:var(--green);\r\n box-shadow:0 0 6px var(--green);animation:pulse 2s ease-in-out infinite;\r\n}\r\n.status-dot.offline{background:var(--red);box-shadow:0 0 6px var(--red)}\r\n.status-dot.not-installed{background:var(--red);box-shadow:0 0 6px var(--red)}\r\n.status-dot.not-connected{background:var(--yellow);box-shadow:0 0 6px var(--yellow)}\r\n@keyframes pulse{0%,100%{opacity:1;transform:scale(1)}50%{opacity:0.5;transform:scale(0.8)}}\r\n.header-meta{color:var(--muted);font-family:var(--font-mono);font-size:0.8rem}\r\n.header-right{font-family:var(--font-mono);font-size:0.85rem;color:var(--muted)}\r\n.header-right .value{color:var(--cyan)}\r\n\r\n/* ===== Stats Cards ===== */\r\n.stats-row{display:grid;grid-template-columns:repeat(4,1fr);gap:16px;margin-bottom:24px}\r\n@media(max-width:768px){.stats-row{grid-template-columns:repeat(2,1fr)}}\r\n.stat-card{\r\n background:var(--card);border:1px solid var(--border);border-radius:8px;\r\n padding:16px 20px;position:relative;overflow:hidden;\r\n transition:border-color 0.3s;\r\n}\r\n.stat-card:hover{border-color:var(--border-bright)}\r\n.stat-card::before{\r\n content:'';position:absolute;top:0;left:0;right:0;height:2px;\r\n}\r\n.stat-card.cyan::before{background:var(--cyan);box-shadow:0 0 10px var(--cyan)}\r\n.stat-card.green::before{background:var(--green);box-shadow:0 0 10px var(--green)}\r\n.stat-card.magenta::before{background:var(--magenta);box-shadow:0 0 10px var(--magenta)}\r\n.stat-card.yellow::before{background:var(--yellow);box-shadow:0 0 10px var(--yellow)}\r\n.stat-value{\r\n font-family:var(--font-mono);font-size:2rem;font-weight:700;\r\n transition:color 0.3s;\r\n}\r\n.stat-card.cyan .stat-value{color:var(--cyan)}\r\n.stat-card.green .stat-value{color:var(--green)}\r\n.stat-card.magenta .stat-value{color:var(--magenta)}\r\n.stat-card.yellow .stat-value{color:var(--yellow)}\r\n.stat-label{color:var(--muted);font-size:0.8rem;margin-top:4px;text-transform:uppercase;letter-spacing:1px}\r\n.stat-value.flash{animation:flash 0.3s ease}\r\n@keyframes flash{0%{opacity:1}50%{opacity:0.3}100%{opacity:1}}\r\n\r\n/* ===== Main Grid ===== */\r\n.main-grid{display:grid;grid-template-columns:2fr 1fr;gap:20px;margin-bottom:24px}\r\n@media(max-width:1024px){.main-grid{grid-template-columns:1fr}}\r\n.section{\r\n background:var(--card);border:1px solid var(--border);border-radius:8px;\r\n overflow:hidden;\r\n}\r\n.section-header{\r\n padding:12px 16px;border-bottom:1px solid var(--border);\r\n font-family:var(--font-mono);font-size:0.8rem;font-weight:600;\r\n color:var(--muted);letter-spacing:2px;text-transform:uppercase;\r\n display:flex;align-items:center;gap:8px;\r\n}\r\n.section-header .dot{width:6px;height:6px;border-radius:50%;background:var(--cyan)}\r\n.section-body{padding:16px}\r\n\r\n/* ===== Topology ===== */\r\n.topology-container{display:flex;justify-content:center;align-items:center;min-height:300px}\r\n.topology-container svg{width:100%;max-width:500px;height:300px}\r\n.topo-node{cursor:pointer;transition:filter 0.3s}\r\n.topo-node:hover{filter:brightness(1.3)}\r\n.topo-label{font-family:var(--font-mono);font-size:10px;fill:var(--muted)}\r\n.topo-badge{\r\n font-family:var(--font-mono);font-size:9px;fill:var(--bg);\r\n font-weight:700;\r\n}\r\n\r\n/* ===== Peer Table ===== */\r\n.peer-table{width:100%;border-collapse:collapse;font-size:0.85rem}\r\n.peer-table th{\r\n text-align:left;padding:8px 10px;color:var(--muted);font-weight:600;\r\n font-family:var(--font-mono);font-size:0.75rem;text-transform:uppercase;\r\n letter-spacing:1px;border-bottom:1px solid var(--border);\r\n}\r\n.peer-table td{padding:8px 10px;border-bottom:1px solid var(--border);font-family:var(--font-mono);font-size:0.8rem}\r\n.peer-table tr:last-child td{border-bottom:none}\r\n.peer-table tr:hover td{background:rgba(255,255,255,0.02)}\r\n.badge{\r\n display:inline-block;padding:2px 8px;border-radius:4px;font-size:0.7rem;\r\n font-weight:600;text-transform:uppercase;letter-spacing:0.5px;\r\n}\r\n.badge-party{background:rgba(0,255,136,0.15);color:var(--green)}\r\n.badge-degraded{background:rgba(255,170,0,0.15);color:var(--yellow)}\r\n.badge-suspect{background:rgba(255,136,0,0.15);color:var(--orange)}\r\n.badge-down{background:rgba(255,51,102,0.15);color:var(--red)}\r\n.badge-unknown{background:rgba(106,106,138,0.15);color:var(--muted)}\r\n.badge-not_server{background:rgba(106,106,138,0.1);color:var(--muted)}\r\n.badge-maybe{background:rgba(0,255,240,0.1);color:var(--cyan)}\r\n.empty-state{text-align:center;color:var(--muted);padding:40px 20px;font-family:var(--font-mono);font-size:0.85rem}\r\n\r\n/* ===== Agent & Message Grid ===== */\r\n.bottom-grid{display:grid;grid-template-columns:1fr 1fr;gap:20px;margin-bottom:24px}\r\n@media(max-width:1024px){.bottom-grid{grid-template-columns:1fr}}\r\n\r\n/* Agent cards */\r\n.agent-list{display:flex;flex-direction:column;gap:8px;max-height:400px;overflow-y:auto}\r\n.agent-card{\r\n display:flex;align-items:center;gap:12px;padding:10px 14px;\r\n border:1px solid var(--border);border-radius:6px;transition:border-color 0.3s;\r\n}\r\n.agent-card:hover{border-color:var(--border-bright)}\r\n.agent-card.local{border-left:3px solid var(--cyan)}\r\n.agent-card.remote{border-left:3px solid var(--green)}\r\n.agent-icon{\r\n width:36px;height:36px;border-radius:6px;display:flex;align-items:center;\r\n justify-content:center;font-family:var(--font-mono);font-size:0.9rem;font-weight:700;\r\n flex-shrink:0;\r\n}\r\n.agent-card.local .agent-icon{background:rgba(0,255,240,0.1);color:var(--cyan)}\r\n.agent-card.remote .agent-icon{background:rgba(0,255,136,0.1);color:var(--green)}\r\n.agent-info{flex:1;min-width:0}\r\n.agent-name{font-weight:600;font-size:0.9rem;white-space:nowrap;overflow:hidden;text-overflow:ellipsis}\r\n.agent-id{font-family:var(--font-mono);font-size:0.7rem;color:var(--muted);margin-top:2px}\r\n.agent-meta{display:flex;gap:6px;flex-shrink:0;align-items:center}\r\n.agent-tag{\r\n font-family:var(--font-mono);font-size:0.65rem;padding:2px 6px;\r\n border-radius:3px;background:rgba(255,255,255,0.05);color:var(--muted);\r\n}\r\n.agent-tag.unreachable{background:rgba(255,51,102,0.1);color:var(--red)}\r\n\r\n/* Message feed */\r\n.msg-feed{display:flex;flex-direction:column;gap:6px;max-height:400px;overflow-y:auto}\r\n.msg-item{\r\n padding:10px 12px;border:1px solid var(--border);border-radius:6px;\r\n border-left:3px solid var(--cyan);font-size:0.8rem;\r\n}\r\n.msg-item.received{border-left-color:var(--magenta)}\r\n.msg-top{display:flex;justify-content:space-between;align-items:center;margin-bottom:4px}\r\n.msg-flow{font-family:var(--font-mono);font-size:0.75rem;color:var(--cyan)}\r\n.msg-flow .arrow{color:var(--muted);margin:0 4px}\r\n.msg-time{font-family:var(--font-mono);font-size:0.65rem;color:var(--muted)}\r\n.msg-content{color:var(--text);font-size:0.8rem;line-height:1.4;overflow:hidden;text-overflow:ellipsis;white-space:nowrap}\r\n\r\n/* Scrollbar */\r\n::-webkit-scrollbar{width:4px}\r\n::-webkit-scrollbar-track{background:var(--bg)}\r\n::-webkit-scrollbar-thumb{background:var(--border);border-radius:2px}\r\n::-webkit-scrollbar-thumb:hover{background:var(--border-bright)}\r\n\r\n/* Footer */\r\n.footer{\r\n text-align:center;padding:20px 0;border-top:1px solid var(--border);\r\n color:var(--muted);font-family:var(--font-mono);font-size:0.75rem;\r\n letter-spacing:1px;\r\n}\r\n\r\n/* Join Network Button */\r\n.btn-join{\r\n font-family:var(--font-mono);font-size:0.75rem;letter-spacing:1px;\r\n padding:6px 14px;border:1px solid var(--cyan);border-radius:4px;\r\n background:rgba(0,255,240,0.08);color:var(--cyan);cursor:pointer;\r\n transition:all 0.2s;text-transform:uppercase;margin-left:12px;\r\n}\r\n.btn-join:hover{background:rgba(0,255,240,0.18);box-shadow:0 0 10px rgba(0,255,240,0.2)}\r\n.btn-join:active{transform:scale(0.97)}\r\n.btn-join.connected{border-color:var(--green);color:var(--green);background:rgba(0,255,136,0.08)}\r\n\r\n/* Modal */\r\n.modal-overlay{\r\n position:fixed;inset:0;background:rgba(0,0,0,0.7);z-index:10000;\r\n display:flex;align-items:center;justify-content:center;\r\n opacity:0;pointer-events:none;transition:opacity 0.25s;\r\n}\r\n.modal-overlay.open{opacity:1;pointer-events:all}\r\n.modal{\r\n background:var(--card);border:1px solid var(--border-bright);border-radius:10px;\r\n padding:28px 32px;width:90%;max-width:440px;\r\n box-shadow:0 0 40px rgba(0,255,240,0.08),0 8px 32px rgba(0,0,0,0.5);\r\n}\r\n.modal-title{\r\n font-family:var(--font-mono);font-size:1rem;font-weight:700;color:var(--cyan);\r\n letter-spacing:2px;margin-bottom:6px;\r\n text-shadow:0 0 8px rgba(0,255,240,0.3);\r\n}\r\n.modal-desc{color:var(--muted);font-size:0.8rem;margin-bottom:20px;line-height:1.5}\r\n.modal-input{\r\n width:100%;padding:10px 14px;background:var(--bg);border:1px solid var(--border);\r\n border-radius:6px;color:var(--text);font-family:var(--font-mono);font-size:0.85rem;\r\n outline:none;transition:border-color 0.2s;\r\n}\r\n.modal-input:focus{border-color:var(--cyan);box-shadow:0 0 8px rgba(0,255,240,0.15)}\r\n.modal-input::placeholder{color:var(--muted)}\r\n.modal-actions{display:flex;gap:10px;margin-top:18px;justify-content:flex-end}\r\n.modal-btn{\r\n font-family:var(--font-mono);font-size:0.8rem;padding:8px 18px;\r\n border-radius:4px;cursor:pointer;transition:all 0.2s;letter-spacing:0.5px;\r\n}\r\n.modal-btn-cancel{\r\n border:1px solid var(--border);background:transparent;color:var(--muted);\r\n}\r\n.modal-btn-cancel:hover{border-color:var(--muted);color:var(--text)}\r\n.modal-btn-submit{\r\n border:1px solid var(--cyan);background:rgba(0,255,240,0.12);color:var(--cyan);\r\n}\r\n.modal-btn-submit:hover{background:rgba(0,255,240,0.22);box-shadow:0 0 10px rgba(0,255,240,0.2)}\r\n.modal-btn-submit:disabled{opacity:0.4;cursor:not-allowed}\r\n.modal-status{\r\n margin-top:12px;padding:8px 12px;border-radius:4px;font-size:0.78rem;\r\n font-family:var(--font-mono);display:none;\r\n}\r\n.modal-status.success{display:block;background:rgba(0,255,136,0.1);border:1px solid rgba(0,255,136,0.3);color:var(--green)}\r\n.modal-status.error{display:block;background:rgba(255,51,102,0.1);border:1px solid rgba(255,51,102,0.3);color:var(--red)}\r\n.spinner{\r\n display:inline-block;width:12px;height:12px;border:2px solid transparent;\r\n border-top-color:var(--cyan);border-radius:50%;animation:spin 0.6s linear infinite;\r\n vertical-align:middle;margin-right:6px;\r\n}\r\n@keyframes spin{to{transform:rotate(360deg)}}\r\n\r\n/* ===== Tailscale Status Panel ===== */\r\n.ts-panel{\r\n margin-top:12px;padding:16px 20px;background:var(--card);border:1px solid var(--border);\r\n border-radius:8px;font-family:var(--font-mono);font-size:0.8rem;\r\n}\r\n.ts-panel-title{\r\n font-weight:700;letter-spacing:2px;text-transform:uppercase;margin-bottom:10px;\r\n display:flex;align-items:center;gap:8px;\r\n}\r\n.ts-panel-title.connected{color:var(--green)}\r\n.ts-panel-title.not-installed{color:var(--red)}\r\n.ts-panel-title.not-connected{color:var(--yellow)}\r\n.ts-info-row{display:flex;gap:8px;margin:4px 0;color:var(--muted)}\r\n.ts-info-row .label{min-width:100px;color:var(--muted)}\r\n.ts-info-row .value{color:var(--text)}\r\n.ts-install-guide{\r\n margin-top:12px;padding:12px 16px;background:rgba(0,0,0,0.3);border:1px solid var(--border);\r\n border-radius:6px;\r\n}\r\n.ts-install-guide .cmd{\r\n display:flex;align-items:center;justify-content:space-between;\r\n padding:6px 10px;background:rgba(255,255,255,0.04);border-radius:4px;\r\n margin:6px 0;font-size:0.8rem;cursor:pointer;transition:background 0.2s;\r\n}\r\n.ts-install-guide .cmd:hover{background:rgba(255,255,255,0.08)}\r\n.ts-install-guide .cmd code{color:var(--cyan)}\r\n.ts-install-guide .copy-hint{color:var(--muted);font-size:0.7rem}\r\n.ts-install-guide a{color:var(--cyan);text-decoration:none}\r\n.ts-install-guide a:hover{text-decoration:underline}\r\n.ts-setup-hint{\r\n margin-top:10px;padding:8px 12px;background:rgba(0,255,240,0.05);border:1px solid rgba(0,255,240,0.15);\r\n border-radius:4px;color:var(--cyan);font-size:0.78rem;\r\n}\r\n.btn-redetect{\r\n font-family:var(--font-mono);font-size:0.7rem;padding:4px 12px;\r\n border:1px solid var(--border-bright);border-radius:4px;background:transparent;\r\n color:var(--muted);cursor:pointer;transition:all 0.2s;margin-top:8px;\r\n}\r\n.btn-redetect:hover{border-color:var(--cyan);color:var(--cyan)}\r\n</style>\r\n</head>\r\n<body>\r\n<div id=\"app\">\r\n <header class=\"header\">\r\n <div class=\"header-title\">OPEN PARTY<span>// Dashboard</span></div>\r\n <div class=\"header-center\">\r\n <div class=\"header-status\">\r\n <div class=\"status-dot\" id=\"statusDot\"></div>\r\n <span id=\"statusText\">CONNECTING</span>\r\n </div>\r\n <div class=\"header-meta\" id=\"serverInfo\">--</div>\r\n </div>\r\n <div class=\"header-right\">\r\n UPTIME <span class=\"value\" id=\"uptime\">--:--:--</span>\r\n <button class=\"btn-join\" id=\"btnJoinNetwork\" title=\"Join Tailscale Network\">Join Network</button>\r\n </div>\r\n </header>\r\n\r\n <!-- Tailscale status panel (shown when not connected) -->\r\n <div id=\"tsPanel\" class=\"ts-panel\" style=\"display:none\"></div>\r\n\r\n <div class=\"stats-row\" id=\"statsRow\">\r\n <div class=\"stat-card cyan\">\r\n <div class=\"stat-value\" id=\"localAgentCount\">-</div>\r\n <div class=\"stat-label\">Local Agents</div>\r\n </div>\r\n <div class=\"stat-card green\">\r\n <div class=\"stat-value\" id=\"remoteAgentCount\">-</div>\r\n <div class=\"stat-label\">Remote Agents</div>\r\n </div>\r\n <div class=\"stat-card magenta\">\r\n <div class=\"stat-value\" id=\"peerCount\">-</div>\r\n <div class=\"stat-label\">Known Peers</div>\r\n </div>\r\n <div class=\"stat-card yellow\">\r\n <div class=\"stat-value\" id=\"partyServerCount\">-</div>\r\n <div class=\"stat-label\">Party Servers</div>\r\n </div>\r\n </div>\r\n\r\n <div class=\"main-grid\">\r\n <div class=\"section\">\r\n <div class=\"section-header\"><div class=\"dot\"></div>NETWORK TOPOLOGY</div>\r\n <div class=\"section-body\">\r\n <div class=\"topology-container\" id=\"topologyContainer\">\r\n <svg id=\"topologySvg\" viewBox=\"0 0 500 300\"></svg>\r\n </div>\r\n </div>\r\n </div>\r\n <div class=\"section\">\r\n <div class=\"section-header\"><div class=\"dot\" style=\"background:var(--green)\"></div>PEER HEALTH</div>\r\n <div class=\"section-body\" style=\"padding:0\">\r\n <div id=\"peerTableContainer\">\r\n <table class=\"peer-table\">\r\n <thead><tr><th>IP</th><th>Status</th><th>Fails</th><th>Last Probe</th></tr></thead>\r\n <tbody id=\"peerTableBody\"></tbody>\r\n </table>\r\n </div>\r\n </div>\r\n </div>\r\n </div>\r\n\r\n <div class=\"bottom-grid\">\r\n <div class=\"section\">\r\n <div class=\"section-header\"><div class=\"dot\" style=\"background:var(--cyan)\"></div>AGENT DIRECTORY</div>\r\n <div class=\"section-body\">\r\n <div class=\"agent-list\" id=\"agentList\"></div>\r\n </div>\r\n </div>\r\n <div class=\"section\">\r\n <div class=\"section-header\"><div class=\"dot\" style=\"background:var(--magenta)\"></div>MESSAGE FLOW</div>\r\n <div class=\"section-body\">\r\n <div class=\"msg-feed\" id=\"msgFeed\"></div>\r\n </div>\r\n </div>\r\n </div>\r\n\r\n <div class=\"footer\">OPEN PARTY v0.1 // DECENTRALIZED AGENT NETWORK</div>\r\n\r\n <!-- Join Network Modal -->\r\n <div class=\"modal-overlay\" id=\"joinModal\">\r\n <div class=\"modal\">\r\n <div class=\"modal-title\">JOIN TAILNET</div>\r\n <div class=\"modal-desc\">Enter your Tailscale auth key to join the network.<br>You can generate one from the Tailscale admin console.</div>\r\n <input type=\"password\" class=\"modal-input\" id=\"authKeyInput\" placeholder=\"tskey-auth-xxxxx...\" autocomplete=\"off\" spellcheck=\"false\" />\r\n <div class=\"modal-status\" id=\"joinStatus\"></div>\r\n <div class=\"modal-actions\">\r\n <button class=\"modal-btn modal-btn-cancel\" id=\"btnCancelJoin\">Cancel</button>\r\n <button class=\"modal-btn modal-btn-submit\" id=\"btnSubmitJoin\">Connect</button>\r\n </div>\r\n </div>\r\n </div>\r\n</div>\r\n\r\n<script>\r\n(function() {\r\n 'use strict';\r\n\r\n // ---- Helpers ----\r\n const $ = (s) => document.querySelector(s);\r\n const $$ = (s) => document.querySelectorAll(s);\r\n\r\n function formatUptime(seconds) {\r\n const h = Math.floor(seconds / 3600);\r\n const m = Math.floor((seconds % 3600) / 60);\r\n const s = seconds % 60;\r\n return String(h).padStart(2,'0') + ':' + String(m).padStart(2,'0') + ':' + String(s).padStart(2,'0');\r\n }\r\n\r\n function timeAgo(ts) {\r\n if (!ts) return '--';\r\n const diff = Math.floor(Date.now() / 1000 - ts);\r\n if (diff < 0) return 'now';\r\n if (diff < 60) return diff + 's ago';\r\n if (diff < 3600) return Math.floor(diff / 60) + 'm ago';\r\n return Math.floor(diff / 3600) + 'h ago';\r\n }\r\n\r\n function flashEl(el) {\r\n el.classList.remove('flash');\r\n void el.offsetWidth;\r\n el.classList.add('flash');\r\n }\r\n\r\n const statusColors = {\r\n PARTY_SERVER: '#00ff88',\r\n DEGRADED: '#ffaa00',\r\n SUSPECT: '#ff8800',\r\n DOWN: '#ff3366',\r\n UNKNOWN: '#6a6a8a',\r\n NOT_SERVER: '#6a6a8a',\r\n MAYBE: '#00fff0',\r\n };\r\n\r\n const statusBadgeClass = {\r\n PARTY_SERVER: 'badge-party',\r\n DEGRADED: 'badge-degraded',\r\n SUSPECT: 'badge-suspect',\r\n DOWN: 'badge-down',\r\n UNKNOWN: 'badge-unknown',\r\n NOT_SERVER: 'badge-not_server',\r\n MAYBE: 'badge-maybe',\r\n };\r\n\r\n // ---- State ----\r\n let overview = null;\r\n let prevStats = null;\r\n\r\n // ---- Fetch helpers ----\r\n async function fetchStats() {\r\n try {\r\n const r = await fetch('/dashboard/api/stats');\r\n return await r.json();\r\n } catch { return null; }\r\n }\r\n\r\n async function fetchOverview() {\r\n try {\r\n const r = await fetch('/dashboard/api/overview');\r\n return await r.json();\r\n } catch { return null; }\r\n }\r\n\r\n // ---- Render functions ----\r\n function renderHeader(data) {\r\n const s = data.server;\r\n if (tsState && tsState.state === 'connected') {\r\n $('#statusDot').className = 'status-dot';\r\n $('#statusText').textContent = 'ONLINE';\r\n $('#serverInfo').textContent = s.tailscale_ip + ' // ' + s.hostname;\r\n } else if (tsState && tsState.state === 'not_connected') {\r\n $('#statusDot').className = 'status-dot not-connected';\r\n $('#statusText').textContent = 'NOT CONNECTED';\r\n $('#serverInfo').textContent = 'Tailscale installed but not authenticated';\r\n } else if (tsState && tsState.state === 'not_installed') {\r\n $('#statusDot').className = 'status-dot not-installed';\r\n $('#statusText').textContent = 'NO TAILSCALE';\r\n $('#serverInfo').textContent = 'Tailscale not installed - local mode';\r\n } else {\r\n $('#statusDot').className = 'status-dot';\r\n $('#statusText').textContent = 'ONLINE';\r\n $('#serverInfo').textContent = s.tailscale_ip + ' // ' + s.hostname;\r\n }\r\n $('#uptime').textContent = formatUptime(s.uptime_seconds);\r\n }\r\n\r\n function renderStats(data) {\r\n const prev = {\r\n local: parseInt($('#localAgentCount').textContent) || 0,\r\n remote: parseInt($('#remoteAgentCount').textContent) || 0,\r\n peer: parseInt($('#peerCount').textContent) || 0,\r\n party: parseInt($('#partyServerCount').textContent) || 0,\r\n };\r\n const vals = {\r\n local: data.agents.local_count,\r\n remote: data.agents.remote_count,\r\n peer: data.peers.total,\r\n party: data.peers.party_servers,\r\n };\r\n if (vals.local !== prev.local) { $('#localAgentCount').textContent = vals.local; flashEl($('#localAgentCount')); }\r\n if (vals.remote !== prev.remote) { $('#remoteAgentCount').textContent = vals.remote; flashEl($('#remoteAgentCount')); }\r\n if (vals.peer !== prev.peer) { $('#peerCount').textContent = vals.peer; flashEl($('#peerCount')); }\r\n if (vals.party !== prev.party) { $('#partyServerCount').textContent = vals.party; flashEl($('#partyServerCount')); }\r\n }\r\n\r\n function renderTopology(data) {\r\n const svg = $('#topologySvg');\r\n const peers = data.peers.details || [];\r\n const cx = 250, cy = 150, radius = 100;\r\n\r\n let html = '';\r\n\r\n // Center node\r\n html += '<circle cx=\"' + cx + '\" cy=\"' + cy + '\" r=\"24\" fill=\"rgba(0,255,240,0.15)\" stroke=\"#00fff0\" stroke-width=\"2\">';\r\n html += '<animate attributeName=\"r\" values=\"24;26;24\" dur=\"3s\" repeatCount=\"indefinite\"/>';\r\n html += '</circle>';\r\n html += '<text x=\"' + cx + '\" y=\"' + cy + '\" text-anchor=\"middle\" dominant-baseline=\"central\" fill=\"#00fff0\" font-family=\"var(--font-mono)\" font-size=\"10\" font-weight=\"700\">SELF</text>';\r\n html += '<text x=\"' + cx + '\" y=\"' + (cy + 38) + '\" text-anchor=\"middle\" class=\"topo-label\">' + data.server.tailscale_ip + '</text>';\r\n\r\n if (peers.length === 0) {\r\n html += '<text x=\"' + cx + '\" y=\"' + (cy + 60) + '\" text-anchor=\"middle\" fill=\"#6a6a8a\" font-family=\"var(--font-mono)\" font-size=\"11\">No peers discovered</text>';\r\n }\r\n\r\n peers.forEach(function(p, i) {\r\n const angle = (2 * Math.PI * i / Math.max(peers.length, 1)) - Math.PI / 2;\r\n const px = cx + radius * Math.cos(angle);\r\n const py = cy + radius * Math.sin(angle);\r\n const color = statusColors[p.status] || '#6a6a8a';\r\n const opacity = (p.status === 'PARTY_SERVER' || p.status === 'DEGRADED' || p.status === 'SUSPECT') ? 1 : 0.4;\r\n\r\n // Connection line\r\n html += '<line x1=\"' + cx + '\" y1=\"' + cy + '\" x2=\"' + px + '\" y2=\"' + py + '\" stroke=\"' + color + '\" stroke-width=\"1\" opacity=\"' + (opacity * 0.3) + '\"/>';\r\n\r\n // Peer node\r\n html += '<g class=\"topo-node\"><circle cx=\"' + px + '\" cy=\"' + py + '\" r=\"16\" fill=\"rgba(255,255,255,0.03)\" stroke=\"' + color + '\" stroke-width=\"1.5\" opacity=\"' + opacity + '\"/>';\r\n html += '<text x=\"' + px + '\" y=\"' + py + '\" text-anchor=\"middle\" dominant-baseline=\"central\" fill=\"' + color + '\" font-family=\"var(--font-mono)\" font-size=\"8\" opacity=\"' + opacity + '\">' + p.ip.split('.').slice(-1)[0] + '</text></g>';\r\n\r\n // IP label\r\n html += '<text x=\"' + px + '\" y=\"' + (py + 26) + '\" text-anchor=\"middle\" class=\"topo-label\">' + p.ip + '</text>';\r\n });\r\n\r\n svg.innerHTML = html;\r\n }\r\n\r\n function renderPeerTable(data) {\r\n const tbody = $('#peerTableBody');\r\n const peers = data.peers.details || [];\r\n\r\n if (peers.length === 0) {\r\n tbody.innerHTML = '<tr><td colspan=\"4\" class=\"empty-state\">No peers discovered</td></tr>';\r\n return;\r\n }\r\n\r\n // Sort: PARTY_SERVER first, then by severity\r\n const order = { PARTY_SERVER: 0, DEGRADED: 1, SUSPECT: 2, MAYBE: 3, UNKNOWN: 4, NOT_SERVER: 5, DOWN: 6 };\r\n const sorted = [...peers].sort(function(a, b) { return (order[a.status] || 99) - (order[b.status] || 99); });\r\n\r\n tbody.innerHTML = sorted.map(function(p) {\r\n const badge = statusBadgeClass[p.status] || 'badge-unknown';\r\n const label = p.status === 'PARTY_SERVER' ? 'SERVER' : p.status === 'NOT_SERVER' ? 'NOT_SVR' : p.status;\r\n return '<tr>'\r\n + '<td>' + p.ip + '</td>'\r\n + '<td><span class=\"badge ' + badge + '\">' + label + '</span></td>'\r\n + '<td>' + p.consecutiveFailures + '</td>'\r\n + '<td>' + timeAgo(p.lastProbeAt) + '</td>'\r\n + '</tr>';\r\n }).join('');\r\n }\r\n\r\n function renderAgents(data) {\r\n const container = $('#agentList');\r\n const local = data.agents.local_agents || [];\r\n const remote = data.agents.remote_agents || [];\r\n const all = [\r\n ...local.map(function(a) { return { ...a, type: 'local' }; }),\r\n ...remote.map(function(a) { return { ...a, type: 'remote' }; }),\r\n ];\r\n\r\n if (all.length === 0) {\r\n container.innerHTML = '<div class=\"empty-state\">No agents registered</div>';\r\n return;\r\n }\r\n\r\n container.innerHTML = all.map(function(a) {\r\n const initials = (a.display_name || a.agent_id).substring(0, 2).toUpperCase();\r\n const tags = [];\r\n if (a.type === 'remote') {\r\n tags.push('<span class=\"agent-tag\">' + a.source_peer_ip + '</span>');\r\n if (!a.reachable) tags.push('<span class=\"agent-tag unreachable\">offline</span>');\r\n } else {\r\n tags.push('<span class=\"agent-tag\">local</span>');\r\n }\r\n return '<div class=\"agent-card ' + a.type + '\">'\r\n + '<div class=\"agent-icon\">' + initials + '</div>'\r\n + '<div class=\"agent-info\">'\r\n + '<div class=\"agent-name\">' + (a.display_name || a.agent_id) + '</div>'\r\n + '<div class=\"agent-id\">' + a.agent_id + '</div>'\r\n + '</div>'\r\n + '<div class=\"agent-meta\">' + tags.join('') + '</div>'\r\n + '</div>';\r\n }).join('');\r\n }\r\n\r\n function renderMessages(data) {\r\n const container = $('#msgFeed');\r\n const msgs = data.messages.recent || [];\r\n\r\n if (msgs.length === 0) {\r\n container.innerHTML = '<div class=\"empty-state\">No recent messages</div>';\r\n return;\r\n }\r\n\r\n container.innerHTML = msgs.map(function(m) {\r\n const dir = m.direction === 'received' ? 'received' : '';\r\n const arrow = m.direction === 'received' ? '&#8592;' : '&#8594;';\r\n const flow = m.direction === 'received'\r\n ? m.sender_id + ' <span class=\"arrow\">' + arrow + '</span> ' + (m.agent_id || '?')\r\n : (m.agent_id || '?') + ' <span class=\"arrow\">' + arrow + '</span> ' + (m.recipient_id || 'broadcast');\r\n return '<div class=\"msg-item ' + dir + '\">'\r\n + '<div class=\"msg-top\">'\r\n + '<div class=\"msg-flow\">' + flow + '</div>'\r\n + '<div class=\"msg-time\">' + timeAgo(m.timestamp) + '</div>'\r\n + '</div>'\r\n + '<div class=\"msg-content\">' + (m.summary || m.content) + '</div>'\r\n + '</div>';\r\n }).join('');\r\n }\r\n\r\n function renderAll(data) {\r\n renderHeader(data);\r\n renderStats(data);\r\n renderTopology(data);\r\n renderPeerTable(data);\r\n renderAgents(data);\r\n renderMessages(data);\r\n }\r\n\r\n // ---- Polling ----\r\n let fullTimer = null;\r\n let fastTimer = null;\r\n\r\n async function fullRefresh() {\r\n const data = await fetchOverview();\r\n if (!data) {\r\n $('#statusDot').className = 'status-dot offline';\r\n $('#statusText').textContent = 'OFFLINE';\r\n return;\r\n }\r\n overview = data;\r\n renderAll(data);\r\n }\r\n\r\n async function fastRefresh() {\r\n const stats = await fetchStats();\r\n if (!stats) return;\r\n const changed = JSON.stringify(stats) !== JSON.stringify(prevStats);\r\n prevStats = stats;\r\n if (changed) {\r\n fullRefresh();\r\n }\r\n }\r\n\r\n // ---- Init ----\r\n fullRefresh();\r\n fastTimer = setInterval(fastRefresh, 3000);\r\n fullTimer = setInterval(fullRefresh, 5000);\r\n\r\n // Clipboard click delegation for .cmd elements\r\n document.addEventListener('click', function(e) {\r\n var cmd = e.target.closest('.cmd');\r\n if (!cmd) return;\r\n var text = cmd.getAttribute('data-clipboard');\r\n if (text) {\r\n navigator.clipboard.writeText(text).then(function() {\r\n var hint = cmd.querySelector('.copy-hint');\r\n if (hint) hint.textContent = 'Copied!';\r\n });\r\n }\r\n });\r\n\r\n // Update uptime display every second\r\n setInterval(function() {\r\n if (overview && overview.server) {\r\n overview.server.uptime_seconds++;\r\n $('#uptime').textContent = formatUptime(overview.server.uptime_seconds);\r\n }\r\n }, 1000);\r\n\r\n // ---- Join Network Modal ----\r\n const joinModal = $('#joinModal');\r\n const btnJoin = $('#btnJoinNetwork');\r\n const btnCancel = $('#btnCancelJoin');\r\n const btnSubmit = $('#btnSubmitJoin');\r\n const authKeyInput = $('#authKeyInput');\r\n const joinStatus = $('#joinStatus');\r\n\r\n function openJoinModal() {\r\n joinStatus.className = 'modal-status';\r\n joinStatus.textContent = '';\r\n authKeyInput.value = '';\r\n joinModal.classList.add('open');\r\n setTimeout(function() { authKeyInput.focus(); }, 100);\r\n }\r\n\r\n function closeJoinModal() {\r\n joinModal.classList.remove('open');\r\n }\r\n\r\n btnJoin.addEventListener('click', openJoinModal);\r\n btnCancel.addEventListener('click', closeJoinModal);\r\n joinModal.addEventListener('click', function(e) {\r\n if (e.target === joinModal) closeJoinModal();\r\n });\r\n authKeyInput.addEventListener('keydown', function(e) {\r\n if (e.key === 'Enter') btnSubmit.click();\r\n if (e.key === 'Escape') closeJoinModal();\r\n });\r\n\r\n btnSubmit.addEventListener('click', async function() {\r\n const key = authKeyInput.value.trim();\r\n if (!key) {\r\n authKeyInput.focus();\r\n return;\r\n }\r\n btnSubmit.disabled = true;\r\n btnSubmit.innerHTML = '<span class=\"spinner\"></span>Connecting...';\r\n joinStatus.className = 'modal-status';\r\n joinStatus.textContent = '';\r\n\r\n try {\r\n const r = await fetch('/dashboard/api/join-network', {\r\n method: 'POST',\r\n headers: { 'Content-Type': 'application/json' },\r\n body: JSON.stringify({ auth_key: key }),\r\n });\r\n const data = await r.json();\r\n if (data.success) {\r\n joinStatus.className = 'modal-status success';\r\n joinStatus.textContent = 'Successfully joined network!';\r\n btnJoin.textContent = 'Connected';\r\n btnJoin.classList.add('connected');\r\n setTimeout(function() { closeJoinModal(); fullRefresh(); }, 1500);\r\n } else {\r\n joinStatus.className = 'modal-status error';\r\n joinStatus.textContent = data.output || 'Failed to join network';\r\n }\r\n } catch (e) {\r\n joinStatus.className = 'modal-status error';\r\n joinStatus.textContent = 'Network error: ' + (e.message || 'unknown');\r\n }\r\n btnSubmit.disabled = false;\r\n btnSubmit.textContent = 'Connect';\r\n });\r\n\r\n // Check initial Tailscale status (tri-state)\r\n let tsState = null;\r\n let tsInstallInfo = null;\r\n\r\n async function checkTailscaleStatus() {\r\n try {\r\n const r = await fetch('/dashboard/api/tailscale-status');\r\n tsState = await r.json();\r\n } catch { tsState = { state: 'not_installed', platform: 'unknown' }; }\r\n\r\n const dot = $('#statusDot');\r\n const text = $('#statusText');\r\n const btnJoin = $('#btnJoinNetwork');\r\n const panel = $('#tsPanel');\r\n\r\n if (tsState.state === 'connected') {\r\n dot.className = 'status-dot';\r\n text.textContent = 'ONLINE';\r\n btnJoin.textContent = 'Connected';\r\n btnJoin.classList.add('connected');\r\n panel.style.display = 'none';\r\n } else if (tsState.state === 'not_installed') {\r\n dot.className = 'status-dot not-installed';\r\n text.textContent = 'NOT INSTALLED';\r\n btnJoin.style.display = 'none';\r\n await renderNotInstalledPanel();\r\n } else {\r\n dot.className = 'status-dot not-connected';\r\n text.textContent = 'NOT CONNECTED';\r\n await renderNotConnectedPanel();\r\n }\r\n }\r\n\r\n async function fetchInstallInfo() {\r\n if (tsInstallInfo) return tsInstallInfo;\r\n try {\r\n const r = await fetch('/dashboard/api/tailscale-install-info');\r\n tsInstallInfo = await r.json();\r\n } catch { tsInstallInfo = null; }\r\n return tsInstallInfo;\r\n }\r\n\r\n async function renderNotInstalledPanel() {\r\n const info = await fetchInstallInfo();\r\n const panel = $('#tsPanel');\r\n let html = '<div class=\"ts-panel-title not-installed\">Tailscale Not Installed</div>';\r\n html += '<div class=\"ts-info-row\"><span class=\"label\">Status:</span><span class=\"value\" style=\"color:var(--red)\">Tailscale is not detected on this system</span></div>';\r\n\r\n if (info && info.commands && info.commands.length > 0) {\r\n html += '<div class=\"ts-install-guide\">';\r\n html += '<div style=\"color:var(--muted);margin-bottom:6px\">Install for ' + info.os + ':</div>';\r\n info.commands.forEach(function(cmd) {\r\n const display = info.needs_sudo ? 'sudo ' + cmd : cmd;\r\n html += '<div class=\"cmd\" data-clipboard=\"' + display.replace(/\"/g, '&quot;') + '\">';\r\n html += '<code>' + display + '</code><span class=\"copy-hint\">Click to copy</span></div>';\r\n });\r\n if (info.download_url) {\r\n html += '<div style=\"margin-top:8px\">Download: <a href=\"' + info.download_url + '\" target=\"_blank\">' + info.download_url + '</a></div>';\r\n }\r\n html += '</div>';\r\n }\r\n\r\n html += '<div class=\"ts-setup-hint\">Or run <code style=\"color:var(--cyan)\">npx open-party setup</code> for guided installation</div>';\r\n html += '<button class=\"btn-redetect\" onclick=\"window.__redetectTailscale()\">Re-detect</button>';\r\n panel.innerHTML = html;\r\n panel.style.display = 'block';\r\n }\r\n\r\n async function renderNotConnectedPanel() {\r\n const panel = $('#tsPanel');\r\n const btnJoin = $('#btnJoinNetwork');\r\n btnJoin.style.display = '';\r\n btnJoin.textContent = 'Join Network';\r\n btnJoin.classList.remove('connected');\r\n\r\n let html = '<div class=\"ts-panel-title not-connected\">Tailscale Not Connected</div>';\r\n html += '<div class=\"ts-info-row\"><span class=\"label\">Status:</span><span class=\"value\" style=\"color:var(--yellow)\">Installed but not authenticated</span></div>';\r\n html += '<div class=\"ts-setup-hint\">Run <code style=\"color:var(--cyan)\">npx open-party setup</code> to log in, or use the Join Network button to enter an Auth Key</div>';\r\n html += '<button class=\"btn-redetect\" onclick=\"window.__redetectTailscale()\">Re-detect</button>';\r\n panel.innerHTML = html;\r\n panel.style.display = 'block';\r\n }\r\n\r\n window.__redetectTailscale = async function() {\r\n const panel = $('#tsPanel');\r\n panel.innerHTML = '<div style=\"color:var(--muted);padding:12px\"><span class=\"spinner\"></span> Re-detecting Tailscale...</div>';\r\n try {\r\n await fetch('/dashboard/api/tailscale-detect', { method: 'POST' });\r\n } catch { /* ignore */ }\r\n await checkTailscaleStatus();\r\n fullRefresh();\r\n };\r\n\r\n checkTailscaleStatus();\r\n\r\n})();\r\n</script>\r\n</body>\r\n</html>`;\r\n","import { Hono } from 'hono';\r\nimport { sanitizeAgentList } from '../models.js';\r\nimport { DASHBOARD_HTML } from '../dashboard-html.js';\r\nimport { getTailnetHostname, getTailscaleInstallationStatus, resetTailscaleBinaryCache, getInstallInstructions, joinTailnet } from '../../infra/tailscale.js';\r\nimport { registry, messageQueue, discovery, getSelfIp, refreshSelfIp, STARTED_AT } from '../state.js';\r\n\r\nconst dashboardRoutes = new Hono();\r\n\r\n// Serve the dashboard HTML page\r\ndashboardRoutes.get('/', (c) => {\r\n return c.html(DASHBOARD_HTML);\r\n});\r\n\r\n// Lightweight stats endpoint for fast polling\r\ndashboardRoutes.get('/api/stats', async (c) => {\r\n const localAgents = registry.listAll();\r\n const remoteAgents = discovery.getReachableRemoteAgents();\r\n const peerStates = discovery.getPeerStates();\r\n const partyServers = peerStates.filter((p) => p.status === 'PARTY_SERVER' || p.status === 'DEGRADED' || p.status === 'SUSPECT');\r\n\r\n return c.json({\r\n local_agent_count: localAgents.length,\r\n remote_agent_count: remoteAgents.length,\r\n peer_count: peerStates.length,\r\n party_server_count: partyServers.length,\r\n });\r\n});\r\n\r\n// Full overview endpoint for complete refresh\r\ndashboardRoutes.get('/api/overview', async (c) => {\r\n let hostname = '127.0.0.1';\r\n try {\r\n hostname = getTailnetHostname();\r\n } catch {\r\n /* ignore */\r\n }\r\n\r\n const localAgents = sanitizeAgentList(registry.listAll());\r\n const remoteEntries = discovery.getRemoteAgentEntries();\r\n const peerStates = discovery.getPeerStates();\r\n\r\n // Collect recent messages from local agents\r\n const recentMessages: Array<{\r\n agent_id: string;\r\n direction: string;\r\n sender_id: string;\r\n recipient_id?: string;\r\n summary?: string;\r\n content: string;\r\n timestamp: number;\r\n }> = [];\r\n\r\n for (const agent of localAgents) {\r\n const history = messageQueue.getHistory(agent.agent_id, 5);\r\n for (const entry of history) {\r\n recentMessages.push({ agent_id: agent.agent_id, ...entry });\r\n }\r\n }\r\n // Sort by timestamp descending, cap at 20\r\n recentMessages.sort((a, b) => b.timestamp - a.timestamp);\r\n if (recentMessages.length > 20) recentMessages.length = 20;\r\n\r\n const partyServers = peerStates.filter(\r\n (p) => p.status === 'PARTY_SERVER' || p.status === 'DEGRADED' || p.status === 'SUSPECT',\r\n );\r\n\r\n return c.json({\r\n server: {\r\n status: 'ok',\r\n tailscale_ip: getSelfIp(),\r\n hostname,\r\n uptime_seconds: Math.floor((Date.now() - STARTED_AT) / 1000),\r\n },\r\n agents: {\r\n local_count: localAgents.length,\r\n remote_count: remoteEntries.length,\r\n local_agents: localAgents,\r\n remote_agents: remoteEntries.map((e) => ({\r\n ...sanitizeAgentList([e.agentInfo])[0],\r\n source_peer_ip: e.sourcePeerIp,\r\n reachable: e.reachable,\r\n })),\r\n },\r\n peers: {\r\n total: peerStates.length,\r\n party_servers: partyServers.length,\r\n down: peerStates.filter((p) => p.status === 'DOWN').length,\r\n unknown: peerStates.filter((p) => p.status === 'UNKNOWN' || p.status === 'MAYBE').length,\r\n details: peerStates,\r\n },\r\n messages: {\r\n recent: recentMessages,\r\n },\r\n });\r\n});\r\n\r\nexport { dashboardRoutes };\r\n\r\n// ---------------------------------------------------------------------------\r\n// Tailscale network management\r\n// ---------------------------------------------------------------------------\r\n\r\ndashboardRoutes.get('/api/tailscale-status', async (c) => {\r\n try {\r\n return c.json(getTailscaleInstallationStatus());\r\n } catch (e) {\r\n return c.json({ state: 'not_installed', platform: process.platform, error: (e as Error).message });\r\n }\r\n});\r\n\r\ndashboardRoutes.post('/api/tailscale-detect', async (c) => {\r\n resetTailscaleBinaryCache();\r\n const state = getTailscaleInstallationStatus();\r\n if (state.state === 'connected') {\r\n refreshSelfIp();\r\n }\r\n return c.json(state);\r\n});\r\n\r\ndashboardRoutes.get('/api/tailscale-install-info', async (c) => {\r\n return c.json(getInstallInstructions(process.platform));\r\n});\r\n\r\ndashboardRoutes.post('/api/join-network', async (c) => {\r\n try {\r\n const body = await c.req.json<{ auth_key?: string }>();\r\n const authKey = (body.auth_key ?? '').trim();\r\n if (!authKey) {\r\n return c.json({ success: false, output: 'auth_key is required' }, 400);\r\n }\r\n const result = joinTailnet(authKey);\r\n return c.json(result, result.success ? 200 : 500);\r\n } catch (e) {\r\n return c.json({ success: false, output: (e as Error).message }, 500);\r\n }\r\n});\r\n","import { Hono } from 'hono';\r\nimport { cors } from 'hono/cors';\r\nimport { serve } from '@hono/node-server';\r\nimport { existsSync, mkdirSync, writeFileSync, unlinkSync } from 'node:fs';\r\nimport { join, dirname } from 'node:path';\r\nimport { homedir } from 'node:os';\r\nimport { PARTY_PORT, CLEANUP_INTERVAL, HEARTBEAT_TIMEOUT } from './config.js';\r\nimport { getSelfIp, discovery } from './state.js';\r\nimport { agentRoutes } from './routes/agent.js';\r\nimport { proxyRoutes } from './routes/proxy.js';\r\nimport { dashboardRoutes } from './routes/dashboard.js';\r\n\r\n// ---------------------------------------------------------------------------\r\n// Background cleanup task\r\n// ---------------------------------------------------------------------------\r\n\r\nasync function periodicCleanup(): Promise<void> {\r\n // TODO: heartbeat mechanism ready then resume cleanup\r\n}\r\n\r\n// ---------------------------------------------------------------------------\r\n// App\r\n// ---------------------------------------------------------------------------\r\n\r\nconst app = new Hono();\r\n\r\napp.use('*', cors());\r\n\r\napp.route('/agent', agentRoutes);\r\napp.route('/proxy', proxyRoutes);\r\napp.route('/dashboard', dashboardRoutes);\r\n\r\n// ---------------------------------------------------------------------------\r\n// Entry point\r\n// ---------------------------------------------------------------------------\r\n\r\n// ---------------------------------------------------------------------------\r\n// PID file self-management (so `stop` and `status` always find us)\r\n// ---------------------------------------------------------------------------\r\n\r\nfunction pidFilePath(): string {\r\n const pluginData = process.env.CLAUDE_PLUGIN_DATA || '';\r\n if (pluginData) return join(pluginData, 'server.pid');\r\n return join(homedir(), '.open-party', 'server.pid');\r\n}\r\n\r\n// ---------------------------------------------------------------------------\r\n// Entry point\r\n// ---------------------------------------------------------------------------\r\n\r\nasync function main() {\r\n // Write PID file so CLI commands can find this process\r\n const pidPath = pidFilePath();\r\n mkdirSync(dirname(pidPath), { recursive: true });\r\n writeFileSync(pidPath, String(process.pid));\r\n\r\n console.log(`Starting Party Server on port ${PARTY_PORT} (Tailscale IP: ${getSelfIp()})`);\r\n\r\n // Ignore SIGHUP so the process survives when the parent terminal exits\r\n process.on('SIGHUP', () => {});\r\n\r\n const server = serve({ fetch: app.fetch, port: PARTY_PORT });\r\n\r\n // Start background tasks\r\n const discoveryPromise = discovery.runLoop();\r\n const cleanupPromise = periodicCleanup();\r\n\r\n // Graceful shutdown\r\n const shutdown = () => {\r\n console.log('\\nShutting down Party Server...');\r\n try { unlinkSync(pidPath); } catch { /* ignore */ }\r\n server.close();\r\n process.exit(0);\r\n };\r\n\r\n process.on('SIGINT', shutdown);\r\n process.on('SIGTERM', shutdown);\r\n\r\n await Promise.race([discoveryPromise, cleanupPromise]);\r\n}\r\n\r\nmain().catch((e) => {\r\n console.error('Fatal error:', e);\r\n process.exit(1);\r\n});\r\n\r\n// Catch unhandled errors that would silently crash the process\r\nprocess.on('uncaughtException', (e) => {\r\n console.error('Uncaught exception:', e);\r\n});\r\nprocess.on('unhandledRejection', (e) => {\r\n console.error('Unhandled rejection:', e);\r\n});\r\n","/**\n * Interactive setup wizard for Open Party.\n *\n * Steps:\n * 1. Detect Tailscale → install if missing → login\n * 2. Detect AI agents → install Open Party plugin\n */\n\nimport { createInterface } from 'node:readline';\nimport {\n getTailscaleInstallationStatus,\n resetTailscaleBinaryCache,\n getInstallInstructions,\n type TailscaleState,\n} from '../infra/tailscale.js';\nimport { installTailscale } from './tailscale-installer.js';\nimport { detectAgents, type DetectedAgent } from './agent-detector.js';\nimport { installPluginToAgent } from './agent-installer.js';\n\nconst rl = createInterface({ input: process.stdin, output: process.stdout });\n\nfunction prompt(question: string): Promise<string> {\n return new Promise((resolve) => rl.question(question, (answer) => resolve(answer.trim())));\n}\n\nfunction cyan(text: string): string {\n return `\\x1b[36m${text}\\x1b[0m`;\n}\nfunction green(text: string): string {\n return `\\x1b[32m${text}\\x1b[0m`;\n}\nfunction yellow(text: string): string {\n return `\\x1b[33m${text}\\x1b[0m`;\n}\nfunction red(text: string): string {\n return `\\x1b[31m${text}\\x1b[0m`;\n}\nfunction bold(text: string): string {\n return `\\x1b[1m${text}\\x1b[0m`;\n}\n\n// ---------------------------------------------------------------------------\n// Step 1: Tailscale\n// ---------------------------------------------------------------------------\n\nasync function stepTailscale(): Promise<void> {\n console.log(`\\n${bold(cyan('🔍 Step 1: Detecting Tailscale...'))}\\n`);\n\n const status = getTailscaleInstallationStatus();\n\n if (status.state === 'connected') {\n console.log(`${green('✅ Tailscale is connected!')}`);\n console.log(` IP: ${status.tailscale_ip} Hostname: ${status.hostname}`);\n return;\n }\n\n if (status.state === 'not_installed') {\n console.log(`${red('❌ Tailscale is not installed.')}`);\n await handleNotInstalled(status.platform);\n // Re-detect after possible installation\n const newStatus = getTailscaleInstallationStatus();\n if (newStatus.state === 'not_installed') {\n console.log(`\\n${yellow('⚠️ Tailscale still not detected. Please install manually and re-run setup.')}`);\n return;\n }\n if (newStatus.state === 'connected') {\n console.log(`\\n${green('✅ Tailscale is connected!')} IP: ${newStatus.tailscale_ip}`);\n return;\n }\n // not_connected → fall through to login\n await handleNotConnected(newStatus.binary);\n return;\n }\n\n // not_connected\n await handleNotConnected(status.binary);\n}\n\nasync function handleNotInstalled(platform: string): Promise<void> {\n const info = getInstallInstructions(platform);\n\n console.log(`\\nInstall Tailscale for ${bold(info.os)}:\\n`);\n\n if (info.commands.length > 0) {\n for (const cmd of info.commands) {\n const prefix = info.needs_sudo ? 'sudo ' : '';\n console.log(` ${cyan(prefix + cmd)}`);\n }\n }\n\n console.log(`\\n Download: ${info.download_url}`);\n\n const autoInstall = info.commands.length > 0 && platform !== 'win32'; // Windows winget may need user interaction\n if (autoInstall) {\n const answer = await prompt(`\\n${bold('Install Tailscale automatically?')} [yes/no]: `);\n if (answer.toLowerCase() === 'yes' || answer.toLowerCase() === 'y') {\n console.log('');\n const result = await installTailscale(platform);\n if (result.success) {\n console.log(`${green('✅ Tailscale installed successfully!')}`);\n } else {\n console.log(`${red('❌ Installation failed:')}\\n${result.output}`);\n console.log(`\\n Please install manually and re-run \\x1b[36mnpx open-party setup\\x1b[0m`);\n }\n return;\n }\n }\n\n console.log(`\\n Or run: ${cyan('npx open-party setup')} (after installing manually)`);\n if (autoInstall) {\n const cont = await prompt('Press Enter when installation is complete, or type \"skip\" to skip...');\n if (cont.toLowerCase() === 'skip') return;\n resetTailscaleBinaryCache();\n }\n}\n\nasync function handleNotConnected(binary: string): Promise<void> {\n console.log(`${yellow('🔒 Tailscale is installed but not connected.')}`);\n console.log('');\n\n console.log('Choose a login method:\\n');\n console.log(' 1. Interactive login (opens browser to authenticate)');\n console.log(' 2. Auth key (enter from Tailscale admin console)');\n console.log('');\n\n const choice = await prompt('Select [1/2]: ');\n\n if (choice === '1') {\n await handleInteractiveLogin(binary);\n } else {\n await handleAuthKeyLogin(binary);\n }\n}\n\nasync function handleInteractiveLogin(binary: string): Promise<void> {\n console.log(`\\n${cyan('Running interactive login...')}`);\n console.log('A browser window should open. Authenticate in the browser, then return here.\\n');\n\n const { spawn } = await import('node:child_process');\n const child = spawn(binary, ['login'], { stdio: 'inherit' });\n\n const exitCode = await new Promise<number | null>((resolve) => {\n child.on('close', resolve);\n });\n\n resetTailscaleBinaryCache();\n const status = getTailscaleInstallationStatus();\n\n if (exitCode === 0 && status.state === 'connected') {\n console.log(`\\n${green('✅ Login successful!')} IP: ${status.tailscale_ip}`);\n showAuthKeyTip();\n } else {\n console.log(`\\n${yellow('⚠️ Login may not have completed. Status: ' + status.state)}`);\n console.log(' Try running: npx open-party setup');\n }\n}\n\nasync function handleAuthKeyLogin(binary: string): Promise<void> {\n console.log('');\n console.log('Ask the network creator to generate an Auth Key at:');\n console.log(`${cyan(' https://login.tailscale.com/admin/settings/keys')}\\n`);\n\n const authKey = await prompt('Enter Auth Key: ');\n if (!authKey) {\n console.log(yellow('No auth key provided, skipping login.'));\n return;\n }\n\n const { joinTailnet } = await import('../infra/tailscale.js');\n const result = joinTailnet(authKey);\n\n if (result.success) {\n resetTailscaleBinaryCache();\n const status = getTailscaleInstallationStatus();\n console.log(`\\n${green('✅ Login successful!')} IP: ${status.state === 'connected' ? status.tailscale_ip : 'unknown'}`);\n showAuthKeyTip();\n } else {\n console.log(`\\n${red('❌ Login failed:')}\\n${result.output}`);\n console.log(' Check your auth key and try again.');\n }\n}\n\nfunction showAuthKeyTip(): void {\n console.log('');\n console.log(`${bold('💡 To share network access with teammates:')}`);\n console.log(' 1. Go to https://login.tailscale.com/admin/settings/keys');\n console.log(' 2. Generate an Auth Key');\n console.log(' 3. Share it with teammates — they can run: npx open-party setup');\n}\n\n// ---------------------------------------------------------------------------\n// Step 2: Agent plugin\n// ---------------------------------------------------------------------------\n\nasync function stepAgentPlugin(): Promise<void> {\n console.log(`\\n${bold(cyan('🔍 Step 2: Detecting AI agents in your environment...'))}\\n`);\n\n const agents = detectAgents();\n const detected = agents.filter((a) => a.detected);\n\n if (detected.length === 0) {\n console.log(yellow('No supported AI agents detected in this environment.'));\n console.log(' Supported agents: Claude Code, Cursor, Gemini CLI');\n console.log('');\n console.log(' Install one and re-run: npx open-party setup');\n return;\n }\n\n console.log('Detected agents:\\n');\n for (const agent of detected) {\n console.log(` ${green('✓')} ${agent.name}`);\n }\n console.log('');\n\n // Select which agents to install\n const selected = await selectAgents(detected);\n if (selected.length === 0) {\n console.log(yellow('No agents selected, skipping plugin installation.'));\n return;\n }\n\n for (const agent of selected) {\n console.log(`\\nInstalling Open Party plugin for ${agent.name}...`);\n const result = await installPluginToAgent(agent.type);\n if (result.success) {\n console.log(`${green('✅')} Plugin installed for ${agent.name}${result.configPath ? ` (${result.configPath})` : ''}`);\n if (result.warning) {\n console.log(` ${yellow('⚠️')} ${result.warning}`);\n }\n if (agent.type === 'claude-code') {\n console.log(` ${bold('Please restart Claude Code')} for changes to take effect.`);\n }\n } else {\n console.log(`${red('❌')} Failed to install for ${agent.name}: ${result.error}`);\n }\n }\n}\n\nasync function selectAgents(agents: DetectedAgent[]): Promise<DetectedAgent[]> {\n console.log('Select agents to install Open Party plugin:\\n');\n for (let i = 0; i < agents.length; i++) {\n console.log(` [${i + 1}] ${agents[i].name}`);\n }\n console.log(` [a] All`);\n console.log(` [n] None (skip)`);\n console.log('');\n\n const answer = await prompt('Enter selection (e.g. \"1 2\" or \"a\" or \"n\"): ');\n\n if (answer.toLowerCase() === 'n' || answer === '') return [];\n if (answer.toLowerCase() === 'a') return agents;\n\n const indices = answer\n .split(/[\\s,]+/)\n .map((s) => parseInt(s, 10) - 1)\n .filter((i) => i >= 0 && i < agents.length);\n\n return indices.map((i) => agents[i]);\n}\n\n// ---------------------------------------------------------------------------\n// Main\n// ---------------------------------------------------------------------------\n\nexport async function setupCommand(): Promise<void> {\n console.log(bold(cyan('\\n🚀 Open Party Setup Wizard\\n')));\n\n await stepTailscale();\n await stepAgentPlugin();\n\n // Start the Party Server in the background\n console.log(`\\n${bold(cyan('🚀 Starting Party Server...'))}`);\n const { spawn } = await import('node:child_process');\n const { resolve, dirname } = await import('node:path');\n const { fileURLToPath } = await import('node:url');\n\n // Find party-server.js relative to this file\n const __dirname = dirname(fileURLToPath(import.meta.url));\n const serverScript = resolve(__dirname, '..', 'party-server.js');\n\n const serverProc = spawn(process.execPath, [serverScript], {\n detached: true,\n stdio: 'ignore',\n windowsHide: true,\n });\n serverProc.unref();\n\n // Wait a moment for the server to start\n await new Promise((r) => setTimeout(r, 2000));\n\n console.log(`\\n${bold(green('🎉 Setup complete!'))}`);\n console.log(` Dashboard: http://127.0.0.1:8000/dashboard`);\n console.log(' Other agents can join with: npx @feynmanzhang/open-party setup\\n');\n\n rl.close();\n}","/**\n * Platform-specific Tailscale installer.\n *\n * Executes installation commands with real-time output (stdio: 'inherit').\n * On Linux/macOS, uses sudo when needed.\n */\n\nimport { spawn } from 'node:child_process';\n\nexport interface InstallResult {\n success: boolean;\n output: string;\n}\n\ntype PlatformEntry = {\n cmd: string;\n args: string[];\n needsSudo: boolean;\n};\n\nconst INSTALL_COMMANDS: Record<string, PlatformEntry> = {\n linux: { cmd: 'bash', args: ['-c', 'curl -fsSL https://tailscale.com/install.sh | sh'], needsSudo: true },\n darwin: { cmd: 'brew', args: ['install', 'tailscale'], needsSudo: false },\n win32: { cmd: 'winget', args: ['install', 'Tailscale.Tailscale', '--accept-source-agreements'], needsSudo: false },\n};\n\nasync function installTailscale(platform: string): Promise<InstallResult> {\n const entry = INSTALL_COMMANDS[platform];\n if (!entry) {\n return {\n success: false,\n output: `Unsupported platform: ${platform}. Please install manually from https://tailscale.com/download`,\n };\n }\n\n const cmd = entry.needsSudo ? 'sudo' : entry.cmd;\n const args = entry.needsSudo ? [entry.cmd, ...entry.args] : entry.args;\n\n console.log(`Running: ${cmd} ${args.join(' ')}\\n`);\n\n return new Promise((resolve) => {\n const child = spawn(cmd, args, {\n stdio: 'inherit',\n windowsHide: true,\n });\n\n let exited = false;\n\n child.on('close', (code) => {\n if (exited) return;\n exited = true;\n if (code === 0) {\n resolve({ success: true, output: 'Installation completed.' });\n } else {\n resolve({ success: false, output: `Installation exited with code ${code}` });\n }\n });\n\n child.on('error', (err) => {\n if (exited) return;\n exited = true;\n resolve({ success: false, output: err.message });\n });\n });\n}\n\nexport { installTailscale };","/**\n * Detect AI agents in the user's environment.\n *\n * Checks for Claude Code, Cursor, and Gemini CLI by looking for\n * configuration files and PATH executables.\n */\n\nimport { existsSync } from 'node:fs';\nimport { join } from 'node:path';\nimport { homedir } from 'node:os';\nimport { execSync } from 'node:child_process';\n\nexport type AgentType = 'claude-code' | 'cursor' | 'gemini-cli';\n\nexport interface DetectedAgent {\n type: AgentType;\n name: string;\n detected: boolean;\n configPath: string;\n}\n\nfunction isExecutableInPath(name: string): boolean {\n try {\n const which = process.platform === 'win32' ? 'where' : 'which';\n execSync(`${which} ${name}`, { timeout: 3000, stdio: 'pipe', windowsHide: true });\n return true;\n } catch {\n return false;\n }\n}\n\nfunction detectClaudeCode(): DetectedAgent {\n const configDir = join(homedir(), '.claude');\n const settingsPath = join(configDir, 'settings.json');\n return {\n type: 'claude-code',\n name: 'Claude Code',\n detected: existsSync(settingsPath) || isExecutableInPath('claude'),\n configPath: settingsPath,\n };\n}\n\nfunction detectCursor(): DetectedAgent {\n const configPath =\n process.platform === 'win32'\n ? join(homedir(), 'AppData', 'Roaming', 'Cursor', 'User', 'globalStorage', 'settings.json')\n : join(homedir(), '.cursor', 'mcp.json');\n // Check for .cursor/mcp.json (new location) or fallback\n const mcpPath = join(homedir(), '.cursor', 'mcp.json');\n return {\n type: 'cursor',\n name: 'Cursor',\n detected: existsSync(mcpPath) || isExecutableInPath('cursor'),\n configPath: mcpPath,\n };\n}\n\nfunction detectGeminiCli(): DetectedAgent {\n return {\n type: 'gemini-cli',\n name: 'Gemini CLI',\n detected: isExecutableInPath('gemini'),\n configPath: join(homedir(), '.gemini', 'settings.json'),\n };\n}\n\nexport function detectAgents(): DetectedAgent[] {\n return [detectClaudeCode(), detectCursor(), detectGeminiCli()];\n}","/**\r\n * Install Open Party plugin to detected AI agents.\r\n *\r\n * Each agent type has its own configuration mechanism:\r\n * - Claude Code: Full plugin registration (copy to plugins/cache, installed_plugins.json, enabledPlugins)\r\n * Claude Code auto-discovers MCP servers (.mcp.json), hooks (hooks/hooks.json), and skills (skills/STAR/SKILL.md)\r\n * from the plugin directory.\r\n * - Cursor: MCP server config in ~/.cursor/mcp.json\r\n * - Gemini CLI: MCP server config in ~/.gemini/settings.json\r\n */\r\n\r\nimport { existsSync, readFileSync, writeFileSync, mkdirSync, cpSync, rmSync, readdirSync } from 'node:fs';\r\nimport { join, dirname, resolve } from 'node:path';\r\nimport { homedir } from 'node:os';\r\nimport type { AgentType } from './agent-detector.js';\r\n\r\nexport interface InstallResult {\r\n success: boolean;\r\n configPath?: string;\r\n error?: string;\r\n warning?: string;\r\n}\r\n\r\n// ---------------------------------------------------------------------------\r\n// Shared helpers\r\n// ---------------------------------------------------------------------------\r\n\r\ninterface SettingsConfig {\r\n mcpServers?: Record<string, McpServerConfig>;\r\n enabledPlugins?: Record<string, boolean>;\r\n [key: string]: unknown;\r\n}\r\n\r\ninterface McpServerConfig {\r\n type?: string;\r\n command: string;\r\n args: string[];\r\n}\r\n\r\ninterface PluginsData {\r\n version?: number;\r\n plugins?: Record<string, unknown[]>;\r\n}\r\n\r\nfunction ensureDir(filePath: string): void {\r\n const dir = dirname(filePath);\r\n if (!existsSync(dir)) {\r\n mkdirSync(dir, { recursive: true });\r\n }\r\n}\r\n\r\nfunction readJsonFile<T>(filePath: string, fallback: T): T {\r\n if (!existsSync(filePath)) return fallback;\r\n try {\r\n return JSON.parse(readFileSync(filePath, 'utf-8')) as T;\r\n } catch {\r\n return fallback;\r\n }\r\n}\r\n\r\nfunction writeJsonFile(filePath: string, data: unknown): void {\r\n ensureDir(filePath);\r\n writeFileSync(filePath, JSON.stringify(data, null, 2) + '\\n');\r\n}\r\n\r\n// ---------------------------------------------------------------------------\r\n// Plugin directory discovery\r\n// ---------------------------------------------------------------------------\r\n\r\n/**\r\n * Find the built plugin directory under dist/claude-code/.\r\n * Returns the absolute path to the plugin root (e.g., dist/claude-code/open-party-0.1.1/)\r\n * or null if not found.\r\n */\r\nfunction findPluginDistDir(): string | null {\r\n const distDir = resolve(import.meta.dirname ?? '.', '..', 'claude-code');\r\n if (!existsSync(distDir)) return null;\r\n\r\n try {\r\n const entries = readdirSync(distDir);\r\n const dirs = entries.filter((e: string) => e.startsWith('open-party-'));\r\n if (dirs.length === 0) return null;\r\n\r\n // Use the latest version if multiple exist\r\n const pluginDir = join(distDir, dirs[dirs.length - 1]);\r\n if (existsSync(join(pluginDir, '.claude-plugin', 'plugin.json'))) {\r\n return pluginDir;\r\n }\r\n } catch {\r\n // ignore\r\n }\r\n return null;\r\n}\r\n\r\n/**\r\n * Find the dist/ directory containing compiled JS files (mcp-server.js, hook-handler.js).\r\n * Used when the full plugin directory isn't available but we still need the MCP server.\r\n */\r\nfunction findDistJsDir(): string | null {\r\n const possiblePaths = [\r\n // When installed globally via npm: <pkg-root>/dist/cli/index.js → <pkg-root>/dist/\r\n resolve(import.meta.dirname ?? '.', '..'),\r\n ];\r\n\r\n for (const p of possiblePaths) {\r\n if (existsSync(join(p, 'mcp-server.js')) && existsSync(join(p, 'hook-handler.js'))) {\r\n return p;\r\n }\r\n }\r\n return null;\r\n}\r\n\r\n/** Read plugin version from plugin.json */\r\nfunction getPluginVersion(): string {\r\n // Try to read from the built plugin directory first\r\n const pluginDir = findPluginDistDir();\r\n if (pluginDir) {\r\n const manifest = readJsonFile<{ version?: string }>(\r\n join(pluginDir, '.claude-plugin', 'plugin.json'),\r\n {},\r\n );\r\n if (manifest.version) return manifest.version;\r\n }\r\n\r\n // Fallback: try source directory\r\n const sourceManifest = resolve(\r\n import.meta.dirname ?? '.',\r\n '..',\r\n '..',\r\n 'src',\r\n 'client',\r\n 'claude-code',\r\n '.claude-plugin',\r\n 'plugin.json',\r\n );\r\n const manifest = readJsonFile<{ version?: string }>(sourceManifest, {});\r\n if (manifest.version) return manifest.version;\r\n\r\n return '0.1.0';\r\n}\r\n\r\n// ---------------------------------------------------------------------------\r\n// Claude Code — Full plugin registration\r\n//\r\n// Claude Code uses convention-based discovery for plugins:\r\n// - .claude-plugin/plugin.json → plugin metadata\r\n// - .mcp.json → MCP server config (uses ${CLAUDE_PLUGIN_ROOT})\r\n// - hooks/hooks.json → lifecycle hooks (uses ${CLAUDE_PLUGIN_ROOT})\r\n// - skills/*/SKILL.md → skill definitions\r\n//\r\n// We only need to:\r\n// 1. Copy plugin files to ~/.claude/plugins/cache/open-party/{version}/\r\n// 2. Register in installed_plugins.json\r\n// 3. Enable in settings.json (enabledPlugins)\r\n// 4. Remove any old manual MCP server entry (from previous installs)\r\n//\r\n// Claude Code handles the rest automatically at runtime by setting\r\n// CLAUDE_PLUGIN_ROOT and discovering the plugin's config files.\r\n// ---------------------------------------------------------------------------\r\n\r\ninterface MarketplaceSource {\r\n source: string;\r\n repo?: string;\r\n path?: string;\r\n}\r\n\r\ninterface MarketplaceEntry {\r\n source: MarketplaceSource;\r\n installLocation: string;\r\n lastUpdated: string;\r\n}\r\n\r\ninterface KnownMarketplaces {\r\n [name: string]: MarketplaceEntry;\r\n}\r\n\r\n// ---------------------------------------------------------------------------\r\n// Marketplace registration\r\n//\r\n// Claude Code requires each installed plugin to belong to a registered\r\n// marketplace. We create a \"open-party\" marketplace that mirrors the GitHub\r\n// repo structure. The marketplace directory contains:\r\n// - .claude-plugin/marketplace.json — marketplace manifest listing the plugin\r\n// - plugin/ — the actual plugin files (symlinked/copied from cache)\r\n//\r\n// The known_marketplaces.json entry uses the GitHub source format so Claude\r\n// Code recognizes it as a valid marketplace.\r\n// ---------------------------------------------------------------------------\r\n\r\n/** Directory for the open-party marketplace: ~/.claude/plugins/marketplaces/open-party/ */\r\nfunction getMarketplaceDir(): string {\r\n return join(homedir(), '.claude', 'plugins', 'marketplaces', 'open-party');\r\n}\r\n\r\n/**\r\n * Register the \"open-party\" marketplace so Claude Code can discover the plugin.\r\n * Creates:\r\n * - ~/.claude/plugins/marketplaces/open-party/.claude-plugin/marketplace.json\r\n * - ~/.claude/plugins/marketplaces/open-party/plugin/.claude-plugin/plugin.json\r\n * (plus other plugin files copied from cache)\r\n * - Entry in ~/.claude/plugins/known_marketplaces.json\r\n */\r\nfunction registerMarketplace(version: string, pluginDir: string): void {\r\n const marketplaceDir = getMarketplaceDir();\r\n const pluginSourceDir = join(marketplaceDir, 'plugin');\r\n\r\n // --- Create marketplace manifest ---\r\n const marketplacePluginDir = join(marketplaceDir, '.claude-plugin');\r\n if (!existsSync(marketplacePluginDir)) {\r\n mkdirSync(marketplacePluginDir, { recursive: true });\r\n }\r\n\r\n const marketplaceManifest: {\r\n name: string;\r\n owner: { name: string };\r\n metadata: { description: string; homepage: string };\r\n plugins: Array<{ name: string; version: string; source: string; description: string }>;\r\n } = {\r\n name: 'open-party',\r\n owner: { name: 'Feynman Zhang' },\r\n metadata: {\r\n description: 'Decentralized Agent communication network for Claude Code',\r\n homepage: 'https://github.com/FeynmanZhang/open-party',\r\n },\r\n plugins: [\r\n {\r\n name: 'open-party',\r\n version,\r\n source: './plugin',\r\n description: 'Decentralized Agent communication network for Claude Code',\r\n },\r\n ],\r\n };\r\n\r\n writeJsonFile(join(marketplacePluginDir, 'marketplace.json'), marketplaceManifest);\r\n\r\n // --- Copy plugin files to marketplace plugin directory ---\r\n if (existsSync(pluginSourceDir)) {\r\n rmSync(pluginSourceDir, { recursive: true });\r\n }\r\n mkdirSync(pluginSourceDir, { recursive: true });\r\n cpSync(pluginDir, pluginSourceDir, { recursive: true });\r\n\r\n // --- Register in known_marketplaces.json ---\r\n const knownMarketplacesPath = join(homedir(), '.claude', 'plugins', 'known_marketplaces.json');\r\n const knownMarketplaces = readJsonFile<KnownMarketplaces>(knownMarketplacesPath, {});\r\n\r\n knownMarketplaces['open-party'] = {\r\n source: {\r\n source: 'github',\r\n repo: 'FeynmanZhang/open-party',\r\n },\r\n installLocation: marketplaceDir,\r\n lastUpdated: new Date().toISOString(),\r\n };\r\n writeJsonFile(knownMarketplacesPath, knownMarketplaces);\r\n}\r\n\r\n/** Unregister the \"open-party\" marketplace and remove its directory. */\r\nfunction unregisterMarketplace(): void {\r\n // Remove from known_marketplaces.json\r\n const knownMarketplacesPath = join(homedir(), '.claude', 'plugins', 'known_marketplaces.json');\r\n const knownMarketplaces = readJsonFile<KnownMarketplaces>(knownMarketplacesPath, {});\r\n\r\n if (knownMarketplaces['open-party']) {\r\n delete knownMarketplaces['open-party'];\r\n writeJsonFile(knownMarketplacesPath, knownMarketplaces);\r\n }\r\n\r\n // Also clean up old \"local\" marketplace entry if it exists (migration)\r\n if (knownMarketplaces['local']) {\r\n delete knownMarketplaces['local'];\r\n writeJsonFile(knownMarketplacesPath, knownMarketplaces);\r\n }\r\n\r\n // Remove marketplace directory\r\n const marketplaceDir = getMarketplaceDir();\r\n if (existsSync(marketplaceDir)) {\r\n rmSync(marketplaceDir, { recursive: true });\r\n }\r\n\r\n // Also remove old local marketplace directory if it exists (migration)\r\n const oldLocalDir = join(homedir(), '.claude', 'plugins', 'marketplaces', 'local');\r\n if (existsSync(oldLocalDir)) {\r\n rmSync(oldLocalDir, { recursive: true });\r\n }\r\n}\r\n\r\nfunction installClaudeCode(): InstallResult {\r\n const pluginDir = findPluginDistDir();\r\n\r\n if (!pluginDir) {\r\n // No built plugin directory available — cannot do full plugin install\r\n return {\r\n success: false,\r\n error: 'Plugin package not found. Run \"npm run build:plugin\" first, or use \"claude --plugin-dir\" to install manually.',\r\n };\r\n }\r\n\r\n const version = getPluginVersion();\r\n\r\n // --- Step 1: Copy plugin files to ~/.claude/plugins/cache/open-party/open-party/{version}/ ---\r\n const installDir = join(homedir(), '.claude', 'plugins', 'cache', 'open-party', 'open-party', version);\r\n\r\n if (existsSync(installDir)) {\r\n rmSync(installDir, { recursive: true });\r\n }\r\n mkdirSync(installDir, { recursive: true });\r\n cpSync(pluginDir, installDir, { recursive: true });\r\n\r\n // Verify essential files exist after copy\r\n const mcpServerPath = join(installDir, 'dist', 'mcp-server.js');\r\n if (!existsSync(mcpServerPath)) {\r\n // The plugin dist/ subdirectory might be missing compiled files\r\n // Try to find them from the main dist/ and copy them in\r\n const distJsDir = findDistJsDir();\r\n if (distJsDir) {\r\n const targetDist = join(installDir, 'dist');\r\n if (!existsSync(targetDist)) mkdirSync(targetDist, { recursive: true });\r\n for (const file of ['mcp-server.js', 'hook-handler.js', 'party-server.js']) {\r\n const src = join(distJsDir, file);\r\n if (existsSync(src)) {\r\n cpSync(src, join(targetDist, file));\r\n }\r\n }\r\n for (const file of ['mcp-server.js.map', 'hook-handler.js.map', 'party-server.js.map']) {\r\n const src = join(distJsDir, file);\r\n if (existsSync(src)) {\r\n cpSync(src, join(targetDist, file));\r\n }\r\n }\r\n }\r\n }\r\n\r\n // Remove orphaned marker if present from a previous failed install\r\n const orphanedPath = join(installDir, 'dist', '.orphaned_at');\r\n if (existsSync(orphanedPath)) {\r\n rmSync(orphanedPath);\r\n }\r\n\r\n // --- Step 2: Register the open-party marketplace ---\r\n // Claude Code requires each plugin to belong to a registered marketplace.\r\n // We create an \"open-party\" marketplace that points to our GitHub repo\r\n // and contains the plugin files.\r\n registerMarketplace(version, pluginDir);\r\n\r\n // --- Step 3: Register in installed_plugins.json ---\r\n const pluginsJsonPath = join(homedir(), '.claude', 'plugins', 'installed_plugins.json');\r\n const pluginsData = readJsonFile<PluginsData>(\r\n pluginsJsonPath,\r\n { version: 2, plugins: {} },\r\n );\r\n if (!pluginsData.plugins) pluginsData.plugins = {};\r\n\r\n // Clean up old \"open-party@local\" entry if it exists (migration from previous version)\r\n if (pluginsData.plugins['open-party@local']) {\r\n delete pluginsData.plugins['open-party@local'];\r\n }\r\n\r\n pluginsData.plugins['open-party@open-party'] = [\r\n {\r\n scope: 'user',\r\n installPath: installDir,\r\n version,\r\n installedAt: new Date().toISOString(),\r\n lastUpdated: new Date().toISOString(),\r\n },\r\n ];\r\n writeJsonFile(pluginsJsonPath, pluginsData);\r\n\r\n // --- Step 4: Enable plugin & clean up old MCP entry in settings.json ---\r\n const settingsPath = join(homedir(), '.claude', 'settings.json');\r\n const settings = readJsonFile<SettingsConfig>(settingsPath, {});\r\n\r\n // Remove old manual MCP server entry (from previous install method)\r\n // The MCP server is now auto-discovered from the plugin's .mcp.json\r\n if (settings.mcpServers?.['open-party']) {\r\n delete settings.mcpServers['open-party'];\r\n if (Object.keys(settings.mcpServers).length === 0) {\r\n delete settings.mcpServers;\r\n }\r\n }\r\n\r\n // Enable the plugin\r\n if (!settings.enabledPlugins) {\r\n settings.enabledPlugins = {};\r\n }\r\n\r\n // Clean up old \"open-party@local\" entry (migration from previous version)\r\n if (settings.enabledPlugins['open-party@local'] !== undefined) {\r\n delete settings.enabledPlugins['open-party@local'];\r\n }\r\n\r\n settings.enabledPlugins['open-party@open-party'] = true;\r\n\r\n writeJsonFile(settingsPath, settings);\r\n\r\n return {\r\n success: true,\r\n configPath: settingsPath,\r\n };\r\n}\r\n\r\n/** Remove Open Party plugin from Claude Code completely */\r\nexport function uninstallClaudeCode(): InstallResult {\r\n const version = getPluginVersion();\r\n\r\n // Remove from settings.json\r\n const settingsPath = join(homedir(), '.claude', 'settings.json');\r\n const settings = readJsonFile<SettingsConfig>(settingsPath, {});\r\n\r\n // Remove old MCP server entry (if present)\r\n if (settings.mcpServers?.['open-party']) {\r\n delete settings.mcpServers['open-party'];\r\n if (Object.keys(settings.mcpServers).length === 0) {\r\n delete settings.mcpServers;\r\n }\r\n }\r\n\r\n // Remove enabled plugin entries (both old and new format)\r\n if (settings.enabledPlugins?.['open-party@open-party'] !== undefined) {\r\n delete settings.enabledPlugins['open-party@open-party'];\r\n }\r\n if (settings.enabledPlugins?.['open-party@local'] !== undefined) {\r\n delete settings.enabledPlugins['open-party@local'];\r\n }\r\n if (settings.enabledPlugins && Object.keys(settings.enabledPlugins).length === 0) {\r\n delete settings.enabledPlugins;\r\n }\r\n\r\n writeJsonFile(settingsPath, settings);\r\n\r\n // Remove from installed_plugins.json\r\n const pluginsJsonPath = join(homedir(), '.claude', 'plugins', 'installed_plugins.json');\r\n const pluginsData = readJsonFile<PluginsData>(\r\n pluginsJsonPath,\r\n { version: 2, plugins: {} },\r\n );\r\n if (pluginsData.plugins) {\r\n delete pluginsData.plugins['open-party@open-party'];\r\n delete pluginsData.plugins['open-party@local']; // Clean up old format too\r\n writeJsonFile(pluginsJsonPath, pluginsData);\r\n }\r\n\r\n // Remove plugin files (both old and new path formats)\r\n const installDirNew = join(homedir(), '.claude', 'plugins', 'cache', 'open-party', 'open-party', version);\r\n const installDirOld = join(homedir(), '.claude', 'plugins', 'cache', 'open-party', version);\r\n for (const dir of [installDirNew, installDirOld]) {\r\n if (existsSync(dir)) {\r\n rmSync(dir, { recursive: true });\r\n }\r\n }\r\n\r\n // Unregister the marketplace\r\n unregisterMarketplace();\r\n\r\n return { success: true, configPath: settingsPath };\r\n}\r\n\r\n// ---------------------------------------------------------------------------\r\n// Cursor — MCP server config only\r\n// ---------------------------------------------------------------------------\r\n\r\nfunction installCursor(): InstallResult {\r\n const configPath = join(homedir(), '.cursor', 'mcp.json');\r\n const { command, args } = getPluginCommand();\r\n\r\n const config = readJsonFile<SettingsConfig>(configPath, {});\r\n if (!config.mcpServers) config.mcpServers = {};\r\n\r\n config.mcpServers['open-party'] = { type: 'stdio', command, args };\r\n writeJsonFile(configPath, config);\r\n\r\n return { success: true, configPath };\r\n}\r\n\r\n// ---------------------------------------------------------------------------\r\n// Gemini CLI — MCP server config only\r\n// ---------------------------------------------------------------------------\r\n\r\nfunction installGeminiCli(): InstallResult {\r\n const configPath = join(homedir(), '.gemini', 'settings.json');\r\n const { command, args } = getPluginCommand();\r\n\r\n const config = readJsonFile<SettingsConfig>(configPath, {});\r\n if (!config.mcpServers) config.mcpServers = {};\r\n\r\n config.mcpServers['open-party'] = { type: 'stdio', command, args };\r\n writeJsonFile(configPath, config);\r\n\r\n return { success: true, configPath };\r\n}\r\n\r\n// ---------------------------------------------------------------------------\r\n// Shared MCP command resolution (for Cursor & Gemini CLI)\r\n// ---------------------------------------------------------------------------\r\n\r\nfunction getPluginCommand(): { command: string; args: string[] } {\r\n const distDir = findDistJsDir();\r\n if (distDir) {\r\n const serverPath = join(distDir, 'mcp-server.js');\r\n if (existsSync(serverPath)) {\r\n return { command: 'node', args: [serverPath] };\r\n }\r\n }\r\n\r\n // Fallback: use npx to run open-party as MCP server\r\n return { command: 'npx', args: ['@feynmanzhang/open-party', 'mcp'] };\r\n}\r\n\r\n// ---------------------------------------------------------------------------\r\n// Public API\r\n// ---------------------------------------------------------------------------\r\n\r\nexport async function installPluginToAgent(agentType: AgentType): Promise<InstallResult> {\r\n switch (agentType) {\r\n case 'claude-code':\r\n return installClaudeCode();\r\n case 'cursor':\r\n return installCursor();\r\n case 'gemini-cli':\r\n return installGeminiCli();\r\n default:\r\n return { success: false, error: `Unknown agent type: ${agentType}` };\r\n }\r\n}","/**\n * Shared utilities for CLI server lifecycle management.\n *\n * Provides PID file management, process detection, health checks,\n * server spawning, and argument parsing for the open-party CLI.\n */\n\nimport { spawn, execSync } from 'node:child_process';\nimport { existsSync, readFileSync, writeFileSync, unlinkSync, mkdirSync, openSync } from 'node:fs';\nimport { join, dirname, resolve } from 'node:path';\nimport { homedir } from 'node:os';\nimport { fileURLToPath } from 'node:url';\n\n// ---------------------------------------------------------------------------\n// Paths\n// ---------------------------------------------------------------------------\n\nconst __dirname = dirname(fileURLToPath(import.meta.url));\n\nexport function pidFilePath(): string {\n const pluginData = process.env.CLAUDE_PLUGIN_DATA || '';\n if (pluginData) return join(pluginData, 'server.pid');\n return join(homedir(), '.open-party', 'server.pid');\n}\n\nexport function logFilePath(): string {\n const pluginData = process.env.CLAUDE_PLUGIN_DATA || '';\n if (pluginData) return join(pluginData, 'server.log');\n return join(homedir(), '.open-party', 'server.log');\n}\n\nfunction serverScriptPath(): string {\n // When bundled, __dirname is dist/cli/ — party-server.js is at dist/party-server.js\n return resolve(__dirname, '..', 'party-server.js');\n}\n\n// ---------------------------------------------------------------------------\n// PID file management\n// ---------------------------------------------------------------------------\n\nexport function readPid(): number | null {\n const path = pidFilePath();\n if (!existsSync(path)) return null;\n try {\n return parseInt(readFileSync(path, 'utf-8').trim(), 10);\n } catch {\n return null;\n }\n}\n\nexport function writePid(pid: number): void {\n const path = pidFilePath();\n const dir = dirname(path);\n if (!existsSync(dir)) mkdirSync(dir, { recursive: true });\n writeFileSync(path, String(pid));\n}\n\nexport function removePidFile(): void {\n try { unlinkSync(pidFilePath()); } catch { /* ignore */ }\n}\n\n// ---------------------------------------------------------------------------\n// Process detection\n// ---------------------------------------------------------------------------\n\nexport function isProcessRunning(pid: number): boolean {\n if (process.platform === 'win32') {\n try {\n const output = execSync(`tasklist /FI \"PID eq ${pid}\" /NH`, {\n encoding: 'utf-8',\n windowsHide: true,\n stdio: ['pipe', 'pipe', 'pipe'],\n });\n return output.includes(String(pid));\n } catch {\n return false;\n }\n }\n try {\n process.kill(pid, 0);\n return true;\n } catch {\n return false;\n }\n}\n\n// ---------------------------------------------------------------------------\n// Health checks\n// ---------------------------------------------------------------------------\n\nexport function resolvePort(): number {\n return parseInt(process.env.PARTY_PORT || '8000', 10);\n}\n\nasync function fetchJson(url: string, timeoutMs: number = 2000): Promise<Record<string, unknown> | null> {\n try {\n const controller = new AbortController();\n const timer = setTimeout(() => controller.abort(), timeoutMs);\n const resp = await fetch(url, { signal: controller.signal });\n clearTimeout(timer);\n if (!resp.ok) return null;\n return (await resp.json()) as Record<string, unknown>;\n } catch {\n return null;\n }\n}\n\nexport async function isServerHealthy(port?: number): Promise<boolean> {\n const p = port ?? resolvePort();\n const data = await fetchJson(`http://127.0.0.1:${p}/proxy/health`);\n return data !== null && data.status === 'ok';\n}\n\nexport async function getServerHealth(port?: number): Promise<Record<string, unknown> | null> {\n const p = port ?? resolvePort();\n return fetchJson(`http://127.0.0.1:${p}/proxy/health`);\n}\n\nexport async function getServerOverview(port?: number): Promise<Record<string, unknown> | null> {\n const p = port ?? resolvePort();\n return fetchJson(`http://127.0.0.1:${p}/dashboard/api/overview`, 3000);\n}\n\n// ---------------------------------------------------------------------------\n// Server spawning\n// ---------------------------------------------------------------------------\n\nexport async function spawnServerInBackground(port: number): Promise<{ pid: number; ok: boolean }> {\n const script = serverScriptPath();\n if (!existsSync(script)) {\n console.error(`Server script not found: ${script}`);\n return { pid: 0, ok: false };\n }\n\n // Redirect stdout/stderr to log file\n const logPath = logFilePath();\n mkdirSync(dirname(logPath), { recursive: true });\n const logFd = openSync(logPath, 'a');\n\n const env = { ...process.env, PARTY_PORT: String(port) };\n const proc = spawn(process.execPath, [script], {\n stdio: ['ignore', logFd, logFd],\n detached: true,\n windowsHide: true,\n env,\n });\n proc.unref();\n\n const pid = proc.pid!;\n writePid(pid);\n\n proc.on('error', (err) => {\n console.error(`Failed to start server: ${err.message}`);\n });\n\n return { pid, ok: true };\n}\n\nexport async function waitForServerReady(port: number, timeoutMs: number = 10_000): Promise<boolean> {\n const deadline = Date.now() + timeoutMs;\n while (Date.now() < deadline) {\n if (await isServerHealthy(port)) return true;\n\n // Check if the spawned process is still alive (only if we have a PID)\n const pid = readPid();\n if (pid !== null && !isProcessRunning(pid)) {\n return false; // Process crashed\n }\n\n await sleep(500);\n }\n return false;\n}\n\n// ---------------------------------------------------------------------------\n// Process killing\n// ---------------------------------------------------------------------------\n\nexport function killServer(pid: number): void {\n try {\n if (process.platform === 'win32') {\n execSync(`taskkill /F /T /PID ${pid}`, { stdio: 'ignore', windowsHide: true });\n } else {\n process.kill(pid, 'SIGTERM');\n }\n } catch { /* process already gone */ }\n}\n\n// ---------------------------------------------------------------------------\n// Argument parsing\n// ---------------------------------------------------------------------------\n\nexport interface StartOptions {\n daemon: boolean;\n port: number | null; // null = use env/default\n}\n\nexport function parseStartArgs(args: string[]): StartOptions {\n let daemon = false;\n let port: number | null = null;\n\n for (let i = 0; i < args.length; i++) {\n if (args[i] === '-d' || args[i] === '--daemon') {\n daemon = true;\n } else if (args[i] === '-p' || args[i] === '--port') {\n const val = args[++i];\n if (val) port = parseInt(val, 10);\n } else if (args[i].startsWith('--port=')) {\n port = parseInt(args[i].split('=')[1], 10);\n }\n }\n\n return { daemon, port };\n}\n\n// ---------------------------------------------------------------------------\n// Helpers\n// ---------------------------------------------------------------------------\n\nfunction sleep(ms: number): Promise<void> {\n return new Promise((resolve) => setTimeout(resolve, ms));\n}","/**\n * Start the Party Server.\n *\n * Supports foreground mode (default) and daemon mode (--daemon / -d).\n * Use --port / -p to override PARTY_PORT for a single invocation.\n */\n\nimport {\n parseStartArgs,\n resolvePort,\n isServerHealthy,\n readPid,\n isProcessRunning,\n removePidFile,\n spawnServerInBackground,\n waitForServerReady,\n logFilePath,\n writePid,\n} from './server-utils.js';\n\nexport async function startServer(args: string[] = []): Promise<void> {\n const opts = parseStartArgs(args);\n const port = opts.port ?? resolvePort();\n\n if (opts.daemon) {\n await startDaemon(port);\n } else {\n await startForeground();\n }\n}\n\n// ---------------------------------------------------------------------------\n// Foreground mode — server runs in current process\n// ---------------------------------------------------------------------------\n\nasync function startForeground(): Promise<void> {\n // Write PID file so `stop` and `status` can find this process\n writePid(process.pid);\n\n // Clean up PID file on any exit\n process.on('exit', () => {\n removePidFile();\n });\n\n // Dynamic import — the server module self-starts on import\n await import('../server/index.js');\n}\n\n// ---------------------------------------------------------------------------\n// Daemon mode — server runs in a detached background process\n// ---------------------------------------------------------------------------\n\nasync function startDaemon(port: number): Promise<void> {\n // Already running?\n if (await isServerHealthy(port)) {\n const pid = readPid();\n console.log(`Party Server is already running (PID ${pid ?? 'unknown'}, port ${port}).`);\n process.exit(0);\n }\n\n // Clean up stale PID file\n const existingPid = readPid();\n if (existingPid !== null && !isProcessRunning(existingPid)) {\n removePidFile();\n }\n\n // Spawn\n const { pid, ok } = await spawnServerInBackground(port);\n if (!ok) {\n process.exit(1);\n }\n\n // Wait for ready\n console.log(`Starting Party Server in background (PID ${pid})...`);\n const ready = await waitForServerReady(port);\n\n if (ready) {\n console.log(`Party Server is running on port ${port}.`);\n console.log(` Dashboard: http://127.0.0.1:${port}/dashboard`);\n console.log(` Logs: ${logFilePath()}`);\n console.log(` Use 'open-party stop' to stop the server.`);\n } else {\n console.error('Party Server failed to start within timeout.');\n console.error(`Check logs: ${logFilePath()}`);\n process.exit(1);\n }\n}","/**\n * Stop the Party Server.\n *\n * Reads the PID file, verifies the process is alive, kills it,\n * and cleans up the PID file.\n */\n\nimport { readPid, isProcessRunning, killServer, removePidFile, isServerHealthy, resolvePort } from './server-utils.js';\n\nexport async function stopServer(): Promise<void> {\n const pid = readPid();\n\n if (pid === null) {\n // No PID file — check if a server is still responding\n const port = resolvePort();\n const healthy = await isServerHealthy(port);\n if (healthy) {\n console.log(`No PID file found, but a server is responding on port ${port}.`);\n console.log('It may have been started manually. Kill it by port or process name.');\n } else {\n console.log('Party Server is not running (no PID file found).');\n }\n return;\n }\n\n if (!isProcessRunning(pid)) {\n // Stale PID file — process is dead\n console.log(`Stale PID file found (PID ${pid} is not running). Cleaning up.`);\n removePidFile();\n return;\n }\n\n // Process is alive — kill it\n console.log(`Stopping Party Server (PID ${pid})...`);\n killServer(pid);\n removePidFile();\n\n // Verify it's actually down\n const port = resolvePort();\n const stillUp = await isServerHealthy(port);\n if (stillUp) {\n console.warn(`Process ${pid} was killed, but port ${port} is still responding.`);\n console.warn('Another process may be using this port.');\n } else {\n console.log('Party Server stopped.');\n }\n}","/**\n * Show the status of the Party Server.\n *\n * Displays PID, port, uptime, agent count, and Tailscale info\n * when the server is running. Handles four states:\n * 1. Running & healthy\n * 2. Process alive but not yet healthy (starting up)\n * 3. Stale PID file (process dead)\n * 4. Not running (no PID file)\n */\n\nimport {\n readPid,\n isProcessRunning,\n isServerHealthy,\n getServerHealth,\n getServerOverview,\n resolvePort,\n} from './server-utils.js';\n\nexport async function statusCommand(): Promise<void> {\n const port = resolvePort();\n const pid = readPid();\n\n let processAlive = false;\n if (pid !== null) {\n processAlive = isProcessRunning(pid);\n }\n\n const healthy = await isServerHealthy(port);\n\n if (healthy) {\n const health = await getServerHealth(port);\n const overview = await getServerOverview(port);\n\n console.log('Party Server is running.');\n console.log(` PID: ${pid ?? 'unknown (no PID file)'}`);\n console.log(` Port: ${port}`);\n console.log(` Tailscale IP: ${(health as any)?.tailscale_ip ?? 'N/A'}`);\n console.log(` Hostname: ${(health as any)?.hostname ?? 'N/A'}`);\n\n if (overview) {\n const server = overview.server as Record<string, unknown> | undefined;\n const agents = overview.agents as Record<string, unknown> | undefined;\n if (server?.uptime_seconds != null) {\n const uptime = server.uptime_seconds as number;\n const mins = Math.floor(uptime / 60);\n const secs = Math.floor(uptime % 60);\n console.log(` Uptime: ${mins}m ${secs}s`);\n }\n console.log(` Local agents: ${(agents?.local_count as number) ?? 'N/A'}`);\n console.log(` Remote agents: ${(agents?.remote_count as number) ?? 'N/A'}`);\n } else {\n console.log(` Local agents: ${(health as any)?.agent_count ?? 'N/A'}`);\n }\n\n console.log(` Dashboard: http://127.0.0.1:${port}/dashboard`);\n } else if (processAlive && pid !== null) {\n // PID file has a live process but health check fails\n console.log('Party Server process exists but is not responding on health endpoint.');\n console.log(` PID: ${pid}`);\n console.log(' The server may be starting up or has crashed.');\n console.log(` Logs: ~/.open-party/server.log`);\n } else if (pid !== null) {\n console.log('Party Server is NOT running (stale PID file).');\n console.log(` PID file references PID ${pid}, which is not a live process.`);\n console.log(' Use: open-party start to start the server.');\n } else {\n console.log('Party Server is NOT running.');\n console.log(' No PID file found.');\n console.log(' Use: open-party start to start the server.');\n }\n}","/**\n * Open Party CLI — decentralized Agent communication network.\n *\n * Usage:\n * open-party start [-d] [-p <port>] Start the Party Server (default)\n * open-party stop Stop the Party Server\n * open-party status Show server status\n * open-party setup Interactive setup wizard\n * open-party help Show help\n */\n\nimport { setupCommand } from './setup.js';\nimport { startServer } from './start-server.js';\nimport { stopServer } from './stop-server.js';\nimport { statusCommand } from './status.js';\n\nfunction showHelp(): void {\n console.log(`Usage: open-party <command> [options]\n\nCommands:\n start Start the Party Server (default when no command given)\n stop Stop the Party Server\n status Show server status\n setup Interactive setup wizard (Tailscale + agent plugins)\n help Show this help message\n\nOptions for 'start':\n -d, --daemon Run in background (daemon mode)\n -p, --port <port> Override port (default: 8000, env: PARTY_PORT)\n\nExamples:\n open-party Start server in foreground\n open-party start Start server in foreground\n open-party start -d Start server in background\n open-party start -d -p 9000 Start server in background on port 9000\n open-party stop Stop the server\n open-party status Check if the server is running`);\n}\n\nconst args = process.argv.slice(2);\nconst command = args[0] ?? 'start';\nconst commandArgs = args.slice(1);\n\nasync function main(): Promise<void> {\n switch (command) {\n case 'setup':\n await setupCommand();\n break;\n case 'start':\n await startServer(commandArgs);\n break;\n case 'stop':\n await stopServer();\n break;\n case 'status':\n await statusCommand();\n break;\n case 'help':\n case '--help':\n case '-h':\n showHelp();\n break;\n default:\n console.log(`Unknown command: ${command}\\n`);\n showHelp();\n process.exit(1);\n }\n}\n\nmain().catch((e) => {\n console.error('Fatal error:', e);\n process.exit(1);\n});"],"mappings":"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AAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;A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","names":["raw","match","cacheKey","raw","args","init_constants","init_constants","args","app","init_router","init_router","init_router","init_router","init_router","Node","init_node","_Node","init_router","init_node","Node","init_router","Hono","Request","init_dist","resolve","sleep","resolve","Hono","registry","discovery","messageQueue","Hono","Hono","mkdirSync","writeFileSync","unlinkSync","join","dirname","homedir","pidFilePath","init_dist","Hono","args","resolve","existsSync","join","execSync","existsSync","join","homedir","manifest","join","homedir","existsSync","join","homedir","command","args","existsSync","resolve","spawn","joinTailnet","dirname","fileURLToPath","__dirname","spawn","execSync","existsSync","readFileSync","writeFileSync","mkdirSync","join","dirname","resolve","homedir","args","args","pid","port","main"]}
1
+ {"version":3,"sources":["../../src/infra/tailscale.ts","../../src/cli/tailscale-installer.ts","../../node_modules/hono/dist/compose.js","../../node_modules/hono/dist/http-exception.js","../../node_modules/hono/dist/request/constants.js","../../node_modules/hono/dist/utils/body.js","../../node_modules/hono/dist/utils/url.js","../../node_modules/hono/dist/request.js","../../node_modules/hono/dist/utils/html.js","../../node_modules/hono/dist/context.js","../../node_modules/hono/dist/router.js","../../node_modules/hono/dist/utils/constants.js","../../node_modules/hono/dist/hono-base.js","../../node_modules/hono/dist/router/reg-exp-router/matcher.js","../../node_modules/hono/dist/router/reg-exp-router/node.js","../../node_modules/hono/dist/router/reg-exp-router/trie.js","../../node_modules/hono/dist/router/reg-exp-router/router.js","../../node_modules/hono/dist/router/reg-exp-router/prepared-router.js","../../node_modules/hono/dist/router/reg-exp-router/index.js","../../node_modules/hono/dist/router/smart-router/router.js","../../node_modules/hono/dist/router/smart-router/index.js","../../node_modules/hono/dist/router/trie-router/node.js","../../node_modules/hono/dist/router/trie-router/router.js","../../node_modules/hono/dist/router/trie-router/index.js","../../node_modules/hono/dist/hono.js","../../node_modules/hono/dist/index.js","../../node_modules/hono/dist/middleware/cors/index.js","../../node_modules/@hono/node-server/dist/index.mjs","../../src/server/config.ts","../../src/server/message-queue.ts","../../src/server/peer-discovery.ts","../../src/server/registry.ts","../../src/server/state.ts","../../src/server/models.ts","../../src/server/routes/agent.ts","../../src/server/routes/proxy.ts","../../src/server/dashboard-html.ts","../../src/server/routes/dashboard.ts","../../src/server/index.ts","../../src/cli/setup.ts","../../src/cli/agent-detector.ts","../../src/cli/agent-installer.ts","../../src/cli/tailscale-login.ts","../../src/cli/tty-utils.ts","../../src/cli/login.ts","../../src/cli/logout.ts","../../src/cli/server-utils.ts","../../src/cli/start-server.ts","../../src/cli/stop-server.ts","../../src/cli/status.ts","../../src/cli/version.ts","../../src/cli/agents.ts","../../src/cli/peers.ts","../../src/cli/install.ts","../../src/cli/index.ts"],"sourcesContent":["/**\r\n * Tailscale integration for the Party Server.\r\n *\r\n * - Binary discovery and caching\r\n * - Tailnet hostname / IP resolution\r\n * - Status JSON reading (network topology)\r\n * - IP → identity whois with TTL cache\r\n * - Funnel / Serve lifecycle management\r\n */\r\n\r\nimport { execFileSync, execSync } from 'node:child_process';\r\nimport { existsSync } from 'node:fs';\r\nimport { join } from 'node:path';\r\n\r\n// ---------------------------------------------------------------------------\r\n// Helpers\r\n// ---------------------------------------------------------------------------\r\n\r\n/** Extract the first valid JSON object from possibly noisy CLI output. */\r\nfunction parsePossiblyNoisyJson(raw: string): Record<string, unknown> {\r\n const trimmed = raw.trim();\r\n const start = trimmed.indexOf('{');\r\n const end = trimmed.lastIndexOf('}');\r\n if (start >= 0 && end > start) {\r\n return JSON.parse(trimmed.slice(start, end + 1));\r\n }\r\n return JSON.parse(trimmed);\r\n}\r\n\r\n/** Run a command synchronously and return stdout. Throws on non-zero exit. */\r\nfunction runExec(cmd: string[], timeout = 5000): string {\r\n return execFileSync(cmd[0], cmd.slice(1), {\r\n timeout,\r\n encoding: 'utf-8',\r\n stdio: ['pipe', 'pipe', 'pipe'],\r\n windowsHide: true,\r\n });\r\n}\r\n\r\n// ---------------------------------------------------------------------------\r\n// Binary discovery\r\n// ---------------------------------------------------------------------------\r\n\r\nlet cachedBinary: string | null = null;\r\n\r\n/** Return true if path exists and can respond to --version. */\r\nfunction checkBinary(path: string, timeout = 3000): boolean {\r\n if (!path || !existsSync(path)) return false;\r\n try {\r\n execFileSync(path, ['--version'], { timeout, encoding: 'utf-8', stdio: 'pipe', windowsHide: true });\r\n return true;\r\n } catch {\r\n return false;\r\n }\r\n}\r\n\r\n/** Locate the tailscale binary using multiple strategies. */\r\nfunction findTailscaleBinary(): string | null {\r\n // Strategy 1 – PATH lookup\r\n try {\r\n const which = process.platform === 'win32' ? 'where' : 'which';\r\n const result = execFileSync(which, ['tailscale'], { timeout: 3000, encoding: 'utf-8', stdio: 'pipe', windowsHide: true });\r\n const fromPath = result.trim().split(/\\r?\\n/)[0];\r\n if (fromPath && checkBinary(fromPath)) return fromPath;\r\n } catch { /* not in PATH */ }\r\n\r\n // Strategy 2 – Known platform paths\r\n const knownPaths = process.platform === 'win32'\r\n ? [\r\n join(process.env.ProgramFiles || 'C:\\\\Program Files', 'Tailscale', 'tailscale.exe'),\r\n join(process.env['ProgramFiles(x86)'] || 'C:\\\\Program Files (x86)', 'Tailscale', 'tailscale.exe'),\r\n join(process.env.LOCALAPPDATA || '', 'Tailscale', 'tailscale.exe'),\r\n ]\r\n : [\r\n '/Applications/Tailscale.app/Contents/MacOS/Tailscale',\r\n '/usr/bin/tailscale',\r\n '/usr/local/bin/tailscale',\r\n ];\r\n\r\n for (const candidate of knownPaths) {\r\n if (checkBinary(candidate)) return candidate;\r\n }\r\n\r\n // Strategy 3 – find in /Applications (macOS only)\r\n if (process.platform !== 'win32') {\r\n try {\r\n const result = execSync(\r\n 'find /Applications -maxdepth 3 -name Tailscale -path \"*/Tailscale.app/Contents/MacOS/Tailscale\"',\r\n { timeout: 5000, encoding: 'utf-8' },\r\n );\r\n const first = result.trim().split('\\n')[0];\r\n if (first && checkBinary(first)) return first;\r\n } catch { /* not found */ }\r\n }\r\n\r\n return null;\r\n}\r\n\r\n/** Return a cached tailscale binary path, discovering it on the first call. */\r\nexport function getTailscaleBinary(): string {\r\n const forced = (process.env.OPENCLAW_TEST_TAILSCALE_BINARY ?? '').trim();\r\n if (forced) {\r\n cachedBinary = forced;\r\n return forced;\r\n }\r\n if (cachedBinary !== null) return cachedBinary;\r\n cachedBinary = findTailscaleBinary() ?? (process.platform === 'win32' ? 'tailscale.exe' : 'tailscale');\r\n return cachedBinary;\r\n}\r\n\r\n// ---------------------------------------------------------------------------\r\n// Status & hostname\r\n// ---------------------------------------------------------------------------\r\n\r\n/** Run `tailscale status --json` and return the parsed result. */\r\nexport function readTailscaleStatus(timeout = 5000): Record<string, unknown> {\r\n const binary = getTailscaleBinary();\r\n const stdout = runExec([binary, 'status', '--json'], timeout);\r\n return stdout.trim() ? parsePossiblyNoisyJson(stdout) : {};\r\n}\r\n\r\n/** Derive the current device's tailnet hostname or fall back to its first Tailscale IP. */\r\nexport function getTailnetHostname(binary?: string): string {\r\n const candidates = binary\r\n ? [binary]\r\n : [\r\n getTailscaleBinary(),\r\n '/Applications/Tailscale.app/Contents/MacOS/Tailscale',\r\n ];\r\n\r\n let lastErr: Error | null = null;\r\n for (const candidate of candidates) {\r\n if (candidate.startsWith('/') && !existsSync(candidate)) continue;\r\n try {\r\n const stdout = runExec([candidate, 'status', '--json'], 5000);\r\n const parsed = stdout.trim() ? parsePossiblyNoisyJson(stdout) : {};\r\n const selfInfo = (parsed.Self ?? {}) as Record<string, unknown>;\r\n\r\n const dns = selfInfo.DNSName as string | undefined;\r\n if (typeof dns === 'string' && dns) return dns.replace(/\\.$/, '');\r\n\r\n const ips = selfInfo.TailscaleIPs as string[] | undefined;\r\n if (ips && ips.length > 0) return ips[0];\r\n\r\n throw new Error('Could not determine Tailscale DNS or IP');\r\n } catch (exc) {\r\n lastErr = exc as Error;\r\n }\r\n }\r\n throw lastErr ?? new Error('Could not determine Tailscale DNS or IP');\r\n}\r\n\r\n/** Return the list of Tailscale IPs for the current device. */\r\nexport function getTailscaleIps(): string[] {\r\n const status = readTailscaleStatus();\r\n const selfInfo = status.Self as Record<string, unknown> | undefined;\r\n if (selfInfo && Array.isArray(selfInfo.TailscaleIPs)) {\r\n return selfInfo.TailscaleIPs as string[];\r\n }\r\n return [];\r\n}\r\n\r\n// ---------------------------------------------------------------------------\r\n// Whois with TTL cache\r\n// ---------------------------------------------------------------------------\r\n\r\nexport class WhoisIdentity {\r\n constructor(\r\n public readonly login: string,\r\n public readonly name?: string,\r\n ) {}\r\n}\r\n\r\ninterface CacheEntry {\r\n value: WhoisIdentity | null;\r\n expiresAt: number;\r\n}\r\n\r\nconst whoisCache = new Map<string, CacheEntry>();\r\n\r\n/** Best-effort extraction of login/name from tailscale whois JSON. */\r\nfunction parseWhoisIdentity(payload: Record<string, unknown>): WhoisIdentity | null {\r\n const userProfile =\r\n (payload.UserProfile as Record<string, unknown>) ??\r\n (payload.userProfile as Record<string, unknown>) ??\r\n (payload.User as Record<string, unknown>) ??\r\n {};\r\n\r\n const login =\r\n (userProfile.LoginName as string) ||\r\n (userProfile.Login as string) ||\r\n (userProfile.login as string) ||\r\n (payload.LoginName as string) ||\r\n (payload.login as string);\r\n\r\n if (typeof login !== 'string' || !login.trim()) return null;\r\n\r\n const rawName =\r\n (userProfile.DisplayName as string) ||\r\n (userProfile.Name as string) ||\r\n (userProfile.displayName as string) ||\r\n (payload.DisplayName as string) ||\r\n (payload.name as string);\r\n\r\n const name = typeof rawName === 'string' ? rawName.trim() : undefined;\r\n return new WhoisIdentity(login.trim(), name);\r\n}\r\n\r\n/** Resolve an IP to a WhoisIdentity via `tailscale whois --json`. */\r\nexport function readWhoisIdentity(\r\n ip: string,\r\n timeout = 5000,\r\n cacheTtl = 60,\r\n errorTtl = 5,\r\n): WhoisIdentity | null {\r\n const normalized = ip.trim();\r\n if (!normalized) return null;\r\n\r\n const now = performance.now() / 1000;\r\n const cached = whoisCache.get(normalized);\r\n if (cached) {\r\n if (cached.expiresAt > now) return cached.value;\r\n whoisCache.delete(normalized);\r\n }\r\n\r\n const binary = getTailscaleBinary();\r\n let identity: WhoisIdentity | null = null;\r\n try {\r\n const stdout = runExec([binary, 'whois', '--json', normalized], timeout);\r\n const parsed = stdout.trim() ? parsePossiblyNoisyJson(stdout) : {};\r\n identity = parseWhoisIdentity(parsed);\r\n } catch { /* lookup failed */ }\r\n\r\n const ttl = identity ? cacheTtl : errorTtl;\r\n whoisCache.set(normalized, { value: identity, expiresAt: now + ttl });\r\n return identity;\r\n}\r\n\r\n// ---------------------------------------------------------------------------\r\n// Funnel / Serve lifecycle\r\n// ---------------------------------------------------------------------------\r\n\r\nconst PERMISSION_KEYWORDS = [\r\n 'permission denied',\r\n 'access denied',\r\n 'operation not permitted',\r\n 'not permitted',\r\n 'requires root',\r\n 'must be run as root',\r\n 'must be run with sudo',\r\n 'requires sudo',\r\n 'need sudo',\r\n];\r\n\r\n/** Run cmd; on permission errors retry with sudo -n. */\r\nfunction execWithSudoFallback(cmd: string[], timeout = 15000): string {\r\n try {\r\n return runExec(cmd, timeout);\r\n } catch (exc: unknown) {\r\n const err = exc as { stderr?: string };\r\n const stderrLower = (err.stderr ?? '').toLowerCase();\r\n if (PERMISSION_KEYWORDS.some((kw) => stderrLower.includes(kw))) {\r\n try {\r\n return runExec(['sudo', '-n', ...cmd], timeout);\r\n } catch { /* sudo failed too */ }\r\n }\r\n throw exc;\r\n }\r\n}\r\n\r\n/** Enable `tailscale serve` on port (background mode). */\r\nexport function enableServe(port: number, timeout = 15000): void {\r\n const binary = getTailscaleBinary();\r\n execWithSudoFallback([binary, 'serve', '--bg', '--yes', String(port)], timeout);\r\n}\r\n\r\n/** Reset `tailscale serve`. */\r\nexport function disableServe(timeout = 15000): void {\r\n const binary = getTailscaleBinary();\r\n execWithSudoFallback([binary, 'serve', 'reset'], timeout);\r\n}\r\n\r\n/** Enable `tailscale funnel` on port (background mode). */\r\nexport function enableFunnel(port: number, timeout = 15000): void {\r\n const binary = getTailscaleBinary();\r\n execWithSudoFallback([binary, 'funnel', '--bg', '--yes', String(port)], timeout);\r\n}\r\n\r\n/** Reset `tailscale funnel`. */\r\nexport function disableFunnel(timeout = 15000): void {\r\n const binary = getTailscaleBinary();\r\n execWithSudoFallback([binary, 'funnel', 'reset'], timeout);\r\n}\r\n\r\n// ---------------------------------------------------------------------------\r\n// Network join & connection status\r\n// ---------------------------------------------------------------------------\r\n\r\n/** Join a Tailscale network using an auth key. */\r\nexport function joinTailnet(\r\n authKey: string,\r\n timeout = 30000,\r\n): { success: boolean; output: string } {\r\n const binary = getTailscaleBinary();\r\n try {\r\n const output = execWithSudoFallback(\r\n [binary, 'up', '--authkey', authKey],\r\n timeout,\r\n );\r\n return { success: true, output: output.trim() };\r\n } catch (e) {\r\n const err = e as { stderr?: string; message?: string };\r\n return { success: false, output: (err.stderr ?? err.message ?? '').trim() };\r\n }\r\n}\r\n\r\n/** Get the current Tailscale connection status. */\r\nexport function getTailscaleConnectionStatus(): {\r\n connected: boolean;\r\n tailscale_ip: string | null;\r\n hostname: string | null;\r\n error?: string;\r\n} {\r\n try {\r\n const status = readTailscaleStatus();\r\n const self = (status.Self ?? {}) as Record<string, unknown>;\r\n const online = self.Online === true;\r\n const ips = self.TailscaleIPs as string[] | undefined;\r\n const dns = (self.DNSName as string | undefined)?.replace(/\\.$/, '');\r\n return {\r\n connected: online,\r\n tailscale_ip: ips?.[0] ?? null,\r\n hostname: dns ?? null,\r\n };\r\n } catch (e) {\r\n return {\r\n connected: false,\r\n tailscale_ip: null,\r\n hostname: null,\r\n error: (e as Error).message,\r\n };\r\n }\r\n}\r\n\r\n// ---------------------------------------------------------------------------\r\n// Tri-state installation status & install instructions\r\n// ---------------------------------------------------------------------------\r\n\r\nexport type TailscaleState =\r\n | { state: 'not_installed'; platform: string }\r\n | { state: 'not_connected'; binary: string; error?: string }\r\n | { state: 'connected'; binary: string; tailscale_ip: string; hostname: string };\r\n\r\n/** Detect Tailscale installation status as one of three states. */\r\nexport function getTailscaleInstallationStatus(): TailscaleState {\r\n // Bypass cache so we always get the real on-disk state\r\n const binary = findTailscaleBinary();\r\n if (!binary) {\r\n return { state: 'not_installed', platform: process.platform };\r\n }\r\n\r\n try {\r\n const status = readTailscaleStatus();\r\n const self = (status.Self ?? {}) as Record<string, unknown>;\r\n const online = self.Online === true;\r\n\r\n if (online) {\r\n const ips = self.TailscaleIPs as string[] | undefined;\r\n const dns = (self.DNSName as string | undefined)?.replace(/\\.$/, '');\r\n return {\r\n state: 'connected',\r\n binary,\r\n tailscale_ip: ips?.[0] ?? '127.0.0.1',\r\n hostname: dns ?? null!,\r\n };\r\n }\r\n\r\n return { state: 'not_connected', binary };\r\n } catch (e) {\r\n return {\r\n state: 'not_connected',\r\n binary,\r\n error: (e as Error).message,\r\n };\r\n }\r\n}\r\n\r\n/** Clear the cached binary path so the next detection re-scans the filesystem. */\r\nexport function resetTailscaleBinaryCache(): void {\r\n cachedBinary = null;\r\n}\r\n\r\n// ---------------------------------------------------------------------------\r\n// Logout (full de-authentication)\r\n// ---------------------------------------------------------------------------\r\n\r\n/** Run `tailscale logout` — disconnects and invalidates the node key. */\r\nexport function logoutTailscale(timeout = 15000): { success: boolean; output: string } {\r\n const binary = getTailscaleBinary();\r\n try {\r\n const output = runExec([binary, 'logout'], timeout);\r\n resetTailscaleBinaryCache();\r\n return { success: true, output: output.trim() };\r\n } catch (e) {\r\n const err = e as { stderr?: string; message?: string };\r\n return { success: false, output: (err.stderr ?? err.message ?? '').trim() };\r\n }\r\n}\r\n\r\n// ---------------------------------------------------------------------------\r\n// Interactive login (browser-based authentication)\r\n// ---------------------------------------------------------------------------\r\n\r\nimport { type ChildProcess, spawn as nodeSpawn } from 'node:child_process';\r\n\r\nexport interface InteractiveLoginResult {\r\n success: boolean;\r\n url?: string;\r\n output: string;\r\n}\r\n\r\n/**\r\n * Start `tailscale login` as a background child process.\r\n *\r\n * `tailscale login` prints an auth URL to stdout, then blocks until the\r\n * user completes authentication in the browser. This function parses that\r\n * URL and returns it immediately — the caller is responsible for polling\r\n * connection status separately (e.g. via `getTailscaleInstallationStatus()`).\r\n *\r\n * The returned `process` reference lets the caller kill it if needed.\r\n */\r\nexport function startInteractiveLogin(): {\r\n promise: Promise<InteractiveLoginResult>;\r\n process: ChildProcess;\r\n} {\r\n const binary = getTailscaleBinary();\r\n const child = nodeSpawn(binary, ['login'], {\r\n stdio: ['pipe', 'pipe', 'pipe'],\r\n windowsHide: true,\r\n });\r\n\r\n const urlRegex = /https:\\/\\/login\\.tailscale\\.com\\/a\\/[^\\s]+/;\r\n\r\n const promise = new Promise<InteractiveLoginResult>((resolve) => {\r\n let stdout = '';\r\n let resolved = false;\r\n\r\n const done = (result: InteractiveLoginResult) => {\r\n if (resolved) return;\r\n resolved = true;\r\n resolve(result);\r\n };\r\n\r\n child.stdout?.on('data', (data: Buffer) => {\r\n stdout += data.toString();\r\n const match = stdout.match(urlRegex);\r\n if (match) {\r\n done({ success: true, url: match[0], output: stdout.trim() });\r\n }\r\n });\r\n\r\n child.stderr?.on('data', (data: Buffer) => {\r\n stdout += data.toString();\r\n });\r\n\r\n child.on('close', (code) => {\r\n if (code === 0) {\r\n done({ success: true, output: stdout.trim() });\r\n } else {\r\n done({ success: false, output: stdout.trim() || `Exited with code ${code}` });\r\n }\r\n });\r\n\r\n child.on('error', (err) => {\r\n done({ success: false, output: err.message });\r\n });\r\n\r\n // Timeout: give up if no URL appears within 30s\r\n setTimeout(() => {\r\n done({ success: false, output: 'Timeout waiting for login URL' });\r\n try { child.kill(); } catch { /* already exited */ }\r\n }, 30000);\r\n });\r\n\r\n return { promise, process: child };\r\n}\r\n\r\n/** Return platform-specific install instructions for Tailscale. */\r\nexport function getInstallInstructions(platform: string): {\r\n os: string;\r\n download_url: string;\r\n commands: string[];\r\n needs_sudo: boolean;\r\n} {\r\n switch (platform) {\r\n case 'linux':\r\n return {\r\n os: 'linux',\r\n download_url: 'https://tailscale.com/download/linux',\r\n commands: ['curl -fsSL https://tailscale.com/install.sh | sh'],\r\n needs_sudo: true,\r\n };\r\n case 'darwin':\r\n return {\r\n os: 'macOS',\r\n download_url: 'https://tailscale.com/download/mac',\r\n commands: ['brew install tailscale'],\r\n needs_sudo: false,\r\n };\r\n case 'win32':\r\n return {\r\n os: 'windows',\r\n download_url: 'https://tailscale.com/download/windows',\r\n commands: ['winget install Tailscale.Tailscale'],\r\n needs_sudo: false,\r\n };\r\n default:\r\n return {\r\n os: platform,\r\n download_url: 'https://tailscale.com/download/',\r\n commands: [],\r\n needs_sudo: false,\r\n };\r\n }\r\n}\r\n","/**\n * Platform-specific Tailscale installer.\n *\n * Executes installation commands with real-time output (stdio: 'inherit').\n * On Linux/macOS, uses sudo when needed.\n */\n\nimport { spawn } from 'node:child_process';\n\nexport interface InstallResult {\n success: boolean;\n output: string;\n}\n\ntype PlatformEntry = {\n cmd: string;\n args: string[];\n needsSudo: boolean;\n};\n\nconst INSTALL_COMMANDS: Record<string, PlatformEntry> = {\n linux: { cmd: 'bash', args: ['-c', 'curl -fsSL https://tailscale.com/install.sh | sh'], needsSudo: true },\n darwin: { cmd: 'brew', args: ['install', 'tailscale'], needsSudo: false },\n win32: { cmd: 'winget', args: ['install', 'Tailscale.Tailscale', '--accept-source-agreements'], needsSudo: false },\n};\n\nasync function installTailscale(platform: string): Promise<InstallResult> {\n const entry = INSTALL_COMMANDS[platform];\n if (!entry) {\n return {\n success: false,\n output: `Unsupported platform: ${platform}. Please install manually from https://tailscale.com/download`,\n };\n }\n\n const cmd = entry.needsSudo ? 'sudo' : entry.cmd;\n const args = entry.needsSudo ? [entry.cmd, ...entry.args] : entry.args;\n\n console.log(`Running: ${cmd} ${args.join(' ')}\\n`);\n\n return new Promise((resolve) => {\n const child = spawn(cmd, args, {\n stdio: 'inherit',\n windowsHide: true,\n });\n\n let exited = false;\n\n child.on('close', (code) => {\n if (exited) return;\n exited = true;\n if (code === 0) {\n resolve({ success: true, output: 'Installation completed.' });\n } else {\n resolve({ success: false, output: `Installation exited with code ${code}` });\n }\n });\n\n child.on('error', (err) => {\n if (exited) return;\n exited = true;\n resolve({ success: false, output: err.message });\n });\n });\n}\n\nexport { installTailscale };","// src/compose.ts\nvar compose = (middleware, onError, onNotFound) => {\n return (context, next) => {\n let index = -1;\n return dispatch(0);\n async function dispatch(i) {\n if (i <= index) {\n throw new Error(\"next() called multiple times\");\n }\n index = i;\n let res;\n let isError = false;\n let handler;\n if (middleware[i]) {\n handler = middleware[i][0][0];\n context.req.routeIndex = i;\n } else {\n handler = i === middleware.length && next || void 0;\n }\n if (handler) {\n try {\n res = await handler(context, () => dispatch(i + 1));\n } catch (err) {\n if (err instanceof Error && onError) {\n context.error = err;\n res = await onError(err, context);\n isError = true;\n } else {\n throw err;\n }\n }\n } else {\n if (context.finalized === false && onNotFound) {\n res = await onNotFound(context);\n }\n }\n if (res && (context.finalized === false || isError)) {\n context.res = res;\n }\n return context;\n }\n };\n};\nexport {\n compose\n};\n","// src/http-exception.ts\nvar HTTPException = class extends Error {\n res;\n status;\n /**\n * Creates an instance of `HTTPException`.\n * @param status - HTTP status code for the exception. Defaults to 500.\n * @param options - Additional options for the exception.\n */\n constructor(status = 500, options) {\n super(options?.message, { cause: options?.cause });\n this.res = options?.res;\n this.status = status;\n }\n /**\n * Returns the response object associated with the exception.\n * If a response object is not provided, a new response is created with the error message and status code.\n * @returns The response object.\n */\n getResponse() {\n if (this.res) {\n const newResponse = new Response(this.res.body, {\n status: this.status,\n headers: this.res.headers\n });\n return newResponse;\n }\n return new Response(this.message, {\n status: this.status\n });\n }\n};\nexport {\n HTTPException\n};\n","// src/request/constants.ts\nvar GET_MATCH_RESULT = /* @__PURE__ */ Symbol();\nexport {\n GET_MATCH_RESULT\n};\n","// src/utils/body.ts\nimport { HonoRequest } from \"../request.js\";\nvar parseBody = async (request, options = /* @__PURE__ */ Object.create(null)) => {\n const { all = false, dot = false } = options;\n const headers = request instanceof HonoRequest ? request.raw.headers : request.headers;\n const contentType = headers.get(\"Content-Type\");\n if (contentType?.startsWith(\"multipart/form-data\") || contentType?.startsWith(\"application/x-www-form-urlencoded\")) {\n return parseFormData(request, { all, dot });\n }\n return {};\n};\nasync function parseFormData(request, options) {\n const formData = await request.formData();\n if (formData) {\n return convertFormDataToBodyData(formData, options);\n }\n return {};\n}\nfunction convertFormDataToBodyData(formData, options) {\n const form = /* @__PURE__ */ Object.create(null);\n formData.forEach((value, key) => {\n const shouldParseAllValues = options.all || key.endsWith(\"[]\");\n if (!shouldParseAllValues) {\n form[key] = value;\n } else {\n handleParsingAllValues(form, key, value);\n }\n });\n if (options.dot) {\n Object.entries(form).forEach(([key, value]) => {\n const shouldParseDotValues = key.includes(\".\");\n if (shouldParseDotValues) {\n handleParsingNestedValues(form, key, value);\n delete form[key];\n }\n });\n }\n return form;\n}\nvar handleParsingAllValues = (form, key, value) => {\n if (form[key] !== void 0) {\n if (Array.isArray(form[key])) {\n ;\n form[key].push(value);\n } else {\n form[key] = [form[key], value];\n }\n } else {\n if (!key.endsWith(\"[]\")) {\n form[key] = value;\n } else {\n form[key] = [value];\n }\n }\n};\nvar handleParsingNestedValues = (form, key, value) => {\n if (/(?:^|\\.)__proto__\\./.test(key)) {\n return;\n }\n let nestedForm = form;\n const keys = key.split(\".\");\n keys.forEach((key2, index) => {\n if (index === keys.length - 1) {\n nestedForm[key2] = value;\n } else {\n if (!nestedForm[key2] || typeof nestedForm[key2] !== \"object\" || Array.isArray(nestedForm[key2]) || nestedForm[key2] instanceof File) {\n nestedForm[key2] = /* @__PURE__ */ Object.create(null);\n }\n nestedForm = nestedForm[key2];\n }\n });\n};\nexport {\n parseBody\n};\n","// src/utils/url.ts\nvar splitPath = (path) => {\n const paths = path.split(\"/\");\n if (paths[0] === \"\") {\n paths.shift();\n }\n return paths;\n};\nvar splitRoutingPath = (routePath) => {\n const { groups, path } = extractGroupsFromPath(routePath);\n const paths = splitPath(path);\n return replaceGroupMarks(paths, groups);\n};\nvar extractGroupsFromPath = (path) => {\n const groups = [];\n path = path.replace(/\\{[^}]+\\}/g, (match, index) => {\n const mark = `@${index}`;\n groups.push([mark, match]);\n return mark;\n });\n return { groups, path };\n};\nvar replaceGroupMarks = (paths, groups) => {\n for (let i = groups.length - 1; i >= 0; i--) {\n const [mark] = groups[i];\n for (let j = paths.length - 1; j >= 0; j--) {\n if (paths[j].includes(mark)) {\n paths[j] = paths[j].replace(mark, groups[i][1]);\n break;\n }\n }\n }\n return paths;\n};\nvar patternCache = {};\nvar getPattern = (label, next) => {\n if (label === \"*\") {\n return \"*\";\n }\n const match = label.match(/^\\:([^\\{\\}]+)(?:\\{(.+)\\})?$/);\n if (match) {\n const cacheKey = `${label}#${next}`;\n if (!patternCache[cacheKey]) {\n if (match[2]) {\n patternCache[cacheKey] = next && next[0] !== \":\" && next[0] !== \"*\" ? [cacheKey, match[1], new RegExp(`^${match[2]}(?=/${next})`)] : [label, match[1], new RegExp(`^${match[2]}$`)];\n } else {\n patternCache[cacheKey] = [label, match[1], true];\n }\n }\n return patternCache[cacheKey];\n }\n return null;\n};\nvar tryDecode = (str, decoder) => {\n try {\n return decoder(str);\n } catch {\n return str.replace(/(?:%[0-9A-Fa-f]{2})+/g, (match) => {\n try {\n return decoder(match);\n } catch {\n return match;\n }\n });\n }\n};\nvar tryDecodeURI = (str) => tryDecode(str, decodeURI);\nvar getPath = (request) => {\n const url = request.url;\n const start = url.indexOf(\"/\", url.indexOf(\":\") + 4);\n let i = start;\n for (; i < url.length; i++) {\n const charCode = url.charCodeAt(i);\n if (charCode === 37) {\n const queryIndex = url.indexOf(\"?\", i);\n const hashIndex = url.indexOf(\"#\", i);\n const end = queryIndex === -1 ? hashIndex === -1 ? void 0 : hashIndex : hashIndex === -1 ? queryIndex : Math.min(queryIndex, hashIndex);\n const path = url.slice(start, end);\n return tryDecodeURI(path.includes(\"%25\") ? path.replace(/%25/g, \"%2525\") : path);\n } else if (charCode === 63 || charCode === 35) {\n break;\n }\n }\n return url.slice(start, i);\n};\nvar getQueryStrings = (url) => {\n const queryIndex = url.indexOf(\"?\", 8);\n return queryIndex === -1 ? \"\" : \"?\" + url.slice(queryIndex + 1);\n};\nvar getPathNoStrict = (request) => {\n const result = getPath(request);\n return result.length > 1 && result.at(-1) === \"/\" ? result.slice(0, -1) : result;\n};\nvar mergePath = (base, sub, ...rest) => {\n if (rest.length) {\n sub = mergePath(sub, ...rest);\n }\n return `${base?.[0] === \"/\" ? \"\" : \"/\"}${base}${sub === \"/\" ? \"\" : `${base?.at(-1) === \"/\" ? \"\" : \"/\"}${sub?.[0] === \"/\" ? sub.slice(1) : sub}`}`;\n};\nvar checkOptionalParameter = (path) => {\n if (path.charCodeAt(path.length - 1) !== 63 || !path.includes(\":\")) {\n return null;\n }\n const segments = path.split(\"/\");\n const results = [];\n let basePath = \"\";\n segments.forEach((segment) => {\n if (segment !== \"\" && !/\\:/.test(segment)) {\n basePath += \"/\" + segment;\n } else if (/\\:/.test(segment)) {\n if (/\\?/.test(segment)) {\n if (results.length === 0 && basePath === \"\") {\n results.push(\"/\");\n } else {\n results.push(basePath);\n }\n const optionalSegment = segment.replace(\"?\", \"\");\n basePath += \"/\" + optionalSegment;\n results.push(basePath);\n } else {\n basePath += \"/\" + segment;\n }\n }\n });\n return results.filter((v, i, a) => a.indexOf(v) === i);\n};\nvar _decodeURI = (value) => {\n if (!/[%+]/.test(value)) {\n return value;\n }\n if (value.indexOf(\"+\") !== -1) {\n value = value.replace(/\\+/g, \" \");\n }\n return value.indexOf(\"%\") !== -1 ? tryDecode(value, decodeURIComponent_) : value;\n};\nvar _getQueryParam = (url, key, multiple) => {\n let encoded;\n if (!multiple && key && !/[%+]/.test(key)) {\n let keyIndex2 = url.indexOf(\"?\", 8);\n if (keyIndex2 === -1) {\n return void 0;\n }\n if (!url.startsWith(key, keyIndex2 + 1)) {\n keyIndex2 = url.indexOf(`&${key}`, keyIndex2 + 1);\n }\n while (keyIndex2 !== -1) {\n const trailingKeyCode = url.charCodeAt(keyIndex2 + key.length + 1);\n if (trailingKeyCode === 61) {\n const valueIndex = keyIndex2 + key.length + 2;\n const endIndex = url.indexOf(\"&\", valueIndex);\n return _decodeURI(url.slice(valueIndex, endIndex === -1 ? void 0 : endIndex));\n } else if (trailingKeyCode == 38 || isNaN(trailingKeyCode)) {\n return \"\";\n }\n keyIndex2 = url.indexOf(`&${key}`, keyIndex2 + 1);\n }\n encoded = /[%+]/.test(url);\n if (!encoded) {\n return void 0;\n }\n }\n const results = {};\n encoded ??= /[%+]/.test(url);\n let keyIndex = url.indexOf(\"?\", 8);\n while (keyIndex !== -1) {\n const nextKeyIndex = url.indexOf(\"&\", keyIndex + 1);\n let valueIndex = url.indexOf(\"=\", keyIndex);\n if (valueIndex > nextKeyIndex && nextKeyIndex !== -1) {\n valueIndex = -1;\n }\n let name = url.slice(\n keyIndex + 1,\n valueIndex === -1 ? nextKeyIndex === -1 ? void 0 : nextKeyIndex : valueIndex\n );\n if (encoded) {\n name = _decodeURI(name);\n }\n keyIndex = nextKeyIndex;\n if (name === \"\") {\n continue;\n }\n let value;\n if (valueIndex === -1) {\n value = \"\";\n } else {\n value = url.slice(valueIndex + 1, nextKeyIndex === -1 ? void 0 : nextKeyIndex);\n if (encoded) {\n value = _decodeURI(value);\n }\n }\n if (multiple) {\n if (!(results[name] && Array.isArray(results[name]))) {\n results[name] = [];\n }\n ;\n results[name].push(value);\n } else {\n results[name] ??= value;\n }\n }\n return key ? results[key] : results;\n};\nvar getQueryParam = _getQueryParam;\nvar getQueryParams = (url, key) => {\n return _getQueryParam(url, key, true);\n};\nvar decodeURIComponent_ = decodeURIComponent;\nexport {\n checkOptionalParameter,\n decodeURIComponent_,\n getPath,\n getPathNoStrict,\n getPattern,\n getQueryParam,\n getQueryParams,\n getQueryStrings,\n mergePath,\n splitPath,\n splitRoutingPath,\n tryDecode,\n tryDecodeURI\n};\n","// src/request.ts\nimport { HTTPException } from \"./http-exception.js\";\nimport { GET_MATCH_RESULT } from \"./request/constants.js\";\nimport { parseBody } from \"./utils/body.js\";\nimport { decodeURIComponent_, getQueryParam, getQueryParams, tryDecode } from \"./utils/url.js\";\nvar tryDecodeURIComponent = (str) => tryDecode(str, decodeURIComponent_);\nvar HonoRequest = class {\n /**\n * `.raw` can get the raw Request object.\n *\n * @see {@link https://hono.dev/docs/api/request#raw}\n *\n * @example\n * ```ts\n * // For Cloudflare Workers\n * app.post('/', async (c) => {\n * const metadata = c.req.raw.cf?.hostMetadata?\n * ...\n * })\n * ```\n */\n raw;\n #validatedData;\n // Short name of validatedData\n #matchResult;\n routeIndex = 0;\n /**\n * `.path` can get the pathname of the request.\n *\n * @see {@link https://hono.dev/docs/api/request#path}\n *\n * @example\n * ```ts\n * app.get('/about/me', (c) => {\n * const pathname = c.req.path // `/about/me`\n * })\n * ```\n */\n path;\n bodyCache = {};\n constructor(request, path = \"/\", matchResult = [[]]) {\n this.raw = request;\n this.path = path;\n this.#matchResult = matchResult;\n this.#validatedData = {};\n }\n param(key) {\n return key ? this.#getDecodedParam(key) : this.#getAllDecodedParams();\n }\n #getDecodedParam(key) {\n const paramKey = this.#matchResult[0][this.routeIndex][1][key];\n const param = this.#getParamValue(paramKey);\n return param && /\\%/.test(param) ? tryDecodeURIComponent(param) : param;\n }\n #getAllDecodedParams() {\n const decoded = {};\n const keys = Object.keys(this.#matchResult[0][this.routeIndex][1]);\n for (const key of keys) {\n const value = this.#getParamValue(this.#matchResult[0][this.routeIndex][1][key]);\n if (value !== void 0) {\n decoded[key] = /\\%/.test(value) ? tryDecodeURIComponent(value) : value;\n }\n }\n return decoded;\n }\n #getParamValue(paramKey) {\n return this.#matchResult[1] ? this.#matchResult[1][paramKey] : paramKey;\n }\n query(key) {\n return getQueryParam(this.url, key);\n }\n queries(key) {\n return getQueryParams(this.url, key);\n }\n header(name) {\n if (name) {\n return this.raw.headers.get(name) ?? void 0;\n }\n const headerData = {};\n this.raw.headers.forEach((value, key) => {\n headerData[key] = value;\n });\n return headerData;\n }\n async parseBody(options) {\n return parseBody(this, options);\n }\n #cachedBody = (key) => {\n const { bodyCache, raw } = this;\n const cachedBody = bodyCache[key];\n if (cachedBody) {\n return cachedBody;\n }\n const anyCachedKey = Object.keys(bodyCache)[0];\n if (anyCachedKey) {\n return bodyCache[anyCachedKey].then((body) => {\n if (anyCachedKey === \"json\") {\n body = JSON.stringify(body);\n }\n return new Response(body)[key]();\n });\n }\n return bodyCache[key] = raw[key]();\n };\n /**\n * `.json()` can parse Request body of type `application/json`\n *\n * @see {@link https://hono.dev/docs/api/request#json}\n *\n * @example\n * ```ts\n * app.post('/entry', async (c) => {\n * const body = await c.req.json()\n * })\n * ```\n */\n json() {\n return this.#cachedBody(\"text\").then((text) => JSON.parse(text));\n }\n /**\n * `.text()` can parse Request body of type `text/plain`\n *\n * @see {@link https://hono.dev/docs/api/request#text}\n *\n * @example\n * ```ts\n * app.post('/entry', async (c) => {\n * const body = await c.req.text()\n * })\n * ```\n */\n text() {\n return this.#cachedBody(\"text\");\n }\n /**\n * `.arrayBuffer()` parse Request body as an `ArrayBuffer`\n *\n * @see {@link https://hono.dev/docs/api/request#arraybuffer}\n *\n * @example\n * ```ts\n * app.post('/entry', async (c) => {\n * const body = await c.req.arrayBuffer()\n * })\n * ```\n */\n arrayBuffer() {\n return this.#cachedBody(\"arrayBuffer\");\n }\n /**\n * `.bytes()` parses the request body as a `Uint8Array`.\n *\n * @see {@link https://hono.dev/docs/api/request#bytes}\n *\n * @example\n * ```ts\n * app.post('/entry', async (c) => {\n * const body = await c.req.bytes()\n * })\n * ```\n */\n bytes() {\n return this.#cachedBody(\"arrayBuffer\").then((buffer) => new Uint8Array(buffer));\n }\n /**\n * Parses the request body as a `Blob`.\n * @example\n * ```ts\n * app.post('/entry', async (c) => {\n * const body = await c.req.blob();\n * });\n * ```\n * @see https://hono.dev/docs/api/request#blob\n */\n blob() {\n return this.#cachedBody(\"blob\");\n }\n /**\n * Parses the request body as `FormData`.\n * @example\n * ```ts\n * app.post('/entry', async (c) => {\n * const body = await c.req.formData();\n * });\n * ```\n * @see https://hono.dev/docs/api/request#formdata\n */\n formData() {\n return this.#cachedBody(\"formData\");\n }\n /**\n * Adds validated data to the request.\n *\n * @param target - The target of the validation.\n * @param data - The validated data to add.\n */\n addValidatedData(target, data) {\n this.#validatedData[target] = data;\n }\n valid(target) {\n return this.#validatedData[target];\n }\n /**\n * `.url()` can get the request url strings.\n *\n * @see {@link https://hono.dev/docs/api/request#url}\n *\n * @example\n * ```ts\n * app.get('/about/me', (c) => {\n * const url = c.req.url // `http://localhost:8787/about/me`\n * ...\n * })\n * ```\n */\n get url() {\n return this.raw.url;\n }\n /**\n * `.method()` can get the method name of the request.\n *\n * @see {@link https://hono.dev/docs/api/request#method}\n *\n * @example\n * ```ts\n * app.get('/about/me', (c) => {\n * const method = c.req.method // `GET`\n * })\n * ```\n */\n get method() {\n return this.raw.method;\n }\n get [GET_MATCH_RESULT]() {\n return this.#matchResult;\n }\n /**\n * `.matchedRoutes()` can return a matched route in the handler\n *\n * @deprecated\n *\n * Use matchedRoutes helper defined in \"hono/route\" instead.\n *\n * @see {@link https://hono.dev/docs/api/request#matchedroutes}\n *\n * @example\n * ```ts\n * app.use('*', async function logger(c, next) {\n * await next()\n * c.req.matchedRoutes.forEach(({ handler, method, path }, i) => {\n * const name = handler.name || (handler.length < 2 ? '[handler]' : '[middleware]')\n * console.log(\n * method,\n * ' ',\n * path,\n * ' '.repeat(Math.max(10 - path.length, 0)),\n * name,\n * i === c.req.routeIndex ? '<- respond from here' : ''\n * )\n * })\n * })\n * ```\n */\n get matchedRoutes() {\n return this.#matchResult[0].map(([[, route]]) => route);\n }\n /**\n * `routePath()` can retrieve the path registered within the handler\n *\n * @deprecated\n *\n * Use routePath helper defined in \"hono/route\" instead.\n *\n * @see {@link https://hono.dev/docs/api/request#routepath}\n *\n * @example\n * ```ts\n * app.get('/posts/:id', (c) => {\n * return c.json({ path: c.req.routePath })\n * })\n * ```\n */\n get routePath() {\n return this.#matchResult[0].map(([[, route]]) => route)[this.routeIndex].path;\n }\n};\nvar cloneRawRequest = async (req) => {\n if (!req.raw.bodyUsed) {\n return req.raw.clone();\n }\n const cacheKey = Object.keys(req.bodyCache)[0];\n if (!cacheKey) {\n throw new HTTPException(500, {\n message: \"Cannot clone request: body was already consumed and not cached. Please use HonoRequest methods (e.g., req.json(), req.text()) instead of consuming req.raw directly.\"\n });\n }\n const requestInit = {\n body: await req[cacheKey](),\n cache: req.raw.cache,\n credentials: req.raw.credentials,\n headers: req.header(),\n integrity: req.raw.integrity,\n keepalive: req.raw.keepalive,\n method: req.method,\n mode: req.raw.mode,\n redirect: req.raw.redirect,\n referrer: req.raw.referrer,\n referrerPolicy: req.raw.referrerPolicy,\n signal: req.raw.signal\n };\n return new Request(req.url, requestInit);\n};\nexport {\n HonoRequest,\n cloneRawRequest\n};\n","// src/utils/html.ts\nvar HtmlEscapedCallbackPhase = {\n Stringify: 1,\n BeforeStream: 2,\n Stream: 3\n};\nvar raw = (value, callbacks) => {\n const escapedString = new String(value);\n escapedString.isEscaped = true;\n escapedString.callbacks = callbacks;\n return escapedString;\n};\nvar escapeRe = /[&<>'\"]/;\nvar stringBufferToString = async (buffer, callbacks) => {\n let str = \"\";\n callbacks ||= [];\n const resolvedBuffer = await Promise.all(buffer);\n for (let i = resolvedBuffer.length - 1; ; i--) {\n str += resolvedBuffer[i];\n i--;\n if (i < 0) {\n break;\n }\n let r = resolvedBuffer[i];\n if (typeof r === \"object\") {\n callbacks.push(...r.callbacks || []);\n }\n const isEscaped = r.isEscaped;\n r = await (typeof r === \"object\" ? r.toString() : r);\n if (typeof r === \"object\") {\n callbacks.push(...r.callbacks || []);\n }\n if (r.isEscaped ?? isEscaped) {\n str += r;\n } else {\n const buf = [str];\n escapeToBuffer(r, buf);\n str = buf[0];\n }\n }\n return raw(str, callbacks);\n};\nvar escapeToBuffer = (str, buffer) => {\n const match = str.search(escapeRe);\n if (match === -1) {\n buffer[0] += str;\n return;\n }\n let escape;\n let index;\n let lastIndex = 0;\n for (index = match; index < str.length; index++) {\n switch (str.charCodeAt(index)) {\n case 34:\n escape = \"&quot;\";\n break;\n case 39:\n escape = \"&#39;\";\n break;\n case 38:\n escape = \"&amp;\";\n break;\n case 60:\n escape = \"&lt;\";\n break;\n case 62:\n escape = \"&gt;\";\n break;\n default:\n continue;\n }\n buffer[0] += str.substring(lastIndex, index) + escape;\n lastIndex = index + 1;\n }\n buffer[0] += str.substring(lastIndex, index);\n};\nvar resolveCallbackSync = (str) => {\n const callbacks = str.callbacks;\n if (!callbacks?.length) {\n return str;\n }\n const buffer = [str];\n const context = {};\n callbacks.forEach((c) => c({ phase: HtmlEscapedCallbackPhase.Stringify, buffer, context }));\n return buffer[0];\n};\nvar resolveCallback = async (str, phase, preserveCallbacks, context, buffer) => {\n if (typeof str === \"object\" && !(str instanceof String)) {\n if (!(str instanceof Promise)) {\n str = str.toString();\n }\n if (str instanceof Promise) {\n str = await str;\n }\n }\n const callbacks = str.callbacks;\n if (!callbacks?.length) {\n return Promise.resolve(str);\n }\n if (buffer) {\n buffer[0] += str;\n } else {\n buffer = [str];\n }\n const resStr = Promise.all(callbacks.map((c) => c({ phase, buffer, context }))).then(\n (res) => Promise.all(\n res.filter(Boolean).map((str2) => resolveCallback(str2, phase, false, context, buffer))\n ).then(() => buffer[0])\n );\n if (preserveCallbacks) {\n return raw(await resStr, callbacks);\n } else {\n return resStr;\n }\n};\nexport {\n HtmlEscapedCallbackPhase,\n escapeToBuffer,\n raw,\n resolveCallback,\n resolveCallbackSync,\n stringBufferToString\n};\n","// src/context.ts\nimport { HonoRequest } from \"./request.js\";\nimport { HtmlEscapedCallbackPhase, resolveCallback } from \"./utils/html.js\";\nvar TEXT_PLAIN = \"text/plain; charset=UTF-8\";\nvar setDefaultContentType = (contentType, headers) => {\n return {\n \"Content-Type\": contentType,\n ...headers\n };\n};\nvar createResponseInstance = (body, init) => new Response(body, init);\nvar Context = class {\n #rawRequest;\n #req;\n /**\n * `.env` can get bindings (environment variables, secrets, KV namespaces, D1 database, R2 bucket etc.) in Cloudflare Workers.\n *\n * @see {@link https://hono.dev/docs/api/context#env}\n *\n * @example\n * ```ts\n * // Environment object for Cloudflare Workers\n * app.get('*', async c => {\n * const counter = c.env.COUNTER\n * })\n * ```\n */\n env = {};\n #var;\n finalized = false;\n /**\n * `.error` can get the error object from the middleware if the Handler throws an error.\n *\n * @see {@link https://hono.dev/docs/api/context#error}\n *\n * @example\n * ```ts\n * app.use('*', async (c, next) => {\n * await next()\n * if (c.error) {\n * // do something...\n * }\n * })\n * ```\n */\n error;\n #status;\n #executionCtx;\n #res;\n #layout;\n #renderer;\n #notFoundHandler;\n #preparedHeaders;\n #matchResult;\n #path;\n /**\n * Creates an instance of the Context class.\n *\n * @param req - The Request object.\n * @param options - Optional configuration options for the context.\n */\n constructor(req, options) {\n this.#rawRequest = req;\n if (options) {\n this.#executionCtx = options.executionCtx;\n this.env = options.env;\n this.#notFoundHandler = options.notFoundHandler;\n this.#path = options.path;\n this.#matchResult = options.matchResult;\n }\n }\n /**\n * `.req` is the instance of {@link HonoRequest}.\n */\n get req() {\n this.#req ??= new HonoRequest(this.#rawRequest, this.#path, this.#matchResult);\n return this.#req;\n }\n /**\n * @see {@link https://hono.dev/docs/api/context#event}\n * The FetchEvent associated with the current request.\n *\n * @throws Will throw an error if the context does not have a FetchEvent.\n */\n get event() {\n if (this.#executionCtx && \"respondWith\" in this.#executionCtx) {\n return this.#executionCtx;\n } else {\n throw Error(\"This context has no FetchEvent\");\n }\n }\n /**\n * @see {@link https://hono.dev/docs/api/context#executionctx}\n * The ExecutionContext associated with the current request.\n *\n * @throws Will throw an error if the context does not have an ExecutionContext.\n */\n get executionCtx() {\n if (this.#executionCtx) {\n return this.#executionCtx;\n } else {\n throw Error(\"This context has no ExecutionContext\");\n }\n }\n /**\n * @see {@link https://hono.dev/docs/api/context#res}\n * The Response object for the current request.\n */\n get res() {\n return this.#res ||= createResponseInstance(null, {\n headers: this.#preparedHeaders ??= new Headers()\n });\n }\n /**\n * Sets the Response object for the current request.\n *\n * @param _res - The Response object to set.\n */\n set res(_res) {\n if (this.#res && _res) {\n _res = createResponseInstance(_res.body, _res);\n for (const [k, v] of this.#res.headers.entries()) {\n if (k === \"content-type\") {\n continue;\n }\n if (k === \"set-cookie\") {\n const cookies = this.#res.headers.getSetCookie();\n _res.headers.delete(\"set-cookie\");\n for (const cookie of cookies) {\n _res.headers.append(\"set-cookie\", cookie);\n }\n } else {\n _res.headers.set(k, v);\n }\n }\n }\n this.#res = _res;\n this.finalized = true;\n }\n /**\n * `.render()` can create a response within a layout.\n *\n * @see {@link https://hono.dev/docs/api/context#render-setrenderer}\n *\n * @example\n * ```ts\n * app.get('/', (c) => {\n * return c.render('Hello!')\n * })\n * ```\n */\n render = (...args) => {\n this.#renderer ??= (content) => this.html(content);\n return this.#renderer(...args);\n };\n /**\n * Sets the layout for the response.\n *\n * @param layout - The layout to set.\n * @returns The layout function.\n */\n setLayout = (layout) => this.#layout = layout;\n /**\n * Gets the current layout for the response.\n *\n * @returns The current layout function.\n */\n getLayout = () => this.#layout;\n /**\n * `.setRenderer()` can set the layout in the custom middleware.\n *\n * @see {@link https://hono.dev/docs/api/context#render-setrenderer}\n *\n * @example\n * ```tsx\n * app.use('*', async (c, next) => {\n * c.setRenderer((content) => {\n * return c.html(\n * <html>\n * <body>\n * <p>{content}</p>\n * </body>\n * </html>\n * )\n * })\n * await next()\n * })\n * ```\n */\n setRenderer = (renderer) => {\n this.#renderer = renderer;\n };\n /**\n * `.header()` can set headers.\n *\n * @see {@link https://hono.dev/docs/api/context#header}\n *\n * @example\n * ```ts\n * app.get('/welcome', (c) => {\n * // Set headers\n * c.header('X-Message', 'Hello!')\n * c.header('Content-Type', 'text/plain')\n *\n * return c.body('Thank you for coming')\n * })\n * ```\n */\n header = (name, value, options) => {\n if (this.finalized) {\n this.#res = createResponseInstance(this.#res.body, this.#res);\n }\n const headers = this.#res ? this.#res.headers : this.#preparedHeaders ??= new Headers();\n if (value === void 0) {\n headers.delete(name);\n } else if (options?.append) {\n headers.append(name, value);\n } else {\n headers.set(name, value);\n }\n };\n status = (status) => {\n this.#status = status;\n };\n /**\n * `.set()` can set the value specified by the key.\n *\n * @see {@link https://hono.dev/docs/api/context#set-get}\n *\n * @example\n * ```ts\n * app.use('*', async (c, next) => {\n * c.set('message', 'Hono is hot!!')\n * await next()\n * })\n * ```\n */\n set = (key, value) => {\n this.#var ??= /* @__PURE__ */ new Map();\n this.#var.set(key, value);\n };\n /**\n * `.get()` can use the value specified by the key.\n *\n * @see {@link https://hono.dev/docs/api/context#set-get}\n *\n * @example\n * ```ts\n * app.get('/', (c) => {\n * const message = c.get('message')\n * return c.text(`The message is \"${message}\"`)\n * })\n * ```\n */\n get = (key) => {\n return this.#var ? this.#var.get(key) : void 0;\n };\n /**\n * `.var` can access the value of a variable.\n *\n * @see {@link https://hono.dev/docs/api/context#var}\n *\n * @example\n * ```ts\n * const result = c.var.client.oneMethod()\n * ```\n */\n // c.var.propName is a read-only\n get var() {\n if (!this.#var) {\n return {};\n }\n return Object.fromEntries(this.#var);\n }\n #newResponse(data, arg, headers) {\n const responseHeaders = this.#res ? new Headers(this.#res.headers) : this.#preparedHeaders ?? new Headers();\n if (typeof arg === \"object\" && \"headers\" in arg) {\n const argHeaders = arg.headers instanceof Headers ? arg.headers : new Headers(arg.headers);\n for (const [key, value] of argHeaders) {\n if (key.toLowerCase() === \"set-cookie\") {\n responseHeaders.append(key, value);\n } else {\n responseHeaders.set(key, value);\n }\n }\n }\n if (headers) {\n for (const [k, v] of Object.entries(headers)) {\n if (typeof v === \"string\") {\n responseHeaders.set(k, v);\n } else {\n responseHeaders.delete(k);\n for (const v2 of v) {\n responseHeaders.append(k, v2);\n }\n }\n }\n }\n const status = typeof arg === \"number\" ? arg : arg?.status ?? this.#status;\n return createResponseInstance(data, { status, headers: responseHeaders });\n }\n newResponse = (...args) => this.#newResponse(...args);\n /**\n * `.body()` can return the HTTP response.\n * You can set headers with `.header()` and set HTTP status code with `.status`.\n * This can also be set in `.text()`, `.json()` and so on.\n *\n * @see {@link https://hono.dev/docs/api/context#body}\n *\n * @example\n * ```ts\n * app.get('/welcome', (c) => {\n * // Set headers\n * c.header('X-Message', 'Hello!')\n * c.header('Content-Type', 'text/plain')\n * // Set HTTP status code\n * c.status(201)\n *\n * // Return the response body\n * return c.body('Thank you for coming')\n * })\n * ```\n */\n body = (data, arg, headers) => this.#newResponse(data, arg, headers);\n /**\n * `.text()` can render text as `Content-Type:text/plain`.\n *\n * @see {@link https://hono.dev/docs/api/context#text}\n *\n * @example\n * ```ts\n * app.get('/say', (c) => {\n * return c.text('Hello!')\n * })\n * ```\n */\n text = (text, arg, headers) => {\n return !this.#preparedHeaders && !this.#status && !arg && !headers && !this.finalized ? new Response(text) : this.#newResponse(\n text,\n arg,\n setDefaultContentType(TEXT_PLAIN, headers)\n );\n };\n /**\n * `.json()` can render JSON as `Content-Type:application/json`.\n *\n * @see {@link https://hono.dev/docs/api/context#json}\n *\n * @example\n * ```ts\n * app.get('/api', (c) => {\n * return c.json({ message: 'Hello!' })\n * })\n * ```\n */\n json = (object, arg, headers) => {\n return this.#newResponse(\n JSON.stringify(object),\n arg,\n setDefaultContentType(\"application/json\", headers)\n );\n };\n html = (html, arg, headers) => {\n const res = (html2) => this.#newResponse(html2, arg, setDefaultContentType(\"text/html; charset=UTF-8\", headers));\n return typeof html === \"object\" ? resolveCallback(html, HtmlEscapedCallbackPhase.Stringify, false, {}).then(res) : res(html);\n };\n /**\n * `.redirect()` can Redirect, default status code is 302.\n *\n * @see {@link https://hono.dev/docs/api/context#redirect}\n *\n * @example\n * ```ts\n * app.get('/redirect', (c) => {\n * return c.redirect('/')\n * })\n * app.get('/redirect-permanently', (c) => {\n * return c.redirect('/', 301)\n * })\n * ```\n */\n redirect = (location, status) => {\n const locationString = String(location);\n this.header(\n \"Location\",\n // Multibyes should be encoded\n // eslint-disable-next-line no-control-regex\n !/[^\\x00-\\xFF]/.test(locationString) ? locationString : encodeURI(locationString)\n );\n return this.newResponse(null, status ?? 302);\n };\n /**\n * `.notFound()` can return the Not Found Response.\n *\n * @see {@link https://hono.dev/docs/api/context#notfound}\n *\n * @example\n * ```ts\n * app.get('/notfound', (c) => {\n * return c.notFound()\n * })\n * ```\n */\n notFound = () => {\n this.#notFoundHandler ??= () => createResponseInstance();\n return this.#notFoundHandler(this);\n };\n};\nexport {\n Context,\n TEXT_PLAIN\n};\n","// src/router.ts\nvar METHOD_NAME_ALL = \"ALL\";\nvar METHOD_NAME_ALL_LOWERCASE = \"all\";\nvar METHODS = [\"get\", \"post\", \"put\", \"delete\", \"options\", \"patch\"];\nvar MESSAGE_MATCHER_IS_ALREADY_BUILT = \"Can not add a route since the matcher is already built.\";\nvar UnsupportedPathError = class extends Error {\n};\nexport {\n MESSAGE_MATCHER_IS_ALREADY_BUILT,\n METHODS,\n METHOD_NAME_ALL,\n METHOD_NAME_ALL_LOWERCASE,\n UnsupportedPathError\n};\n","// src/utils/constants.ts\nvar COMPOSED_HANDLER = \"__COMPOSED_HANDLER\";\nexport {\n COMPOSED_HANDLER\n};\n","// src/hono-base.ts\nimport { compose } from \"./compose.js\";\nimport { Context } from \"./context.js\";\nimport { METHODS, METHOD_NAME_ALL, METHOD_NAME_ALL_LOWERCASE } from \"./router.js\";\nimport { COMPOSED_HANDLER } from \"./utils/constants.js\";\nimport { getPath, getPathNoStrict, mergePath } from \"./utils/url.js\";\nvar notFoundHandler = (c) => {\n return c.text(\"404 Not Found\", 404);\n};\nvar errorHandler = (err, c) => {\n if (\"getResponse\" in err) {\n const res = err.getResponse();\n return c.newResponse(res.body, res);\n }\n console.error(err);\n return c.text(\"Internal Server Error\", 500);\n};\nvar Hono = class _Hono {\n get;\n post;\n put;\n delete;\n options;\n patch;\n all;\n on;\n use;\n /*\n This class is like an abstract class and does not have a router.\n To use it, inherit the class and implement router in the constructor.\n */\n router;\n getPath;\n // Cannot use `#` because it requires visibility at JavaScript runtime.\n _basePath = \"/\";\n #path = \"/\";\n routes = [];\n constructor(options = {}) {\n const allMethods = [...METHODS, METHOD_NAME_ALL_LOWERCASE];\n allMethods.forEach((method) => {\n this[method] = (args1, ...args) => {\n if (typeof args1 === \"string\") {\n this.#path = args1;\n } else {\n this.#addRoute(method, this.#path, args1);\n }\n args.forEach((handler) => {\n this.#addRoute(method, this.#path, handler);\n });\n return this;\n };\n });\n this.on = (method, path, ...handlers) => {\n for (const p of [path].flat()) {\n this.#path = p;\n for (const m of [method].flat()) {\n handlers.map((handler) => {\n this.#addRoute(m.toUpperCase(), this.#path, handler);\n });\n }\n }\n return this;\n };\n this.use = (arg1, ...handlers) => {\n if (typeof arg1 === \"string\") {\n this.#path = arg1;\n } else {\n this.#path = \"*\";\n handlers.unshift(arg1);\n }\n handlers.forEach((handler) => {\n this.#addRoute(METHOD_NAME_ALL, this.#path, handler);\n });\n return this;\n };\n const { strict, ...optionsWithoutStrict } = options;\n Object.assign(this, optionsWithoutStrict);\n this.getPath = strict ?? true ? options.getPath ?? getPath : getPathNoStrict;\n }\n #clone() {\n const clone = new _Hono({\n router: this.router,\n getPath: this.getPath\n });\n clone.errorHandler = this.errorHandler;\n clone.#notFoundHandler = this.#notFoundHandler;\n clone.routes = this.routes;\n return clone;\n }\n #notFoundHandler = notFoundHandler;\n // Cannot use `#` because it requires visibility at JavaScript runtime.\n errorHandler = errorHandler;\n /**\n * `.route()` allows grouping other Hono instance in routes.\n *\n * @see {@link https://hono.dev/docs/api/routing#grouping}\n *\n * @param {string} path - base Path\n * @param {Hono} app - other Hono instance\n * @returns {Hono} routed Hono instance\n *\n * @example\n * ```ts\n * const app = new Hono()\n * const app2 = new Hono()\n *\n * app2.get(\"/user\", (c) => c.text(\"user\"))\n * app.route(\"/api\", app2) // GET /api/user\n * ```\n */\n route(path, app) {\n const subApp = this.basePath(path);\n app.routes.map((r) => {\n let handler;\n if (app.errorHandler === errorHandler) {\n handler = r.handler;\n } else {\n handler = async (c, next) => (await compose([], app.errorHandler)(c, () => r.handler(c, next))).res;\n handler[COMPOSED_HANDLER] = r.handler;\n }\n subApp.#addRoute(r.method, r.path, handler, r.basePath);\n });\n return this;\n }\n /**\n * `.basePath()` allows base paths to be specified.\n *\n * @see {@link https://hono.dev/docs/api/routing#base-path}\n *\n * @param {string} path - base Path\n * @returns {Hono} changed Hono instance\n *\n * @example\n * ```ts\n * const api = new Hono().basePath('/api')\n * ```\n */\n basePath(path) {\n const subApp = this.#clone();\n subApp._basePath = mergePath(this._basePath, path);\n return subApp;\n }\n /**\n * `.onError()` handles an error and returns a customized Response.\n *\n * @see {@link https://hono.dev/docs/api/hono#error-handling}\n *\n * @param {ErrorHandler} handler - request Handler for error\n * @returns {Hono} changed Hono instance\n *\n * @example\n * ```ts\n * app.onError((err, c) => {\n * console.error(`${err}`)\n * return c.text('Custom Error Message', 500)\n * })\n * ```\n */\n onError = (handler) => {\n this.errorHandler = handler;\n return this;\n };\n /**\n * `.notFound()` allows you to customize a Not Found Response.\n *\n * @see {@link https://hono.dev/docs/api/hono#not-found}\n *\n * @param {NotFoundHandler} handler - request handler for not-found\n * @returns {Hono} changed Hono instance\n *\n * @example\n * ```ts\n * app.notFound((c) => {\n * return c.text('Custom 404 Message', 404)\n * })\n * ```\n */\n notFound = (handler) => {\n this.#notFoundHandler = handler;\n return this;\n };\n /**\n * `.mount()` allows you to mount applications built with other frameworks into your Hono application.\n *\n * @see {@link https://hono.dev/docs/api/hono#mount}\n *\n * @param {string} path - base Path\n * @param {Function} applicationHandler - other Request Handler\n * @param {MountOptions} [options] - options of `.mount()`\n * @returns {Hono} mounted Hono instance\n *\n * @example\n * ```ts\n * import { Router as IttyRouter } from 'itty-router'\n * import { Hono } from 'hono'\n * // Create itty-router application\n * const ittyRouter = IttyRouter()\n * // GET /itty-router/hello\n * ittyRouter.get('/hello', () => new Response('Hello from itty-router'))\n *\n * const app = new Hono()\n * app.mount('/itty-router', ittyRouter.handle)\n * ```\n *\n * @example\n * ```ts\n * const app = new Hono()\n * // Send the request to another application without modification.\n * app.mount('/app', anotherApp, {\n * replaceRequest: (req) => req,\n * })\n * ```\n */\n mount(path, applicationHandler, options) {\n let replaceRequest;\n let optionHandler;\n if (options) {\n if (typeof options === \"function\") {\n optionHandler = options;\n } else {\n optionHandler = options.optionHandler;\n if (options.replaceRequest === false) {\n replaceRequest = (request) => request;\n } else {\n replaceRequest = options.replaceRequest;\n }\n }\n }\n const getOptions = optionHandler ? (c) => {\n const options2 = optionHandler(c);\n return Array.isArray(options2) ? options2 : [options2];\n } : (c) => {\n let executionContext = void 0;\n try {\n executionContext = c.executionCtx;\n } catch {\n }\n return [c.env, executionContext];\n };\n replaceRequest ||= (() => {\n const mergedPath = mergePath(this._basePath, path);\n const pathPrefixLength = mergedPath === \"/\" ? 0 : mergedPath.length;\n return (request) => {\n const url = new URL(request.url);\n url.pathname = this.getPath(request).slice(pathPrefixLength) || \"/\";\n return new Request(url, request);\n };\n })();\n const handler = async (c, next) => {\n const res = await applicationHandler(replaceRequest(c.req.raw), ...getOptions(c));\n if (res) {\n return res;\n }\n await next();\n };\n this.#addRoute(METHOD_NAME_ALL, mergePath(path, \"*\"), handler);\n return this;\n }\n #addRoute(method, path, handler, baseRoutePath) {\n method = method.toUpperCase();\n path = mergePath(this._basePath, path);\n const r = {\n basePath: baseRoutePath !== void 0 ? mergePath(this._basePath, baseRoutePath) : this._basePath,\n path,\n method,\n handler\n };\n this.router.add(method, path, [handler, r]);\n this.routes.push(r);\n }\n #handleError(err, c) {\n if (err instanceof Error) {\n return this.errorHandler(err, c);\n }\n throw err;\n }\n #dispatch(request, executionCtx, env, method) {\n if (method === \"HEAD\") {\n return (async () => new Response(null, await this.#dispatch(request, executionCtx, env, \"GET\")))();\n }\n const path = this.getPath(request, { env });\n const matchResult = this.router.match(method, path);\n const c = new Context(request, {\n path,\n matchResult,\n env,\n executionCtx,\n notFoundHandler: this.#notFoundHandler\n });\n if (matchResult[0].length === 1) {\n let res;\n try {\n res = matchResult[0][0][0][0](c, async () => {\n c.res = await this.#notFoundHandler(c);\n });\n } catch (err) {\n return this.#handleError(err, c);\n }\n return res instanceof Promise ? res.then(\n (resolved) => resolved || (c.finalized ? c.res : this.#notFoundHandler(c))\n ).catch((err) => this.#handleError(err, c)) : res ?? this.#notFoundHandler(c);\n }\n const composed = compose(matchResult[0], this.errorHandler, this.#notFoundHandler);\n return (async () => {\n try {\n const context = await composed(c);\n if (!context.finalized) {\n throw new Error(\n \"Context is not finalized. Did you forget to return a Response object or `await next()`?\"\n );\n }\n return context.res;\n } catch (err) {\n return this.#handleError(err, c);\n }\n })();\n }\n /**\n * `.fetch()` will be entry point of your app.\n *\n * @see {@link https://hono.dev/docs/api/hono#fetch}\n *\n * @param {Request} request - request Object of request\n * @param {Env} Env - env Object\n * @param {ExecutionContext} - context of execution\n * @returns {Response | Promise<Response>} response of request\n *\n */\n fetch = (request, ...rest) => {\n return this.#dispatch(request, rest[1], rest[0], request.method);\n };\n /**\n * `.request()` is a useful method for testing.\n * You can pass a URL or pathname to send a GET request.\n * app will return a Response object.\n * ```ts\n * test('GET /hello is ok', async () => {\n * const res = await app.request('/hello')\n * expect(res.status).toBe(200)\n * })\n * ```\n * @see https://hono.dev/docs/api/hono#request\n */\n request = (input, requestInit, Env, executionCtx) => {\n if (input instanceof Request) {\n return this.fetch(requestInit ? new Request(input, requestInit) : input, Env, executionCtx);\n }\n input = input.toString();\n return this.fetch(\n new Request(\n /^https?:\\/\\//.test(input) ? input : `http://localhost${mergePath(\"/\", input)}`,\n requestInit\n ),\n Env,\n executionCtx\n );\n };\n /**\n * `.fire()` automatically adds a global fetch event listener.\n * This can be useful for environments that adhere to the Service Worker API, such as non-ES module Cloudflare Workers.\n * @deprecated\n * Use `fire` from `hono/service-worker` instead.\n * ```ts\n * import { Hono } from 'hono'\n * import { fire } from 'hono/service-worker'\n *\n * const app = new Hono()\n * // ...\n * fire(app)\n * ```\n * @see https://hono.dev/docs/api/hono#fire\n * @see https://developer.mozilla.org/en-US/docs/Web/API/Service_Worker_API\n * @see https://developers.cloudflare.com/workers/reference/migrate-to-module-workers/\n */\n fire = () => {\n addEventListener(\"fetch\", (event) => {\n event.respondWith(this.#dispatch(event.request, event, void 0, event.request.method));\n });\n };\n};\nexport {\n Hono as HonoBase\n};\n","// src/router/reg-exp-router/matcher.ts\nimport { METHOD_NAME_ALL } from \"../../router.js\";\nvar emptyParam = [];\nfunction match(method, path) {\n const matchers = this.buildAllMatchers();\n const match2 = ((method2, path2) => {\n const matcher = matchers[method2] || matchers[METHOD_NAME_ALL];\n const staticMatch = matcher[2][path2];\n if (staticMatch) {\n return staticMatch;\n }\n const match3 = path2.match(matcher[0]);\n if (!match3) {\n return [[], emptyParam];\n }\n const index = match3.indexOf(\"\", 1);\n return [matcher[1][index], match3];\n });\n this.match = match2;\n return match2(method, path);\n}\nexport {\n emptyParam,\n match\n};\n","// src/router/reg-exp-router/node.ts\nvar LABEL_REG_EXP_STR = \"[^/]+\";\nvar ONLY_WILDCARD_REG_EXP_STR = \".*\";\nvar TAIL_WILDCARD_REG_EXP_STR = \"(?:|/.*)\";\nvar PATH_ERROR = /* @__PURE__ */ Symbol();\nvar regExpMetaChars = new Set(\".\\\\+*[^]$()\");\nfunction compareKey(a, b) {\n if (a.length === 1) {\n return b.length === 1 ? a < b ? -1 : 1 : -1;\n }\n if (b.length === 1) {\n return 1;\n }\n if (a === ONLY_WILDCARD_REG_EXP_STR || a === TAIL_WILDCARD_REG_EXP_STR) {\n return 1;\n } else if (b === ONLY_WILDCARD_REG_EXP_STR || b === TAIL_WILDCARD_REG_EXP_STR) {\n return -1;\n }\n if (a === LABEL_REG_EXP_STR) {\n return 1;\n } else if (b === LABEL_REG_EXP_STR) {\n return -1;\n }\n return a.length === b.length ? a < b ? -1 : 1 : b.length - a.length;\n}\nvar Node = class _Node {\n #index;\n #varIndex;\n #children = /* @__PURE__ */ Object.create(null);\n insert(tokens, index, paramMap, context, pathErrorCheckOnly) {\n if (tokens.length === 0) {\n if (this.#index !== void 0) {\n throw PATH_ERROR;\n }\n if (pathErrorCheckOnly) {\n return;\n }\n this.#index = index;\n return;\n }\n const [token, ...restTokens] = tokens;\n const pattern = token === \"*\" ? restTokens.length === 0 ? [\"\", \"\", ONLY_WILDCARD_REG_EXP_STR] : [\"\", \"\", LABEL_REG_EXP_STR] : token === \"/*\" ? [\"\", \"\", TAIL_WILDCARD_REG_EXP_STR] : token.match(/^\\:([^\\{\\}]+)(?:\\{(.+)\\})?$/);\n let node;\n if (pattern) {\n const name = pattern[1];\n let regexpStr = pattern[2] || LABEL_REG_EXP_STR;\n if (name && pattern[2]) {\n if (regexpStr === \".*\") {\n throw PATH_ERROR;\n }\n regexpStr = regexpStr.replace(/^\\((?!\\?:)(?=[^)]+\\)$)/, \"(?:\");\n if (/\\((?!\\?:)/.test(regexpStr)) {\n throw PATH_ERROR;\n }\n }\n node = this.#children[regexpStr];\n if (!node) {\n if (Object.keys(this.#children).some(\n (k) => k !== ONLY_WILDCARD_REG_EXP_STR && k !== TAIL_WILDCARD_REG_EXP_STR\n )) {\n throw PATH_ERROR;\n }\n if (pathErrorCheckOnly) {\n return;\n }\n node = this.#children[regexpStr] = new _Node();\n if (name !== \"\") {\n node.#varIndex = context.varIndex++;\n }\n }\n if (!pathErrorCheckOnly && name !== \"\") {\n paramMap.push([name, node.#varIndex]);\n }\n } else {\n node = this.#children[token];\n if (!node) {\n if (Object.keys(this.#children).some(\n (k) => k.length > 1 && k !== ONLY_WILDCARD_REG_EXP_STR && k !== TAIL_WILDCARD_REG_EXP_STR\n )) {\n throw PATH_ERROR;\n }\n if (pathErrorCheckOnly) {\n return;\n }\n node = this.#children[token] = new _Node();\n }\n }\n node.insert(restTokens, index, paramMap, context, pathErrorCheckOnly);\n }\n buildRegExpStr() {\n const childKeys = Object.keys(this.#children).sort(compareKey);\n const strList = childKeys.map((k) => {\n const c = this.#children[k];\n return (typeof c.#varIndex === \"number\" ? `(${k})@${c.#varIndex}` : regExpMetaChars.has(k) ? `\\\\${k}` : k) + c.buildRegExpStr();\n });\n if (typeof this.#index === \"number\") {\n strList.unshift(`#${this.#index}`);\n }\n if (strList.length === 0) {\n return \"\";\n }\n if (strList.length === 1) {\n return strList[0];\n }\n return \"(?:\" + strList.join(\"|\") + \")\";\n }\n};\nexport {\n Node,\n PATH_ERROR\n};\n","// src/router/reg-exp-router/trie.ts\nimport { Node } from \"./node.js\";\nvar Trie = class {\n #context = { varIndex: 0 };\n #root = new Node();\n insert(path, index, pathErrorCheckOnly) {\n const paramAssoc = [];\n const groups = [];\n for (let i = 0; ; ) {\n let replaced = false;\n path = path.replace(/\\{[^}]+\\}/g, (m) => {\n const mark = `@\\\\${i}`;\n groups[i] = [mark, m];\n i++;\n replaced = true;\n return mark;\n });\n if (!replaced) {\n break;\n }\n }\n const tokens = path.match(/(?::[^\\/]+)|(?:\\/\\*$)|./g) || [];\n for (let i = groups.length - 1; i >= 0; i--) {\n const [mark] = groups[i];\n for (let j = tokens.length - 1; j >= 0; j--) {\n if (tokens[j].indexOf(mark) !== -1) {\n tokens[j] = tokens[j].replace(mark, groups[i][1]);\n break;\n }\n }\n }\n this.#root.insert(tokens, index, paramAssoc, this.#context, pathErrorCheckOnly);\n return paramAssoc;\n }\n buildRegExp() {\n let regexp = this.#root.buildRegExpStr();\n if (regexp === \"\") {\n return [/^$/, [], []];\n }\n let captureIndex = 0;\n const indexReplacementMap = [];\n const paramReplacementMap = [];\n regexp = regexp.replace(/#(\\d+)|@(\\d+)|\\.\\*\\$/g, (_, handlerIndex, paramIndex) => {\n if (handlerIndex !== void 0) {\n indexReplacementMap[++captureIndex] = Number(handlerIndex);\n return \"$()\";\n }\n if (paramIndex !== void 0) {\n paramReplacementMap[Number(paramIndex)] = ++captureIndex;\n return \"\";\n }\n return \"\";\n });\n return [new RegExp(`^${regexp}`), indexReplacementMap, paramReplacementMap];\n }\n};\nexport {\n Trie\n};\n","// src/router/reg-exp-router/router.ts\nimport {\n MESSAGE_MATCHER_IS_ALREADY_BUILT,\n METHOD_NAME_ALL,\n UnsupportedPathError\n} from \"../../router.js\";\nimport { checkOptionalParameter } from \"../../utils/url.js\";\nimport { match, emptyParam } from \"./matcher.js\";\nimport { PATH_ERROR } from \"./node.js\";\nimport { Trie } from \"./trie.js\";\nvar nullMatcher = [/^$/, [], /* @__PURE__ */ Object.create(null)];\nvar wildcardRegExpCache = /* @__PURE__ */ Object.create(null);\nfunction buildWildcardRegExp(path) {\n return wildcardRegExpCache[path] ??= new RegExp(\n path === \"*\" ? \"\" : `^${path.replace(\n /\\/\\*$|([.\\\\+*[^\\]$()])/g,\n (_, metaChar) => metaChar ? `\\\\${metaChar}` : \"(?:|/.*)\"\n )}$`\n );\n}\nfunction clearWildcardRegExpCache() {\n wildcardRegExpCache = /* @__PURE__ */ Object.create(null);\n}\nfunction buildMatcherFromPreprocessedRoutes(routes) {\n const trie = new Trie();\n const handlerData = [];\n if (routes.length === 0) {\n return nullMatcher;\n }\n const routesWithStaticPathFlag = routes.map(\n (route) => [!/\\*|\\/:/.test(route[0]), ...route]\n ).sort(\n ([isStaticA, pathA], [isStaticB, pathB]) => isStaticA ? 1 : isStaticB ? -1 : pathA.length - pathB.length\n );\n const staticMap = /* @__PURE__ */ Object.create(null);\n for (let i = 0, j = -1, len = routesWithStaticPathFlag.length; i < len; i++) {\n const [pathErrorCheckOnly, path, handlers] = routesWithStaticPathFlag[i];\n if (pathErrorCheckOnly) {\n staticMap[path] = [handlers.map(([h]) => [h, /* @__PURE__ */ Object.create(null)]), emptyParam];\n } else {\n j++;\n }\n let paramAssoc;\n try {\n paramAssoc = trie.insert(path, j, pathErrorCheckOnly);\n } catch (e) {\n throw e === PATH_ERROR ? new UnsupportedPathError(path) : e;\n }\n if (pathErrorCheckOnly) {\n continue;\n }\n handlerData[j] = handlers.map(([h, paramCount]) => {\n const paramIndexMap = /* @__PURE__ */ Object.create(null);\n paramCount -= 1;\n for (; paramCount >= 0; paramCount--) {\n const [key, value] = paramAssoc[paramCount];\n paramIndexMap[key] = value;\n }\n return [h, paramIndexMap];\n });\n }\n const [regexp, indexReplacementMap, paramReplacementMap] = trie.buildRegExp();\n for (let i = 0, len = handlerData.length; i < len; i++) {\n for (let j = 0, len2 = handlerData[i].length; j < len2; j++) {\n const map = handlerData[i][j]?.[1];\n if (!map) {\n continue;\n }\n const keys = Object.keys(map);\n for (let k = 0, len3 = keys.length; k < len3; k++) {\n map[keys[k]] = paramReplacementMap[map[keys[k]]];\n }\n }\n }\n const handlerMap = [];\n for (const i in indexReplacementMap) {\n handlerMap[i] = handlerData[indexReplacementMap[i]];\n }\n return [regexp, handlerMap, staticMap];\n}\nfunction findMiddleware(middleware, path) {\n if (!middleware) {\n return void 0;\n }\n for (const k of Object.keys(middleware).sort((a, b) => b.length - a.length)) {\n if (buildWildcardRegExp(k).test(path)) {\n return [...middleware[k]];\n }\n }\n return void 0;\n}\nvar RegExpRouter = class {\n name = \"RegExpRouter\";\n #middleware;\n #routes;\n constructor() {\n this.#middleware = { [METHOD_NAME_ALL]: /* @__PURE__ */ Object.create(null) };\n this.#routes = { [METHOD_NAME_ALL]: /* @__PURE__ */ Object.create(null) };\n }\n add(method, path, handler) {\n const middleware = this.#middleware;\n const routes = this.#routes;\n if (!middleware || !routes) {\n throw new Error(MESSAGE_MATCHER_IS_ALREADY_BUILT);\n }\n if (!middleware[method]) {\n ;\n [middleware, routes].forEach((handlerMap) => {\n handlerMap[method] = /* @__PURE__ */ Object.create(null);\n Object.keys(handlerMap[METHOD_NAME_ALL]).forEach((p) => {\n handlerMap[method][p] = [...handlerMap[METHOD_NAME_ALL][p]];\n });\n });\n }\n if (path === \"/*\") {\n path = \"*\";\n }\n const paramCount = (path.match(/\\/:/g) || []).length;\n if (/\\*$/.test(path)) {\n const re = buildWildcardRegExp(path);\n if (method === METHOD_NAME_ALL) {\n Object.keys(middleware).forEach((m) => {\n middleware[m][path] ||= findMiddleware(middleware[m], path) || findMiddleware(middleware[METHOD_NAME_ALL], path) || [];\n });\n } else {\n middleware[method][path] ||= findMiddleware(middleware[method], path) || findMiddleware(middleware[METHOD_NAME_ALL], path) || [];\n }\n Object.keys(middleware).forEach((m) => {\n if (method === METHOD_NAME_ALL || method === m) {\n Object.keys(middleware[m]).forEach((p) => {\n re.test(p) && middleware[m][p].push([handler, paramCount]);\n });\n }\n });\n Object.keys(routes).forEach((m) => {\n if (method === METHOD_NAME_ALL || method === m) {\n Object.keys(routes[m]).forEach(\n (p) => re.test(p) && routes[m][p].push([handler, paramCount])\n );\n }\n });\n return;\n }\n const paths = checkOptionalParameter(path) || [path];\n for (let i = 0, len = paths.length; i < len; i++) {\n const path2 = paths[i];\n Object.keys(routes).forEach((m) => {\n if (method === METHOD_NAME_ALL || method === m) {\n routes[m][path2] ||= [\n ...findMiddleware(middleware[m], path2) || findMiddleware(middleware[METHOD_NAME_ALL], path2) || []\n ];\n routes[m][path2].push([handler, paramCount - len + i + 1]);\n }\n });\n }\n }\n match = match;\n buildAllMatchers() {\n const matchers = /* @__PURE__ */ Object.create(null);\n Object.keys(this.#routes).concat(Object.keys(this.#middleware)).forEach((method) => {\n matchers[method] ||= this.#buildMatcher(method);\n });\n this.#middleware = this.#routes = void 0;\n clearWildcardRegExpCache();\n return matchers;\n }\n #buildMatcher(method) {\n const routes = [];\n let hasOwnRoute = method === METHOD_NAME_ALL;\n [this.#middleware, this.#routes].forEach((r) => {\n const ownRoute = r[method] ? Object.keys(r[method]).map((path) => [path, r[method][path]]) : [];\n if (ownRoute.length !== 0) {\n hasOwnRoute ||= true;\n routes.push(...ownRoute);\n } else if (method !== METHOD_NAME_ALL) {\n routes.push(\n ...Object.keys(r[METHOD_NAME_ALL]).map((path) => [path, r[METHOD_NAME_ALL][path]])\n );\n }\n });\n if (!hasOwnRoute) {\n return null;\n } else {\n return buildMatcherFromPreprocessedRoutes(routes);\n }\n }\n};\nexport {\n RegExpRouter\n};\n","// src/router/reg-exp-router/prepared-router.ts\nimport { METHOD_NAME_ALL } from \"../../router.js\";\nimport { match, emptyParam } from \"./matcher.js\";\nimport { RegExpRouter } from \"./router.js\";\nvar PreparedRegExpRouter = class {\n name = \"PreparedRegExpRouter\";\n #matchers;\n #relocateMap;\n constructor(matchers, relocateMap) {\n this.#matchers = matchers;\n this.#relocateMap = relocateMap;\n }\n #addWildcard(method, handlerData) {\n const matcher = this.#matchers[method];\n matcher[1].forEach((list) => list && list.push(handlerData));\n Object.values(matcher[2]).forEach((list) => list[0].push(handlerData));\n }\n #addPath(method, path, handler, indexes, map) {\n const matcher = this.#matchers[method];\n if (!map) {\n matcher[2][path][0].push([handler, {}]);\n } else {\n indexes.forEach((index) => {\n if (typeof index === \"number\") {\n matcher[1][index].push([handler, map]);\n } else {\n ;\n matcher[2][index || path][0].push([handler, map]);\n }\n });\n }\n }\n add(method, path, handler) {\n if (!this.#matchers[method]) {\n const all = this.#matchers[METHOD_NAME_ALL];\n const staticMap = {};\n for (const key in all[2]) {\n staticMap[key] = [all[2][key][0].slice(), emptyParam];\n }\n this.#matchers[method] = [\n all[0],\n all[1].map((list) => Array.isArray(list) ? list.slice() : 0),\n staticMap\n ];\n }\n if (path === \"/*\" || path === \"*\") {\n const handlerData = [handler, {}];\n if (method === METHOD_NAME_ALL) {\n for (const m in this.#matchers) {\n this.#addWildcard(m, handlerData);\n }\n } else {\n this.#addWildcard(method, handlerData);\n }\n return;\n }\n const data = this.#relocateMap[path];\n if (!data) {\n throw new Error(`Path ${path} is not registered`);\n }\n for (const [indexes, map] of data) {\n if (method === METHOD_NAME_ALL) {\n for (const m in this.#matchers) {\n this.#addPath(m, path, handler, indexes, map);\n }\n } else {\n this.#addPath(method, path, handler, indexes, map);\n }\n }\n }\n buildAllMatchers() {\n return this.#matchers;\n }\n match = match;\n};\nvar buildInitParams = ({ paths }) => {\n const RegExpRouterWithMatcherExport = class extends RegExpRouter {\n buildAndExportAllMatchers() {\n return this.buildAllMatchers();\n }\n };\n const router = new RegExpRouterWithMatcherExport();\n for (const path of paths) {\n router.add(METHOD_NAME_ALL, path, path);\n }\n const matchers = router.buildAndExportAllMatchers();\n const all = matchers[METHOD_NAME_ALL];\n const relocateMap = {};\n for (const path of paths) {\n if (path === \"/*\" || path === \"*\") {\n continue;\n }\n all[1].forEach((list, i) => {\n list.forEach(([p, map]) => {\n if (p === path) {\n if (relocateMap[path]) {\n relocateMap[path][0][1] = {\n ...relocateMap[path][0][1],\n ...map\n };\n } else {\n relocateMap[path] = [[[], map]];\n }\n if (relocateMap[path][0][0].findIndex((j) => j === i) === -1) {\n relocateMap[path][0][0].push(i);\n }\n }\n });\n });\n for (const path2 in all[2]) {\n all[2][path2][0].forEach(([p]) => {\n if (p === path) {\n relocateMap[path] ||= [[[]]];\n const value = path2 === path ? \"\" : path2;\n if (relocateMap[path][0][0].findIndex((v) => v === value) === -1) {\n relocateMap[path][0][0].push(value);\n }\n }\n });\n }\n }\n for (let i = 0, len = all[1].length; i < len; i++) {\n all[1][i] = all[1][i] ? [] : 0;\n }\n for (const path in all[2]) {\n all[2][path][0] = [];\n }\n return [matchers, relocateMap];\n};\nvar serializeInitParams = ([matchers, relocateMap]) => {\n const matchersStr = JSON.stringify(\n matchers,\n (_, value) => value instanceof RegExp ? `##${value.toString()}##` : value\n ).replace(/\"##(.+?)##\"/g, (_, str) => str.replace(/\\\\\\\\/g, \"\\\\\"));\n const relocateMapStr = JSON.stringify(relocateMap);\n return `[${matchersStr},${relocateMapStr}]`;\n};\nexport {\n PreparedRegExpRouter,\n buildInitParams,\n serializeInitParams\n};\n","// src/router/reg-exp-router/index.ts\nimport { RegExpRouter } from \"./router.js\";\nimport { PreparedRegExpRouter, buildInitParams, serializeInitParams } from \"./prepared-router.js\";\nexport {\n PreparedRegExpRouter,\n RegExpRouter,\n buildInitParams,\n serializeInitParams\n};\n","// src/router/smart-router/router.ts\nimport { MESSAGE_MATCHER_IS_ALREADY_BUILT, UnsupportedPathError } from \"../../router.js\";\nvar SmartRouter = class {\n name = \"SmartRouter\";\n #routers = [];\n #routes = [];\n constructor(init) {\n this.#routers = init.routers;\n }\n add(method, path, handler) {\n if (!this.#routes) {\n throw new Error(MESSAGE_MATCHER_IS_ALREADY_BUILT);\n }\n this.#routes.push([method, path, handler]);\n }\n match(method, path) {\n if (!this.#routes) {\n throw new Error(\"Fatal error\");\n }\n const routers = this.#routers;\n const routes = this.#routes;\n const len = routers.length;\n let i = 0;\n let res;\n for (; i < len; i++) {\n const router = routers[i];\n try {\n for (let i2 = 0, len2 = routes.length; i2 < len2; i2++) {\n router.add(...routes[i2]);\n }\n res = router.match(method, path);\n } catch (e) {\n if (e instanceof UnsupportedPathError) {\n continue;\n }\n throw e;\n }\n this.match = router.match.bind(router);\n this.#routers = [router];\n this.#routes = void 0;\n break;\n }\n if (i === len) {\n throw new Error(\"Fatal error\");\n }\n this.name = `SmartRouter + ${this.activeRouter.name}`;\n return res;\n }\n get activeRouter() {\n if (this.#routes || this.#routers.length !== 1) {\n throw new Error(\"No active router has been determined yet.\");\n }\n return this.#routers[0];\n }\n};\nexport {\n SmartRouter\n};\n","// src/router/smart-router/index.ts\nimport { SmartRouter } from \"./router.js\";\nexport {\n SmartRouter\n};\n","// src/router/trie-router/node.ts\nimport { METHOD_NAME_ALL } from \"../../router.js\";\nimport { getPattern, splitPath, splitRoutingPath } from \"../../utils/url.js\";\nvar emptyParams = /* @__PURE__ */ Object.create(null);\nvar hasChildren = (children) => {\n for (const _ in children) {\n return true;\n }\n return false;\n};\nvar Node = class _Node {\n #methods;\n #children;\n #patterns;\n #order = 0;\n #params = emptyParams;\n constructor(method, handler, children) {\n this.#children = children || /* @__PURE__ */ Object.create(null);\n this.#methods = [];\n if (method && handler) {\n const m = /* @__PURE__ */ Object.create(null);\n m[method] = { handler, possibleKeys: [], score: 0 };\n this.#methods = [m];\n }\n this.#patterns = [];\n }\n insert(method, path, handler) {\n this.#order = ++this.#order;\n let curNode = this;\n const parts = splitRoutingPath(path);\n const possibleKeys = [];\n for (let i = 0, len = parts.length; i < len; i++) {\n const p = parts[i];\n const nextP = parts[i + 1];\n const pattern = getPattern(p, nextP);\n const key = Array.isArray(pattern) ? pattern[0] : p;\n if (key in curNode.#children) {\n curNode = curNode.#children[key];\n if (pattern) {\n possibleKeys.push(pattern[1]);\n }\n continue;\n }\n curNode.#children[key] = new _Node();\n if (pattern) {\n curNode.#patterns.push(pattern);\n possibleKeys.push(pattern[1]);\n }\n curNode = curNode.#children[key];\n }\n curNode.#methods.push({\n [method]: {\n handler,\n possibleKeys: possibleKeys.filter((v, i, a) => a.indexOf(v) === i),\n score: this.#order\n }\n });\n return curNode;\n }\n #pushHandlerSets(handlerSets, node, method, nodeParams, params) {\n for (let i = 0, len = node.#methods.length; i < len; i++) {\n const m = node.#methods[i];\n const handlerSet = m[method] || m[METHOD_NAME_ALL];\n const processedSet = {};\n if (handlerSet !== void 0) {\n handlerSet.params = /* @__PURE__ */ Object.create(null);\n handlerSets.push(handlerSet);\n if (nodeParams !== emptyParams || params && params !== emptyParams) {\n for (let i2 = 0, len2 = handlerSet.possibleKeys.length; i2 < len2; i2++) {\n const key = handlerSet.possibleKeys[i2];\n const processed = processedSet[handlerSet.score];\n handlerSet.params[key] = params?.[key] && !processed ? params[key] : nodeParams[key] ?? params?.[key];\n processedSet[handlerSet.score] = true;\n }\n }\n }\n }\n }\n search(method, path) {\n const handlerSets = [];\n this.#params = emptyParams;\n const curNode = this;\n let curNodes = [curNode];\n const parts = splitPath(path);\n const curNodesQueue = [];\n const len = parts.length;\n let partOffsets = null;\n for (let i = 0; i < len; i++) {\n const part = parts[i];\n const isLast = i === len - 1;\n const tempNodes = [];\n for (let j = 0, len2 = curNodes.length; j < len2; j++) {\n const node = curNodes[j];\n const nextNode = node.#children[part];\n if (nextNode) {\n nextNode.#params = node.#params;\n if (isLast) {\n if (nextNode.#children[\"*\"]) {\n this.#pushHandlerSets(handlerSets, nextNode.#children[\"*\"], method, node.#params);\n }\n this.#pushHandlerSets(handlerSets, nextNode, method, node.#params);\n } else {\n tempNodes.push(nextNode);\n }\n }\n for (let k = 0, len3 = node.#patterns.length; k < len3; k++) {\n const pattern = node.#patterns[k];\n const params = node.#params === emptyParams ? {} : { ...node.#params };\n if (pattern === \"*\") {\n const astNode = node.#children[\"*\"];\n if (astNode) {\n this.#pushHandlerSets(handlerSets, astNode, method, node.#params);\n astNode.#params = params;\n tempNodes.push(astNode);\n }\n continue;\n }\n const [key, name, matcher] = pattern;\n if (!part && !(matcher instanceof RegExp)) {\n continue;\n }\n const child = node.#children[key];\n if (matcher instanceof RegExp) {\n if (partOffsets === null) {\n partOffsets = new Array(len);\n let offset = path[0] === \"/\" ? 1 : 0;\n for (let p = 0; p < len; p++) {\n partOffsets[p] = offset;\n offset += parts[p].length + 1;\n }\n }\n const restPathString = path.substring(partOffsets[i]);\n const m = matcher.exec(restPathString);\n if (m) {\n params[name] = m[0];\n this.#pushHandlerSets(handlerSets, child, method, node.#params, params);\n if (hasChildren(child.#children)) {\n child.#params = params;\n const componentCount = m[0].match(/\\//)?.length ?? 0;\n const targetCurNodes = curNodesQueue[componentCount] ||= [];\n targetCurNodes.push(child);\n }\n continue;\n }\n }\n if (matcher === true || matcher.test(part)) {\n params[name] = part;\n if (isLast) {\n this.#pushHandlerSets(handlerSets, child, method, params, node.#params);\n if (child.#children[\"*\"]) {\n this.#pushHandlerSets(\n handlerSets,\n child.#children[\"*\"],\n method,\n params,\n node.#params\n );\n }\n } else {\n child.#params = params;\n tempNodes.push(child);\n }\n }\n }\n }\n const shifted = curNodesQueue.shift();\n curNodes = shifted ? tempNodes.concat(shifted) : tempNodes;\n }\n if (handlerSets.length > 1) {\n handlerSets.sort((a, b) => {\n return a.score - b.score;\n });\n }\n return [handlerSets.map(({ handler, params }) => [handler, params])];\n }\n};\nexport {\n Node\n};\n","// src/router/trie-router/router.ts\nimport { checkOptionalParameter } from \"../../utils/url.js\";\nimport { Node } from \"./node.js\";\nvar TrieRouter = class {\n name = \"TrieRouter\";\n #node;\n constructor() {\n this.#node = new Node();\n }\n add(method, path, handler) {\n const results = checkOptionalParameter(path);\n if (results) {\n for (let i = 0, len = results.length; i < len; i++) {\n this.#node.insert(method, results[i], handler);\n }\n return;\n }\n this.#node.insert(method, path, handler);\n }\n match(method, path) {\n return this.#node.search(method, path);\n }\n};\nexport {\n TrieRouter\n};\n","// src/router/trie-router/index.ts\nimport { TrieRouter } from \"./router.js\";\nexport {\n TrieRouter\n};\n","// src/hono.ts\nimport { HonoBase } from \"./hono-base.js\";\nimport { RegExpRouter } from \"./router/reg-exp-router/index.js\";\nimport { SmartRouter } from \"./router/smart-router/index.js\";\nimport { TrieRouter } from \"./router/trie-router/index.js\";\nvar Hono = class extends HonoBase {\n /**\n * Creates an instance of the Hono class.\n *\n * @param options - Optional configuration options for the Hono instance.\n */\n constructor(options = {}) {\n super(options);\n this.router = options.router ?? new SmartRouter({\n routers: [new RegExpRouter(), new TrieRouter()]\n });\n }\n};\nexport {\n Hono\n};\n","// src/index.ts\nimport { Hono } from \"./hono.js\";\nimport { Context } from \"./context.js\";\nexport {\n Context,\n Hono\n};\n","// src/middleware/cors/index.ts\nvar cors = (options) => {\n const opts = {\n origin: \"*\",\n allowMethods: [\"GET\", \"HEAD\", \"PUT\", \"POST\", \"DELETE\", \"PATCH\"],\n allowHeaders: [],\n exposeHeaders: [],\n ...options\n };\n const findAllowOrigin = ((optsOrigin) => {\n if (typeof optsOrigin === \"string\") {\n if (optsOrigin === \"*\") {\n if (opts.credentials) {\n return (origin) => origin || null;\n }\n return () => optsOrigin;\n } else {\n return (origin) => optsOrigin === origin ? origin : null;\n }\n } else if (typeof optsOrigin === \"function\") {\n return optsOrigin;\n } else {\n return (origin) => optsOrigin.includes(origin) ? origin : null;\n }\n })(opts.origin);\n const findAllowMethods = ((optsAllowMethods) => {\n if (typeof optsAllowMethods === \"function\") {\n return optsAllowMethods;\n } else if (Array.isArray(optsAllowMethods)) {\n return () => optsAllowMethods;\n } else {\n return () => [];\n }\n })(opts.allowMethods);\n return async function cors2(c, next) {\n function set(key, value) {\n c.res.headers.set(key, value);\n }\n const allowOrigin = await findAllowOrigin(c.req.header(\"origin\") || \"\", c);\n if (allowOrigin) {\n set(\"Access-Control-Allow-Origin\", allowOrigin);\n }\n if (opts.credentials) {\n set(\"Access-Control-Allow-Credentials\", \"true\");\n }\n if (opts.exposeHeaders?.length) {\n set(\"Access-Control-Expose-Headers\", opts.exposeHeaders.join(\",\"));\n }\n if (c.req.method === \"OPTIONS\") {\n if (opts.origin !== \"*\" || opts.credentials) {\n set(\"Vary\", \"Origin\");\n }\n if (opts.maxAge != null) {\n set(\"Access-Control-Max-Age\", opts.maxAge.toString());\n }\n const allowMethods = await findAllowMethods(c.req.header(\"origin\") || \"\", c);\n if (allowMethods.length) {\n set(\"Access-Control-Allow-Methods\", allowMethods.join(\",\"));\n }\n let headers = opts.allowHeaders;\n if (!headers?.length) {\n const requestHeaders = c.req.header(\"Access-Control-Request-Headers\");\n if (requestHeaders) {\n headers = requestHeaders.split(/\\s*,\\s*/);\n }\n }\n if (headers?.length) {\n set(\"Access-Control-Allow-Headers\", headers.join(\",\"));\n c.res.headers.append(\"Vary\", \"Access-Control-Request-Headers\");\n }\n c.res.headers.delete(\"Content-Length\");\n c.res.headers.delete(\"Content-Type\");\n return new Response(null, {\n headers: c.res.headers,\n status: 204,\n statusText: \"No Content\"\n });\n }\n await next();\n if (opts.origin !== \"*\" || opts.credentials) {\n c.header(\"Vary\", \"Origin\", { append: true });\n }\n };\n};\nexport {\n cors\n};\n","// src/server.ts\nimport { createServer as createServerHTTP } from \"http\";\n\n// src/listener.ts\nimport { Http2ServerRequest as Http2ServerRequest2, constants as h2constants } from \"http2\";\n\n// src/request.ts\nimport { Http2ServerRequest } from \"http2\";\nimport { Readable } from \"stream\";\nvar RequestError = class extends Error {\n constructor(message, options) {\n super(message, options);\n this.name = \"RequestError\";\n }\n};\nvar toRequestError = (e) => {\n if (e instanceof RequestError) {\n return e;\n }\n return new RequestError(e.message, { cause: e });\n};\nvar GlobalRequest = global.Request;\nvar Request = class extends GlobalRequest {\n constructor(input, options) {\n if (typeof input === \"object\" && getRequestCache in input) {\n input = input[getRequestCache]();\n }\n if (typeof options?.body?.getReader !== \"undefined\") {\n ;\n options.duplex ??= \"half\";\n }\n super(input, options);\n }\n};\nvar newHeadersFromIncoming = (incoming) => {\n const headerRecord = [];\n const rawHeaders = incoming.rawHeaders;\n for (let i = 0; i < rawHeaders.length; i += 2) {\n const { [i]: key, [i + 1]: value } = rawHeaders;\n if (key.charCodeAt(0) !== /*:*/\n 58) {\n headerRecord.push([key, value]);\n }\n }\n return new Headers(headerRecord);\n};\nvar wrapBodyStream = Symbol(\"wrapBodyStream\");\nvar newRequestFromIncoming = (method, url, headers, incoming, abortController) => {\n const init = {\n method,\n headers,\n signal: abortController.signal\n };\n if (method === \"TRACE\") {\n init.method = \"GET\";\n const req = new Request(url, init);\n Object.defineProperty(req, \"method\", {\n get() {\n return \"TRACE\";\n }\n });\n return req;\n }\n if (!(method === \"GET\" || method === \"HEAD\")) {\n if (\"rawBody\" in incoming && incoming.rawBody instanceof Buffer) {\n init.body = new ReadableStream({\n start(controller) {\n controller.enqueue(incoming.rawBody);\n controller.close();\n }\n });\n } else if (incoming[wrapBodyStream]) {\n let reader;\n init.body = new ReadableStream({\n async pull(controller) {\n try {\n reader ||= Readable.toWeb(incoming).getReader();\n const { done, value } = await reader.read();\n if (done) {\n controller.close();\n } else {\n controller.enqueue(value);\n }\n } catch (error) {\n controller.error(error);\n }\n }\n });\n } else {\n init.body = Readable.toWeb(incoming);\n }\n }\n return new Request(url, init);\n};\nvar getRequestCache = Symbol(\"getRequestCache\");\nvar requestCache = Symbol(\"requestCache\");\nvar incomingKey = Symbol(\"incomingKey\");\nvar urlKey = Symbol(\"urlKey\");\nvar headersKey = Symbol(\"headersKey\");\nvar abortControllerKey = Symbol(\"abortControllerKey\");\nvar getAbortController = Symbol(\"getAbortController\");\nvar requestPrototype = {\n get method() {\n return this[incomingKey].method || \"GET\";\n },\n get url() {\n return this[urlKey];\n },\n get headers() {\n return this[headersKey] ||= newHeadersFromIncoming(this[incomingKey]);\n },\n [getAbortController]() {\n this[getRequestCache]();\n return this[abortControllerKey];\n },\n [getRequestCache]() {\n this[abortControllerKey] ||= new AbortController();\n return this[requestCache] ||= newRequestFromIncoming(\n this.method,\n this[urlKey],\n this.headers,\n this[incomingKey],\n this[abortControllerKey]\n );\n }\n};\n[\n \"body\",\n \"bodyUsed\",\n \"cache\",\n \"credentials\",\n \"destination\",\n \"integrity\",\n \"mode\",\n \"redirect\",\n \"referrer\",\n \"referrerPolicy\",\n \"signal\",\n \"keepalive\"\n].forEach((k) => {\n Object.defineProperty(requestPrototype, k, {\n get() {\n return this[getRequestCache]()[k];\n }\n });\n});\n[\"arrayBuffer\", \"blob\", \"clone\", \"formData\", \"json\", \"text\"].forEach((k) => {\n Object.defineProperty(requestPrototype, k, {\n value: function() {\n return this[getRequestCache]()[k]();\n }\n });\n});\nObject.defineProperty(requestPrototype, Symbol.for(\"nodejs.util.inspect.custom\"), {\n value: function(depth, options, inspectFn) {\n const props = {\n method: this.method,\n url: this.url,\n headers: this.headers,\n nativeRequest: this[requestCache]\n };\n return `Request (lightweight) ${inspectFn(props, { ...options, depth: depth == null ? null : depth - 1 })}`;\n }\n});\nObject.setPrototypeOf(requestPrototype, Request.prototype);\nvar newRequest = (incoming, defaultHostname) => {\n const req = Object.create(requestPrototype);\n req[incomingKey] = incoming;\n const incomingUrl = incoming.url || \"\";\n if (incomingUrl[0] !== \"/\" && // short-circuit for performance. most requests are relative URL.\n (incomingUrl.startsWith(\"http://\") || incomingUrl.startsWith(\"https://\"))) {\n if (incoming instanceof Http2ServerRequest) {\n throw new RequestError(\"Absolute URL for :path is not allowed in HTTP/2\");\n }\n try {\n const url2 = new URL(incomingUrl);\n req[urlKey] = url2.href;\n } catch (e) {\n throw new RequestError(\"Invalid absolute URL\", { cause: e });\n }\n return req;\n }\n const host = (incoming instanceof Http2ServerRequest ? incoming.authority : incoming.headers.host) || defaultHostname;\n if (!host) {\n throw new RequestError(\"Missing host header\");\n }\n let scheme;\n if (incoming instanceof Http2ServerRequest) {\n scheme = incoming.scheme;\n if (!(scheme === \"http\" || scheme === \"https\")) {\n throw new RequestError(\"Unsupported scheme\");\n }\n } else {\n scheme = incoming.socket && incoming.socket.encrypted ? \"https\" : \"http\";\n }\n const url = new URL(`${scheme}://${host}${incomingUrl}`);\n if (url.hostname.length !== host.length && url.hostname !== host.replace(/:\\d+$/, \"\")) {\n throw new RequestError(\"Invalid host header\");\n }\n req[urlKey] = url.href;\n return req;\n};\n\n// src/response.ts\nvar responseCache = Symbol(\"responseCache\");\nvar getResponseCache = Symbol(\"getResponseCache\");\nvar cacheKey = Symbol(\"cache\");\nvar GlobalResponse = global.Response;\nvar Response2 = class _Response {\n #body;\n #init;\n [getResponseCache]() {\n delete this[cacheKey];\n return this[responseCache] ||= new GlobalResponse(this.#body, this.#init);\n }\n constructor(body, init) {\n let headers;\n this.#body = body;\n if (init instanceof _Response) {\n const cachedGlobalResponse = init[responseCache];\n if (cachedGlobalResponse) {\n this.#init = cachedGlobalResponse;\n this[getResponseCache]();\n return;\n } else {\n this.#init = init.#init;\n headers = new Headers(init.#init.headers);\n }\n } else {\n this.#init = init;\n }\n if (typeof body === \"string\" || typeof body?.getReader !== \"undefined\" || body instanceof Blob || body instanceof Uint8Array) {\n ;\n this[cacheKey] = [init?.status || 200, body, headers || init?.headers];\n }\n }\n get headers() {\n const cache = this[cacheKey];\n if (cache) {\n if (!(cache[2] instanceof Headers)) {\n cache[2] = new Headers(\n cache[2] || { \"content-type\": \"text/plain; charset=UTF-8\" }\n );\n }\n return cache[2];\n }\n return this[getResponseCache]().headers;\n }\n get status() {\n return this[cacheKey]?.[0] ?? this[getResponseCache]().status;\n }\n get ok() {\n const status = this.status;\n return status >= 200 && status < 300;\n }\n};\n[\"body\", \"bodyUsed\", \"redirected\", \"statusText\", \"trailers\", \"type\", \"url\"].forEach((k) => {\n Object.defineProperty(Response2.prototype, k, {\n get() {\n return this[getResponseCache]()[k];\n }\n });\n});\n[\"arrayBuffer\", \"blob\", \"clone\", \"formData\", \"json\", \"text\"].forEach((k) => {\n Object.defineProperty(Response2.prototype, k, {\n value: function() {\n return this[getResponseCache]()[k]();\n }\n });\n});\nObject.defineProperty(Response2.prototype, Symbol.for(\"nodejs.util.inspect.custom\"), {\n value: function(depth, options, inspectFn) {\n const props = {\n status: this.status,\n headers: this.headers,\n ok: this.ok,\n nativeResponse: this[responseCache]\n };\n return `Response (lightweight) ${inspectFn(props, { ...options, depth: depth == null ? null : depth - 1 })}`;\n }\n});\nObject.setPrototypeOf(Response2, GlobalResponse);\nObject.setPrototypeOf(Response2.prototype, GlobalResponse.prototype);\n\n// src/utils.ts\nasync function readWithoutBlocking(readPromise) {\n return Promise.race([readPromise, Promise.resolve().then(() => Promise.resolve(void 0))]);\n}\nfunction writeFromReadableStreamDefaultReader(reader, writable, currentReadPromise) {\n const cancel = (error) => {\n reader.cancel(error).catch(() => {\n });\n };\n writable.on(\"close\", cancel);\n writable.on(\"error\", cancel);\n (currentReadPromise ?? reader.read()).then(flow, handleStreamError);\n return reader.closed.finally(() => {\n writable.off(\"close\", cancel);\n writable.off(\"error\", cancel);\n });\n function handleStreamError(error) {\n if (error) {\n writable.destroy(error);\n }\n }\n function onDrain() {\n reader.read().then(flow, handleStreamError);\n }\n function flow({ done, value }) {\n try {\n if (done) {\n writable.end();\n } else if (!writable.write(value)) {\n writable.once(\"drain\", onDrain);\n } else {\n return reader.read().then(flow, handleStreamError);\n }\n } catch (e) {\n handleStreamError(e);\n }\n }\n}\nfunction writeFromReadableStream(stream, writable) {\n if (stream.locked) {\n throw new TypeError(\"ReadableStream is locked.\");\n } else if (writable.destroyed) {\n return;\n }\n return writeFromReadableStreamDefaultReader(stream.getReader(), writable);\n}\nvar buildOutgoingHttpHeaders = (headers) => {\n const res = {};\n if (!(headers instanceof Headers)) {\n headers = new Headers(headers ?? void 0);\n }\n const cookies = [];\n for (const [k, v] of headers) {\n if (k === \"set-cookie\") {\n cookies.push(v);\n } else {\n res[k] = v;\n }\n }\n if (cookies.length > 0) {\n res[\"set-cookie\"] = cookies;\n }\n res[\"content-type\"] ??= \"text/plain; charset=UTF-8\";\n return res;\n};\n\n// src/utils/response/constants.ts\nvar X_ALREADY_SENT = \"x-hono-already-sent\";\n\n// src/globals.ts\nimport crypto from \"crypto\";\nif (typeof global.crypto === \"undefined\") {\n global.crypto = crypto;\n}\n\n// src/listener.ts\nvar outgoingEnded = Symbol(\"outgoingEnded\");\nvar incomingDraining = Symbol(\"incomingDraining\");\nvar DRAIN_TIMEOUT_MS = 500;\nvar MAX_DRAIN_BYTES = 64 * 1024 * 1024;\nvar drainIncoming = (incoming) => {\n const incomingWithDrainState = incoming;\n if (incoming.destroyed || incomingWithDrainState[incomingDraining]) {\n return;\n }\n incomingWithDrainState[incomingDraining] = true;\n if (incoming instanceof Http2ServerRequest2) {\n try {\n ;\n incoming.stream?.close?.(h2constants.NGHTTP2_NO_ERROR);\n } catch {\n }\n return;\n }\n let bytesRead = 0;\n const cleanup = () => {\n clearTimeout(timer);\n incoming.off(\"data\", onData);\n incoming.off(\"end\", cleanup);\n incoming.off(\"error\", cleanup);\n };\n const forceClose = () => {\n cleanup();\n const socket = incoming.socket;\n if (socket && !socket.destroyed) {\n socket.destroySoon();\n }\n };\n const timer = setTimeout(forceClose, DRAIN_TIMEOUT_MS);\n timer.unref?.();\n const onData = (chunk) => {\n bytesRead += chunk.length;\n if (bytesRead > MAX_DRAIN_BYTES) {\n forceClose();\n }\n };\n incoming.on(\"data\", onData);\n incoming.on(\"end\", cleanup);\n incoming.on(\"error\", cleanup);\n incoming.resume();\n};\nvar handleRequestError = () => new Response(null, {\n status: 400\n});\nvar handleFetchError = (e) => new Response(null, {\n status: e instanceof Error && (e.name === \"TimeoutError\" || e.constructor.name === \"TimeoutError\") ? 504 : 500\n});\nvar handleResponseError = (e, outgoing) => {\n const err = e instanceof Error ? e : new Error(\"unknown error\", { cause: e });\n if (err.code === \"ERR_STREAM_PREMATURE_CLOSE\") {\n console.info(\"The user aborted a request.\");\n } else {\n console.error(e);\n if (!outgoing.headersSent) {\n outgoing.writeHead(500, { \"Content-Type\": \"text/plain\" });\n }\n outgoing.end(`Error: ${err.message}`);\n outgoing.destroy(err);\n }\n};\nvar flushHeaders = (outgoing) => {\n if (\"flushHeaders\" in outgoing && outgoing.writable) {\n outgoing.flushHeaders();\n }\n};\nvar responseViaCache = async (res, outgoing) => {\n let [status, body, header] = res[cacheKey];\n let hasContentLength = false;\n if (!header) {\n header = { \"content-type\": \"text/plain; charset=UTF-8\" };\n } else if (header instanceof Headers) {\n hasContentLength = header.has(\"content-length\");\n header = buildOutgoingHttpHeaders(header);\n } else if (Array.isArray(header)) {\n const headerObj = new Headers(header);\n hasContentLength = headerObj.has(\"content-length\");\n header = buildOutgoingHttpHeaders(headerObj);\n } else {\n for (const key in header) {\n if (key.length === 14 && key.toLowerCase() === \"content-length\") {\n hasContentLength = true;\n break;\n }\n }\n }\n if (!hasContentLength) {\n if (typeof body === \"string\") {\n header[\"Content-Length\"] = Buffer.byteLength(body);\n } else if (body instanceof Uint8Array) {\n header[\"Content-Length\"] = body.byteLength;\n } else if (body instanceof Blob) {\n header[\"Content-Length\"] = body.size;\n }\n }\n outgoing.writeHead(status, header);\n if (typeof body === \"string\" || body instanceof Uint8Array) {\n outgoing.end(body);\n } else if (body instanceof Blob) {\n outgoing.end(new Uint8Array(await body.arrayBuffer()));\n } else {\n flushHeaders(outgoing);\n await writeFromReadableStream(body, outgoing)?.catch(\n (e) => handleResponseError(e, outgoing)\n );\n }\n ;\n outgoing[outgoingEnded]?.();\n};\nvar isPromise = (res) => typeof res.then === \"function\";\nvar responseViaResponseObject = async (res, outgoing, options = {}) => {\n if (isPromise(res)) {\n if (options.errorHandler) {\n try {\n res = await res;\n } catch (err) {\n const errRes = await options.errorHandler(err);\n if (!errRes) {\n return;\n }\n res = errRes;\n }\n } else {\n res = await res.catch(handleFetchError);\n }\n }\n if (cacheKey in res) {\n return responseViaCache(res, outgoing);\n }\n const resHeaderRecord = buildOutgoingHttpHeaders(res.headers);\n if (res.body) {\n const reader = res.body.getReader();\n const values = [];\n let done = false;\n let currentReadPromise = void 0;\n if (resHeaderRecord[\"transfer-encoding\"] !== \"chunked\") {\n let maxReadCount = 2;\n for (let i = 0; i < maxReadCount; i++) {\n currentReadPromise ||= reader.read();\n const chunk = await readWithoutBlocking(currentReadPromise).catch((e) => {\n console.error(e);\n done = true;\n });\n if (!chunk) {\n if (i === 1) {\n await new Promise((resolve) => setTimeout(resolve));\n maxReadCount = 3;\n continue;\n }\n break;\n }\n currentReadPromise = void 0;\n if (chunk.value) {\n values.push(chunk.value);\n }\n if (chunk.done) {\n done = true;\n break;\n }\n }\n if (done && !(\"content-length\" in resHeaderRecord)) {\n resHeaderRecord[\"content-length\"] = values.reduce((acc, value) => acc + value.length, 0);\n }\n }\n outgoing.writeHead(res.status, resHeaderRecord);\n values.forEach((value) => {\n ;\n outgoing.write(value);\n });\n if (done) {\n outgoing.end();\n } else {\n if (values.length === 0) {\n flushHeaders(outgoing);\n }\n await writeFromReadableStreamDefaultReader(reader, outgoing, currentReadPromise);\n }\n } else if (resHeaderRecord[X_ALREADY_SENT]) {\n } else {\n outgoing.writeHead(res.status, resHeaderRecord);\n outgoing.end();\n }\n ;\n outgoing[outgoingEnded]?.();\n};\nvar getRequestListener = (fetchCallback, options = {}) => {\n const autoCleanupIncoming = options.autoCleanupIncoming ?? true;\n if (options.overrideGlobalObjects !== false && global.Request !== Request) {\n Object.defineProperty(global, \"Request\", {\n value: Request\n });\n Object.defineProperty(global, \"Response\", {\n value: Response2\n });\n }\n return async (incoming, outgoing) => {\n let res, req;\n try {\n req = newRequest(incoming, options.hostname);\n let incomingEnded = !autoCleanupIncoming || incoming.method === \"GET\" || incoming.method === \"HEAD\";\n if (!incomingEnded) {\n ;\n incoming[wrapBodyStream] = true;\n incoming.on(\"end\", () => {\n incomingEnded = true;\n });\n if (incoming instanceof Http2ServerRequest2) {\n ;\n outgoing[outgoingEnded] = () => {\n if (!incomingEnded) {\n setTimeout(() => {\n if (!incomingEnded) {\n setTimeout(() => {\n drainIncoming(incoming);\n });\n }\n });\n }\n };\n }\n outgoing.on(\"finish\", () => {\n if (!incomingEnded) {\n drainIncoming(incoming);\n }\n });\n }\n outgoing.on(\"close\", () => {\n const abortController = req[abortControllerKey];\n if (abortController) {\n if (incoming.errored) {\n req[abortControllerKey].abort(incoming.errored.toString());\n } else if (!outgoing.writableFinished) {\n req[abortControllerKey].abort(\"Client connection prematurely closed.\");\n }\n }\n if (!incomingEnded) {\n setTimeout(() => {\n if (!incomingEnded) {\n setTimeout(() => {\n drainIncoming(incoming);\n });\n }\n });\n }\n });\n res = fetchCallback(req, { incoming, outgoing });\n if (cacheKey in res) {\n return responseViaCache(res, outgoing);\n }\n } catch (e) {\n if (!res) {\n if (options.errorHandler) {\n res = await options.errorHandler(req ? e : toRequestError(e));\n if (!res) {\n return;\n }\n } else if (!req) {\n res = handleRequestError();\n } else {\n res = handleFetchError(e);\n }\n } else {\n return handleResponseError(e, outgoing);\n }\n }\n try {\n return await responseViaResponseObject(res, outgoing, options);\n } catch (e) {\n return handleResponseError(e, outgoing);\n }\n };\n};\n\n// src/server.ts\nvar createAdaptorServer = (options) => {\n const fetchCallback = options.fetch;\n const requestListener = getRequestListener(fetchCallback, {\n hostname: options.hostname,\n overrideGlobalObjects: options.overrideGlobalObjects,\n autoCleanupIncoming: options.autoCleanupIncoming\n });\n const createServer = options.createServer || createServerHTTP;\n const server = createServer(options.serverOptions || {}, requestListener);\n return server;\n};\nvar serve = (options, listeningListener) => {\n const server = createAdaptorServer(options);\n server.listen(options?.port ?? 3e3, options.hostname, () => {\n const serverInfo = server.address();\n listeningListener && listeningListener(serverInfo);\n });\n return server;\n};\nexport {\n RequestError,\n createAdaptorServer,\n getRequestListener,\n serve\n};\n","// Server\nexport const PARTY_PORT = parseInt(process.env.PARTY_PORT || '8000', 10);\n\n// Agent lifecycle\nexport const HEARTBEAT_TIMEOUT = parseFloat(process.env.HEARTBEAT_TIMEOUT || '30');\nexport const CLEANUP_INTERVAL = parseFloat(process.env.CLEANUP_INTERVAL || '10');\n\n// Peer discovery\nexport const DISCOVERY_INTERVAL = parseFloat(process.env.DISCOVERY_INTERVAL || '20');\nexport const REMOTE_STALE_FACTOR = parseInt(process.env.REMOTE_STALE_FACTOR || '3', 10);\nexport const PROBE_TIMEOUT = parseFloat(process.env.PROBE_TIMEOUT || '5');\n","import type { MessageEnvelope, HistoryEntry } from './models.js';\n\nconst HISTORY_CAP = 200; // max history entries per agent\n\n/** Per-agent message queue stored in memory. */\nexport class MessageQueue {\n private _queues = new Map<string, MessageEnvelope[]>();\n private _history = new Map<string, HistoryEntry[]>();\n\n /** Enqueue a message for agentId. Returns the queue length after enqueue. */\n enqueue(agentId: string, envelope: MessageEnvelope): number {\n let q = this._queues.get(agentId);\n if (!q) {\n q = [];\n this._queues.set(agentId, q);\n }\n q.push(envelope);\n return q.length;\n }\n\n /** Pop up to maxCount messages for agentId. */\n dequeue(agentId: string, maxCount = 50): MessageEnvelope[] {\n const q = this._queues.get(agentId);\n if (!q) return [];\n return q.splice(0, Math.min(maxCount, q.length));\n }\n\n /** Return the number of pending messages for agentId. */\n pendingCount(agentId: string): number {\n return this._queues.get(agentId)?.length ?? 0;\n }\n\n /** Clean up queue when agent is removed. */\n removeAgent(agentId: string): void {\n this._queues.delete(agentId);\n }\n\n /** Record a message in an agent's history. */\n logToHistory(agentId: string, direction: 'sent' | 'received', envelope: MessageEnvelope): void {\n let h = this._history.get(agentId);\n if (!h) {\n h = [];\n this._history.set(agentId, h);\n }\n h.push({\n direction,\n sender_id: envelope.sender_id,\n recipient_id: envelope.recipient_id,\n summary: envelope.summary,\n content: envelope.content,\n timestamp: envelope.timestamp ?? Date.now() / 1000,\n });\n if (h.length > HISTORY_CAP) {\n this._history.set(agentId, h.slice(-HISTORY_CAP));\n }\n }\n\n /** Get recent N history entries for an agent. */\n getHistory(agentId: string, limit = 20): HistoryEntry[] {\n const h = this._history.get(agentId);\n if (!h) return [];\n return h.slice(-limit);\n }\n\n /** Clean up history when agent is removed. */\n removeAgentHistory(agentId: string): void {\n this._history.delete(agentId);\n }\n}\n","/**\r\n * Peer discovery and remote agent cache for cross-host communication.\r\n *\r\n * Pull-based discovery loop:\r\n * 1. Read Tailscale peers via readTailscaleStatus()\r\n * 2. Probe each peer's /proxy/health to identify Party Servers\r\n * 3. Fetch agent lists from healthy peers via /proxy/list_agents\r\n * 4. Maintain a remote agent cache with per-peer reachability tracking\r\n */\r\n\r\nimport type { AgentInfo } from './models.js';\r\nimport { DISCOVERY_INTERVAL, PARTY_PORT, PROBE_TIMEOUT, REMOTE_STALE_FACTOR } from './config.js';\r\nimport { readTailscaleStatus } from '../infra/tailscale.js';\r\n\r\n// ---------------------------------------------------------------------------\r\n// Peer state machine constants\r\n// ---------------------------------------------------------------------------\r\n\r\nconst UNKNOWN = 'UNKNOWN';\r\nconst PARTY_SERVER = 'PARTY_SERVER';\r\nconst DEGRADED = 'DEGRADED';\r\nconst SUSPECT = 'SUSPECT';\r\nconst DOWN = 'DOWN';\r\nconst NOT_SERVER = 'NOT_SERVER';\r\nconst MAYBE = 'MAYBE';\r\n\r\nconst MAYBE_MAX_RETRIES = 3;\r\nconst BACKOFF_BASE = 60;\r\nconst BACKOFF_CAP = 900;\r\nconst FAILURE_DEGRADED = 1;\r\nconst FAILURE_SUSPECT = 2;\r\nconst FAILURE_DOWN = 3;\r\n\r\n// ---------------------------------------------------------------------------\r\n// Data types\r\n// ---------------------------------------------------------------------------\r\n\r\ninterface PeerState {\r\n ip: string;\r\n status: string;\r\n consecutiveFailures: number;\r\n lastProbeAt: number;\r\n backoffUntil: number | null;\r\n maybeRetries: number;\r\n}\r\n\r\ninterface RemoteAgentEntry {\r\n agentInfo: AgentInfo;\r\n sourcePeerIp: string;\r\n lastSyncedAt: number;\r\n reachable: boolean;\r\n}\r\n\r\n// ---------------------------------------------------------------------------\r\n// Error classification for native fetch\r\n// ---------------------------------------------------------------------------\r\n\r\n/** Classify a fetch error as refused (false), timeout/uncertain (null). */\r\nfunction classifyFetchError(error: unknown): boolean | null {\r\n if (error instanceof TypeError && /fetch failed/i.test(error.message)) {\r\n const cause = (error as NodeJS.ErrnoException).cause as { code?: string } | undefined;\r\n if (cause?.code === 'ECONNREFUSED') return false;\r\n return null; // timeout or other network error = uncertain\r\n }\r\n if (error instanceof DOMException && error.name === 'AbortError') return null;\r\n return null;\r\n}\r\n\r\n// ---------------------------------------------------------------------------\r\n// PeerDiscovery\r\n// ---------------------------------------------------------------------------\r\n\r\nexport class PeerDiscovery {\r\n private _selfIp: string;\r\n private _peers = new Map<string, PeerState>();\r\n private _remoteAgents = new Map<string, RemoteAgentEntry>();\r\n\r\n constructor(selfIp: string) {\r\n this._selfIp = selfIp;\r\n }\r\n\r\n // ------------------------------------------------------------------\r\n // Public query API (used by router)\r\n // ------------------------------------------------------------------\r\n\r\n /** Return the peer IP that owns agentId, or undefined. */\r\n getPeerForAgent(agentId: string): string | undefined {\r\n return this._remoteAgents.get(agentId)?.sourcePeerIp;\r\n }\r\n\r\n /** Return true if the peer is in a serving state. */\r\n isPeerReachable(peerIp: string): boolean {\r\n const ps = this._peers.get(peerIp);\r\n return ps !== undefined && (ps.status === PARTY_SERVER || ps.status === DEGRADED || ps.status === SUSPECT);\r\n }\r\n\r\n /** Return all remote agents (including unreachable ones). */\r\n getAllRemoteAgents(): AgentInfo[] {\r\n return Array.from(this._remoteAgents.values(), (e) => e.agentInfo);\r\n }\r\n\r\n /** Return only reachable remote agents. */\r\n getReachableRemoteAgents(): AgentInfo[] {\r\n return Array.from(this._remoteAgents.values())\r\n .filter((e) => e.reachable)\r\n .map((e) => e.agentInfo);\r\n }\r\n\r\n /** Return all known peer states for dashboard consumption. */\r\n getPeerStates(): Array<{\r\n ip: string;\r\n status: string;\r\n consecutiveFailures: number;\r\n lastProbeAt: number;\r\n backoffUntil: number | null;\r\n }> {\r\n return Array.from(this._peers.values()).map((ps) => ({\r\n ip: ps.ip,\r\n status: ps.status,\r\n consecutiveFailures: ps.consecutiveFailures,\r\n lastProbeAt: ps.lastProbeAt,\r\n backoffUntil: ps.backoffUntil,\r\n }));\r\n }\r\n\r\n /** Return remote agent entries with peer mapping info. */\r\n getRemoteAgentEntries(): Array<{\r\n agentInfo: AgentInfo;\r\n sourcePeerIp: string;\r\n lastSyncedAt: number;\r\n reachable: boolean;\r\n }> {\r\n return Array.from(this._remoteAgents.values()).map((e) => ({\r\n agentInfo: e.agentInfo,\r\n sourcePeerIp: e.sourcePeerIp,\r\n lastSyncedAt: e.lastSyncedAt,\r\n reachable: e.reachable,\r\n }));\r\n }\r\n\r\n // ------------------------------------------------------------------\r\n // Main discovery loop\r\n // ------------------------------------------------------------------\r\n\r\n async runLoop(): Promise<void> {\r\n while (true) {\r\n try {\r\n await this.discoveryCycle();\r\n } catch (e) {\r\n console.error('[Discovery] Cycle failed:', e);\r\n }\r\n await sleep(DISCOVERY_INTERVAL * 1000);\r\n }\r\n }\r\n\r\n private async discoveryCycle(): Promise<void> {\r\n const peerIps = this.getTailscalePeers();\r\n\r\n // Add new peers as UNKNOWN\r\n for (const ip of peerIps) {\r\n if (!this._peers.has(ip)) {\r\n this._peers.set(ip, { ip, status: UNKNOWN, consecutiveFailures: 0, lastProbeAt: 0, backoffUntil: null, maybeRetries: 0 });\r\n }\r\n }\r\n\r\n const now = performance.now() / 1000;\r\n\r\n // Process each peer\r\n for (const ip of peerIps) {\r\n const ps = this._peers.get(ip)!;\r\n\r\n // Skip NOT_SERVER peers still in backoff\r\n if (ps.status === NOT_SERVER) {\r\n if (ps.backoffUntil !== null && now < ps.backoffUntil) continue;\r\n ps.status = UNKNOWN;\r\n }\r\n\r\n // Skip MAYBE peers that exhausted retries\r\n if (ps.status === MAYBE && ps.maybeRetries >= MAYBE_MAX_RETRIES) {\r\n this.transition(ps, NOT_SERVER);\r\n continue;\r\n }\r\n\r\n await this.probePeer(ps);\r\n }\r\n\r\n this.evictDownAgents();\r\n this.evictStaleAgents();\r\n }\r\n\r\n // ------------------------------------------------------------------\r\n // Tailscale peer enumeration\r\n // ------------------------------------------------------------------\r\n\r\n private getTailscalePeers(): string[] {\r\n let status: Record<string, unknown>;\r\n try {\r\n status = readTailscaleStatus();\r\n } catch {\r\n console.warn('[Discovery] Failed to read Tailscale status');\r\n return [];\r\n }\r\n\r\n const peersRaw = status.Peer as Record<string, Record<string, unknown>> | undefined;\r\n if (!peersRaw || typeof peersRaw !== 'object') return [];\r\n\r\n const ips: string[] = [];\r\n for (const peer of Object.values(peersRaw)) {\r\n if (typeof peer !== 'object' || !peer) continue;\r\n if (!peer.Online) continue;\r\n if (peer.ExitNode) continue;\r\n const tailscaleIps = peer.TailscaleIPs as string[] | undefined;\r\n if (!tailscaleIps) continue;\r\n for (const ip of tailscaleIps) {\r\n if (ip.includes('.') && !ip.includes(':')) {\r\n if (ip !== this._selfIp) ips.push(ip);\r\n break;\r\n }\r\n }\r\n }\r\n return ips;\r\n }\r\n\r\n // ------------------------------------------------------------------\r\n // Peer probing\r\n // ------------------------------------------------------------------\r\n\r\n private async probePeer(ps: PeerState): Promise<void> {\r\n ps.lastProbeAt = performance.now() / 1000;\r\n const healthy = await this.checkHealth(ps.ip);\r\n\r\n if (healthy === null) {\r\n this.handleProbeFailure(ps, true);\r\n } else if (healthy) {\r\n await this.handleProbeSuccess(ps);\r\n } else {\r\n this.handleProbeFailure(ps, false);\r\n }\r\n }\r\n\r\n private async checkHealth(ip: string): Promise<boolean | null> {\r\n const url = `http://${ip}:${PARTY_PORT}/proxy/health`;\r\n try {\r\n const controller = new AbortController();\r\n const timer = setTimeout(() => controller.abort(), PROBE_TIMEOUT * 1000);\r\n const resp = await fetch(url, { signal: controller.signal });\r\n clearTimeout(timer);\r\n const data = (await resp.json()) as Record<string, unknown>;\r\n return resp.status === 200 && 'status' in data;\r\n } catch (e) {\r\n return classifyFetchError(e);\r\n }\r\n }\r\n\r\n // ------------------------------------------------------------------\r\n // State machine transitions\r\n // ------------------------------------------------------------------\r\n\r\n private async handleProbeSuccess(ps: PeerState): Promise<void> {\r\n ps.consecutiveFailures = 0;\r\n ps.maybeRetries = 0;\r\n ps.backoffUntil = null;\r\n\r\n if (ps.status !== PARTY_SERVER) {\r\n this.transition(ps, PARTY_SERVER);\r\n }\r\n\r\n await this.syncAgents(ps.ip);\r\n }\r\n\r\n private handleProbeFailure(ps: PeerState, timeout: boolean): void {\r\n if (ps.status === UNKNOWN || ps.status === MAYBE) {\r\n if (timeout) {\r\n ps.maybeRetries++;\r\n if (ps.maybeRetries >= MAYBE_MAX_RETRIES) {\r\n this.transition(ps, NOT_SERVER);\r\n } else {\r\n this.transition(ps, MAYBE);\r\n }\r\n } else {\r\n this.transition(ps, NOT_SERVER);\r\n }\r\n } else if (ps.status === PARTY_SERVER) {\r\n ps.consecutiveFailures++;\r\n if (ps.consecutiveFailures >= FAILURE_DOWN) {\r\n this.transition(ps, DOWN);\r\n } else if (ps.consecutiveFailures >= FAILURE_SUSPECT) {\r\n this.transition(ps, SUSPECT);\r\n } else {\r\n this.transition(ps, DEGRADED);\r\n }\r\n } else if (ps.status === DEGRADED || ps.status === SUSPECT) {\r\n ps.consecutiveFailures++;\r\n if (ps.consecutiveFailures >= FAILURE_DOWN) {\r\n this.transition(ps, DOWN);\r\n } else if (ps.status === DEGRADED && ps.consecutiveFailures >= FAILURE_SUSPECT) {\r\n this.transition(ps, SUSPECT);\r\n }\r\n }\r\n // DOWN → nothing changes\r\n }\r\n\r\n private transition(ps: PeerState, newStatus: string): void {\r\n const old = ps.status;\r\n ps.status = newStatus;\r\n if (old !== newStatus) {\r\n console.log(`[Discovery] Peer ${ps.ip}: ${old} -> ${newStatus}`);\r\n }\r\n\r\n if (newStatus === NOT_SERVER) {\r\n const retries = ps.maybeRetries > 0 ? ps.maybeRetries : 1;\r\n const delay = Math.min(BACKOFF_BASE * Math.pow(2, retries - 1), BACKOFF_CAP);\r\n ps.backoffUntil = performance.now() / 1000 + delay;\r\n }\r\n\r\n if (newStatus === DOWN) {\r\n for (const entry of this._remoteAgents.values()) {\r\n if (entry.sourcePeerIp === ps.ip) {\r\n entry.reachable = false;\r\n }\r\n }\r\n }\r\n }\r\n\r\n // ------------------------------------------------------------------\r\n // Agent synchronization\r\n // ------------------------------------------------------------------\r\n\r\n private async syncAgents(peerIp: string): Promise<void> {\r\n const url = `http://${peerIp}:${PARTY_PORT}/proxy/list_agents`;\r\n try {\r\n const controller = new AbortController();\r\n const timer = setTimeout(() => controller.abort(), PROBE_TIMEOUT * 1000);\r\n const resp = await fetch(url, { signal: controller.signal });\r\n clearTimeout(timer);\r\n const data = (await resp.json()) as { agents?: AgentInfo[] };\r\n const agentsRaw = data.agents ?? [];\r\n const now = Date.now() / 1000;\r\n\r\n const seenIds = new Set<string>();\r\n for (const a of agentsRaw) {\r\n seenIds.add(a.agent_id);\r\n this._remoteAgents.set(a.agent_id, {\r\n agentInfo: a,\r\n sourcePeerIp: peerIp,\r\n lastSyncedAt: now,\r\n reachable: true,\r\n });\r\n }\r\n\r\n // Remove agents from this peer no longer present\r\n for (const [aid, entry] of this._remoteAgents) {\r\n if (entry.sourcePeerIp === peerIp && !seenIds.has(aid)) {\r\n this._remoteAgents.delete(aid);\r\n }\r\n }\r\n } catch {\r\n console.warn(`[Discovery] Failed to sync agents from ${peerIp}`);\r\n }\r\n }\r\n\r\n // ------------------------------------------------------------------\r\n // Cleanup\r\n // ------------------------------------------------------------------\r\n\r\n private evictDownAgents(): void {\r\n const downPeers = new Set<string>();\r\n for (const [ip, ps] of this._peers) {\r\n if (ps.status === DOWN) downPeers.add(ip);\r\n }\r\n if (!downPeers.size) return;\r\n\r\n for (const [aid, entry] of this._remoteAgents) {\r\n if (downPeers.has(entry.sourcePeerIp)) {\r\n this._remoteAgents.delete(aid);\r\n }\r\n }\r\n }\r\n\r\n private evictStaleAgents(): void {\r\n const threshold = DISCOVERY_INTERVAL * REMOTE_STALE_FACTOR;\r\n const now = Date.now() / 1000;\r\n\r\n for (const [aid, entry] of this._remoteAgents) {\r\n if (now - entry.lastSyncedAt > threshold) {\r\n this._remoteAgents.delete(aid);\r\n }\r\n }\r\n }\r\n}\r\n\r\n// ---------------------------------------------------------------------------\r\n// Utility\r\n// ---------------------------------------------------------------------------\r\n\r\nfunction sleep(ms: number): Promise<void> {\r\n return new Promise((resolve) => setTimeout(resolve, ms));\r\n}\r\n","import type { AgentInfo, RegistrationRequest } from './models.js';\n\n/** In-memory registry of agents connected to this server. */\nexport class AgentRegistry {\n private _agents = new Map<string, AgentInfo>();\n private _selfIp: string;\n\n constructor(selfIp: string) {\n this._selfIp = selfIp;\n }\n\n register(req: RegistrationRequest): AgentInfo {\n const now = Date.now() / 1000;\n const info: AgentInfo = {\n agent_id: req.agent_id,\n display_name: req.display_name,\n host_ip: this._selfIp,\n registered_at: now,\n last_heartbeat: now,\n metadata: req.metadata ?? {},\n };\n this._agents.set(req.agent_id, info);\n return info;\n }\n\n remove(agentId: string): boolean {\n return this._agents.delete(agentId);\n }\n\n heartbeat(agentId: string): AgentInfo {\n const info = this._agents.get(agentId);\n if (!info) throw new Error(`Agent '${agentId}' not registered`);\n info.last_heartbeat = Date.now() / 1000;\n return info;\n }\n\n get(agentId: string): AgentInfo | undefined {\n return this._agents.get(agentId);\n }\n\n listAll(): AgentInfo[] {\n return Array.from(this._agents.values());\n }\n\n /** Remove agents whose last heartbeat is older than timeout seconds. */\n cleanupStale(timeout: number): string[] {\n const now = Date.now() / 1000;\n const stale: string[] = [];\n for (const [aid, info] of this._agents) {\n if (now - info.last_heartbeat > timeout) {\n stale.push(aid);\n }\n }\n for (const aid of stale) {\n this._agents.delete(aid);\n }\n return stale;\n }\n}\n","/**\r\n * Shared state singletons for the Party Server.\r\n *\r\n * All mutable state (registry, messageQueue, discovery) is owned here.\r\n */\r\nimport { getTailscaleIps } from '../infra/tailscale.js';\r\nimport { MessageQueue } from './message-queue.js';\r\nimport { PeerDiscovery } from './peer-discovery.js';\r\nimport { AgentRegistry } from './registry.js';\r\n\r\nfunction resolveSelfIp(): string {\r\n try {\r\n const ips = getTailscaleIps();\r\n for (const ip of ips) {\r\n if (ip.includes('.') && !ip.includes(':')) return ip;\r\n }\r\n return ips.length > 0 ? ips[0] : '127.0.0.1';\r\n } catch {\r\n return '127.0.0.1';\r\n }\r\n}\r\n\r\nlet _selfIp = resolveSelfIp();\r\n\r\nexport function getSelfIp(): string {\r\n return _selfIp;\r\n}\r\n\r\nexport function refreshSelfIp(): string {\r\n _selfIp = resolveSelfIp();\r\n return _selfIp;\r\n}\r\n\r\nexport const STARTED_AT = Date.now();\r\nexport const registry = new AgentRegistry(getSelfIp());\r\nexport const messageQueue = new MessageQueue();\r\nexport const discovery = new PeerDiscovery(getSelfIp());\r\n","/** Agent registration request. */\nexport interface RegistrationRequest {\n agent_id: string;\n display_name?: string;\n metadata?: Record<string, unknown>;\n}\n\n/** Heartbeat request. */\nexport interface HeartbeatRequest {\n agent_id: string;\n}\n\n/** Registered agent info (internal — includes host_ip for routing). */\nexport interface AgentInfo {\n agent_id: string;\n display_name?: string;\n host_ip: string;\n registered_at: number;\n last_heartbeat: number;\n metadata: Record<string, unknown>;\n}\n\n/** Agent info safe to return to agents (no host_ip). */\nexport interface PublicAgentInfo {\n agent_id: string;\n display_name?: string;\n registered_at: number;\n last_heartbeat: number;\n metadata: Record<string, unknown>;\n}\n\n/** Strip host_ip from AgentInfo for agent-facing responses. */\nexport function sanitizeAgentInfo(info: AgentInfo): PublicAgentInfo {\n const { host_ip: _, ...publicInfo } = info;\n return publicInfo;\n}\n\n/** Strip host_ip from an array of AgentInfo. */\nexport function sanitizeAgentList(agents: AgentInfo[]): PublicAgentInfo[] {\n return agents.map(sanitizeAgentInfo);\n}\n\n/** Agent list response. */\nexport interface AgentListResponse {\n agents: AgentInfo[];\n count: number;\n}\n\n/** Remove request. */\nexport interface RemoveRequest {\n agent_id: string;\n}\n\n/** Message envelope. */\nexport interface MessageEnvelope {\n sender_id: string;\n recipient_id?: string; // undefined = broadcast\n group_id?: string;\n summary?: string; // short summary for terminal display\n content: string;\n timestamp?: number;\n}\n\n/** A single entry in an agent's message history. */\nexport interface HistoryEntry {\n direction: 'sent' | 'received';\n sender_id: string;\n recipient_id?: string;\n summary?: string;\n content: string;\n timestamp: number;\n}\n\n/** Send result. */\nexport interface SendResult {\n status: 'delivered_locally' | 'forwarded' | 'agent_not_found' | 'error' | 'peer_unreachable';\n target?: string;\n error?: string;\n}\n","import { Hono } from 'hono';\nimport type { MessageEnvelope, RegistrationRequest, HeartbeatRequest, RemoveRequest, SendResult } from '../models.js';\nimport { sanitizeAgentInfo, sanitizeAgentList } from '../models.js';\nimport { PARTY_PORT } from '../config.js';\n\nconst agentRoutes = new Hono();\n\nasync function forwardToPeer(peerIp: string, envelope: MessageEnvelope): Promise<SendResult> {\n const url = `http://${peerIp}:${PARTY_PORT}/proxy/receive`;\n const payload: Record<string, string> = { sender_id: envelope.sender_id, content: envelope.content };\n if (envelope.recipient_id) payload.recipient_id = envelope.recipient_id;\n if (envelope.group_id) payload.group_id = envelope.group_id;\n try {\n await fetch(url, {\n method: 'POST',\n headers: { 'Content-Type': 'application/json' },\n body: JSON.stringify(payload),\n });\n return { status: 'forwarded', target: peerIp };\n } catch (e) {\n return { status: 'error', target: peerIp, error: (e as Error).message };\n }\n}\n\nagentRoutes.post('/register', async (c) => {\n const { registry } = await import('../state.js');\n const req = await c.req.json<RegistrationRequest>();\n const info = registry.register(req);\n return c.json(sanitizeAgentInfo(info));\n});\n\nagentRoutes.post('/remove', async (c) => {\n const { registry } = await import('../state.js');\n const req = await c.req.json<RemoveRequest>();\n const removed = registry.remove(req.agent_id);\n return c.json({ status: removed ? 'removed' : 'not_found' });\n});\n\nagentRoutes.post('/heartbeat', async (c) => {\n const { registry } = await import('../state.js');\n const req = await c.req.json<HeartbeatRequest>();\n const info = registry.heartbeat(req.agent_id);\n return c.json({ status: 'ok', last_heartbeat: info.last_heartbeat });\n});\n\nagentRoutes.get('/list', async (c) => {\n const { registry, discovery } = await import('../state.js');\n const localAgents = registry.listAll();\n const remoteAgents = discovery.getReachableRemoteAgents();\n const allAgents = localAgents.concat(remoteAgents);\n return c.json({ agents: sanitizeAgentList(allAgents), count: allAgents.length });\n});\n\nagentRoutes.post('/send', async (c) => {\n const { registry, messageQueue, discovery } = await import('../state.js');\n const envelope = await c.req.json<MessageEnvelope>();\n const recipient = envelope.recipient_id;\n\n if (!recipient) {\n return c.json({ status: 'error' as const, error: 'recipient_id is required' });\n }\n\n // Check if recipient is local\n if (registry.get(recipient)) {\n const stamped = { ...envelope, timestamp: envelope.timestamp ?? Date.now() / 1000 };\n const count = messageQueue.enqueue(recipient, stamped);\n messageQueue.logToHistory(envelope.sender_id, 'sent', stamped);\n messageQueue.logToHistory(recipient, 'received', stamped);\n return c.json({ status: 'delivered_locally' as const, target: recipient });\n }\n\n // Check remote agent cache\n const peerIp = discovery.getPeerForAgent(recipient);\n if (peerIp) {\n if (discovery.isPeerReachable(peerIp)) {\n const result = await forwardToPeer(peerIp, envelope);\n if (result.status === 'forwarded') {\n messageQueue.logToHistory(envelope.sender_id, 'sent', { ...envelope, timestamp: envelope.timestamp ?? Date.now() / 1000 });\n }\n return c.json(result);\n } else {\n return c.json({ status: 'peer_unreachable' as const, target: peerIp });\n }\n }\n\n return c.json({ status: 'agent_not_found' as const, target: recipient });\n});\n\nagentRoutes.get('/messages/:agent_id', async (c) => {\n const { messageQueue } = await import('../state.js');\n const agentId = c.req.param('agent_id');\n const maxCount = parseInt(c.req.query('max_count') || '50', 10);\n const msgs = messageQueue.dequeue(agentId, maxCount);\n return c.json({ messages: msgs, count: msgs.length });\n});\n\n/** Get message history for an agent. */\nagentRoutes.get('/history/:agent_id', async (c) => {\n const { messageQueue } = await import('../state.js');\n const agentId = c.req.param('agent_id');\n const limit = parseInt(c.req.query('limit') || '20', 10);\n const history = messageQueue.getHistory(agentId, limit);\n return c.json({ history, count: history.length });\n});\n\nexport { agentRoutes };\n","import { Hono } from 'hono';\r\nimport type { MessageEnvelope, SendResult } from '../models.js';\r\nimport { PARTY_PORT } from '../config.js';\r\nimport { getTailnetHostname } from '../../infra/tailscale.js';\r\nimport { registry, messageQueue, getSelfIp } from '../state.js';\r\n\r\nconst proxyRoutes = new Hono();\r\n\r\nproxyRoutes.get('/health', async (c) => {\r\n let hostname = '127.0.0.1';\r\n try {\r\n hostname = getTailnetHostname();\r\n } catch { /* ignore */ }\r\n\r\n return c.json({\r\n status: 'ok',\r\n tailscale_ip: getSelfIp(),\r\n hostname,\r\n agent_count: registry.listAll().length,\r\n });\r\n});\r\n\r\nproxyRoutes.get('/list_agents', async (c) => {\r\n const agents = registry.listAll();\r\n return c.json({ agents, count: agents.length });\r\n});\r\n\r\nproxyRoutes.post('/receive', async (c) => {\r\n const envelope = await c.req.json<MessageEnvelope>();\r\n const stamped = { ...envelope, timestamp: envelope.timestamp ?? Date.now() / 1000 };\r\n\r\n if (envelope.recipient_id) {\r\n messageQueue.enqueue(envelope.recipient_id, stamped);\r\n messageQueue.logToHistory(envelope.recipient_id, 'received', stamped);\r\n console.log(`[收到消息] ${envelope.sender_id} -> ${envelope.recipient_id}: ${envelope.content}`);\r\n } else {\r\n // Broadcast\r\n for (const agent of registry.listAll()) {\r\n messageQueue.enqueue(agent.agent_id, stamped);\r\n messageQueue.logToHistory(agent.agent_id, 'received', stamped);\r\n }\r\n console.log(`[广播消息] ${envelope.sender_id} -> all: ${envelope.content}`);\r\n }\r\n return c.json({ status: 'received' });\r\n});\r\n\r\nproxyRoutes.post('/send/:target_ip', async (c) => {\r\n const targetIp = c.req.param('target_ip');\r\n const envelope = await c.req.json<MessageEnvelope>();\r\n\r\n const senderId = envelope.sender_id || getSelfIp();\r\n const url = `http://${targetIp}:${PARTY_PORT}/proxy/receive`;\r\n const payload: Record<string, string> = { sender_id: senderId, content: envelope.content };\r\n if (envelope.recipient_id) payload.recipient_id = envelope.recipient_id;\r\n if (envelope.group_id) payload.group_id = envelope.group_id;\r\n\r\n try {\r\n await fetch(url, {\r\n method: 'POST',\r\n headers: { 'Content-Type': 'application/json' },\r\n body: JSON.stringify(payload),\r\n });\r\n console.log(`[发送成功] 目标 ${targetIp}`);\r\n return c.json({ status: 'forwarded' as const, target: targetIp });\r\n } catch (e) {\r\n console.log(`[发送失败] 目标 ${targetIp}, 错误: ${(e as Error).message}`);\r\n return c.json({ status: 'error' as const, target: targetIp, error: (e as Error).message });\r\n }\r\n});\r\n\r\nexport { proxyRoutes };\r\n","export const DASHBOARD_HTML = `<!DOCTYPE html>\r\n<html lang=\"zh-CN\">\r\n<head>\r\n<meta charset=\"UTF-8\">\r\n<meta name=\"viewport\" content=\"width=device-width, initial-scale=1.0\">\r\n<title>OPEN PARTY // Dashboard</title>\r\n<style>\r\n*,*::before,*::after{box-sizing:border-box;margin:0;padding:0}\r\n:root{\r\n --bg:#0a0a0f;--card:#12121a;--border:#1e1e2e;--border-bright:#2a2a3e;\r\n --text:#e0e0e8;--muted:#6a6a8a;\r\n --cyan:#00fff0;--magenta:#ff00ff;--green:#00ff88;--red:#ff3366;--yellow:#ffaa00;--orange:#ff8800;\r\n --font-mono:'JetBrains Mono','Fira Code','Cascadia Code','Consolas','Courier New',monospace;\r\n --font-sans:'Inter','Segoe UI',system-ui,-apple-system,sans-serif;\r\n}\r\nhtml{font-size:14px}\r\nbody{\r\n background:var(--bg);color:var(--text);font-family:var(--font-sans);\r\n min-height:100vh;overflow-x:hidden;position:relative;\r\n}\r\n/* Grid background */\r\nbody::before{\r\n content:'';position:fixed;inset:0;z-index:0;pointer-events:none;\r\n background-image:\r\n linear-gradient(rgba(0,255,240,0.03) 1px,transparent 1px),\r\n linear-gradient(90deg,rgba(0,255,240,0.03) 1px,transparent 1px);\r\n background-size:40px 40px;\r\n}\r\n/* Scanline overlay */\r\nbody::after{\r\n content:'';position:fixed;inset:0;z-index:9999;pointer-events:none;\r\n background:repeating-linear-gradient(\r\n 0deg,transparent,transparent 2px,rgba(0,0,0,0.08) 2px,rgba(0,0,0,0.08) 4px\r\n );\r\n}\r\n#app{position:relative;z-index:1;max-width:1400px;margin:0 auto;padding:0 20px 40px}\r\n\r\n/* ===== Header ===== */\r\n.header{\r\n display:flex;align-items:center;justify-content:space-between;\r\n padding:16px 0;border-bottom:1px solid var(--border);\r\n margin-bottom:20px;flex-wrap:wrap;gap:12px;\r\n}\r\n.header-title{\r\n font-family:var(--font-mono);font-size:1.4rem;font-weight:700;\r\n color:var(--cyan);letter-spacing:3px;\r\n text-shadow:0 0 10px rgba(0,255,240,0.5),0 0 30px rgba(0,255,240,0.2);\r\n}\r\n.header-title span{color:var(--muted);font-weight:400;font-size:0.9rem;margin-left:8px;letter-spacing:1px}\r\n.header-center{display:flex;align-items:center;gap:16px;flex-wrap:wrap}\r\n.header-status{display:flex;align-items:center;gap:6px;font-family:var(--font-mono);font-size:0.85rem}\r\n.status-dot{\r\n width:8px;height:8px;border-radius:50%;background:var(--green);\r\n box-shadow:0 0 6px var(--green);animation:pulse 2s ease-in-out infinite;\r\n}\r\n.status-dot.offline{background:var(--red);box-shadow:0 0 6px var(--red)}\r\n.status-dot.not-installed{background:var(--red);box-shadow:0 0 6px var(--red)}\r\n.status-dot.not-connected{background:var(--yellow);box-shadow:0 0 6px var(--yellow)}\r\n@keyframes pulse{0%,100%{opacity:1;transform:scale(1)}50%{opacity:0.5;transform:scale(0.8)}}\r\n.header-meta{color:var(--muted);font-family:var(--font-mono);font-size:0.8rem}\r\n.header-right{font-family:var(--font-mono);font-size:0.85rem;color:var(--muted)}\r\n.header-right .value{color:var(--cyan)}\r\n\r\n/* ===== Stats Cards ===== */\r\n.stats-row{display:grid;grid-template-columns:repeat(4,1fr);gap:16px;margin-bottom:24px}\r\n@media(max-width:768px){.stats-row{grid-template-columns:repeat(2,1fr)}}\r\n.stat-card{\r\n background:var(--card);border:1px solid var(--border);border-radius:8px;\r\n padding:16px 20px;position:relative;overflow:hidden;\r\n transition:border-color 0.3s;\r\n}\r\n.stat-card:hover{border-color:var(--border-bright)}\r\n.stat-card::before{\r\n content:'';position:absolute;top:0;left:0;right:0;height:2px;\r\n}\r\n.stat-card.cyan::before{background:var(--cyan);box-shadow:0 0 10px var(--cyan)}\r\n.stat-card.green::before{background:var(--green);box-shadow:0 0 10px var(--green)}\r\n.stat-card.magenta::before{background:var(--magenta);box-shadow:0 0 10px var(--magenta)}\r\n.stat-card.yellow::before{background:var(--yellow);box-shadow:0 0 10px var(--yellow)}\r\n.stat-value{\r\n font-family:var(--font-mono);font-size:2rem;font-weight:700;\r\n transition:color 0.3s;\r\n}\r\n.stat-card.cyan .stat-value{color:var(--cyan)}\r\n.stat-card.green .stat-value{color:var(--green)}\r\n.stat-card.magenta .stat-value{color:var(--magenta)}\r\n.stat-card.yellow .stat-value{color:var(--yellow)}\r\n.stat-label{color:var(--muted);font-size:0.8rem;margin-top:4px;text-transform:uppercase;letter-spacing:1px}\r\n.stat-value.flash{animation:flash 0.3s ease}\r\n@keyframes flash{0%{opacity:1}50%{opacity:0.3}100%{opacity:1}}\r\n\r\n/* ===== Main Grid ===== */\r\n.main-grid{display:grid;grid-template-columns:2fr 1fr;gap:20px;margin-bottom:24px}\r\n@media(max-width:1024px){.main-grid{grid-template-columns:1fr}}\r\n.section{\r\n background:var(--card);border:1px solid var(--border);border-radius:8px;\r\n overflow:hidden;\r\n}\r\n.section-header{\r\n padding:12px 16px;border-bottom:1px solid var(--border);\r\n font-family:var(--font-mono);font-size:0.8rem;font-weight:600;\r\n color:var(--muted);letter-spacing:2px;text-transform:uppercase;\r\n display:flex;align-items:center;gap:8px;\r\n}\r\n.section-header .dot{width:6px;height:6px;border-radius:50%;background:var(--cyan)}\r\n.section-body{padding:16px}\r\n\r\n/* ===== Topology ===== */\r\n.topology-container{display:flex;justify-content:center;align-items:center;min-height:300px}\r\n.topology-container svg{width:100%;max-width:500px;height:300px}\r\n.topo-node{cursor:pointer;transition:filter 0.3s}\r\n.topo-node:hover{filter:brightness(1.3)}\r\n.topo-label{font-family:var(--font-mono);font-size:10px;fill:var(--muted)}\r\n.topo-badge{\r\n font-family:var(--font-mono);font-size:9px;fill:var(--bg);\r\n font-weight:700;\r\n}\r\n\r\n/* ===== Peer Table ===== */\r\n.peer-table{width:100%;border-collapse:collapse;font-size:0.85rem}\r\n.peer-table th{\r\n text-align:left;padding:8px 10px;color:var(--muted);font-weight:600;\r\n font-family:var(--font-mono);font-size:0.75rem;text-transform:uppercase;\r\n letter-spacing:1px;border-bottom:1px solid var(--border);\r\n}\r\n.peer-table td{padding:8px 10px;border-bottom:1px solid var(--border);font-family:var(--font-mono);font-size:0.8rem}\r\n.peer-table tr:last-child td{border-bottom:none}\r\n.peer-table tr:hover td{background:rgba(255,255,255,0.02)}\r\n.badge{\r\n display:inline-block;padding:2px 8px;border-radius:4px;font-size:0.7rem;\r\n font-weight:600;text-transform:uppercase;letter-spacing:0.5px;\r\n}\r\n.badge-party{background:rgba(0,255,136,0.15);color:var(--green)}\r\n.badge-degraded{background:rgba(255,170,0,0.15);color:var(--yellow)}\r\n.badge-suspect{background:rgba(255,136,0,0.15);color:var(--orange)}\r\n.badge-down{background:rgba(255,51,102,0.15);color:var(--red)}\r\n.badge-unknown{background:rgba(106,106,138,0.15);color:var(--muted)}\r\n.badge-not_server{background:rgba(106,106,138,0.1);color:var(--muted)}\r\n.badge-maybe{background:rgba(0,255,240,0.1);color:var(--cyan)}\r\n.empty-state{text-align:center;color:var(--muted);padding:40px 20px;font-family:var(--font-mono);font-size:0.85rem}\r\n\r\n/* ===== Agent & Message Grid ===== */\r\n.bottom-grid{display:grid;grid-template-columns:1fr 1fr;gap:20px;margin-bottom:24px}\r\n@media(max-width:1024px){.bottom-grid{grid-template-columns:1fr}}\r\n\r\n/* Agent cards */\r\n.agent-list{display:flex;flex-direction:column;gap:8px;max-height:400px;overflow-y:auto}\r\n.agent-card{\r\n display:flex;align-items:center;gap:12px;padding:10px 14px;\r\n border:1px solid var(--border);border-radius:6px;transition:border-color 0.3s;\r\n}\r\n.agent-card:hover{border-color:var(--border-bright)}\r\n.agent-card.local{border-left:3px solid var(--cyan)}\r\n.agent-card.remote{border-left:3px solid var(--green)}\r\n.agent-icon{\r\n width:36px;height:36px;border-radius:6px;display:flex;align-items:center;\r\n justify-content:center;font-family:var(--font-mono);font-size:0.9rem;font-weight:700;\r\n flex-shrink:0;\r\n}\r\n.agent-card.local .agent-icon{background:rgba(0,255,240,0.1);color:var(--cyan)}\r\n.agent-card.remote .agent-icon{background:rgba(0,255,136,0.1);color:var(--green)}\r\n.agent-info{flex:1;min-width:0}\r\n.agent-name{font-weight:600;font-size:0.9rem;white-space:nowrap;overflow:hidden;text-overflow:ellipsis}\r\n.agent-id{font-family:var(--font-mono);font-size:0.7rem;color:var(--muted);margin-top:2px}\r\n.agent-meta{display:flex;gap:6px;flex-shrink:0;align-items:center}\r\n.agent-tag{\r\n font-family:var(--font-mono);font-size:0.65rem;padding:2px 6px;\r\n border-radius:3px;background:rgba(255,255,255,0.05);color:var(--muted);\r\n}\r\n.agent-tag.unreachable{background:rgba(255,51,102,0.1);color:var(--red)}\r\n\r\n/* Message feed */\r\n.msg-feed{display:flex;flex-direction:column;gap:6px;max-height:400px;overflow-y:auto}\r\n.msg-item{\r\n padding:10px 12px;border:1px solid var(--border);border-radius:6px;\r\n border-left:3px solid var(--cyan);font-size:0.8rem;\r\n}\r\n.msg-item.received{border-left-color:var(--magenta)}\r\n.msg-top{display:flex;justify-content:space-between;align-items:center;margin-bottom:4px}\r\n.msg-flow{font-family:var(--font-mono);font-size:0.75rem;color:var(--cyan)}\r\n.msg-flow .arrow{color:var(--muted);margin:0 4px}\r\n.msg-time{font-family:var(--font-mono);font-size:0.65rem;color:var(--muted)}\r\n.msg-content{color:var(--text);font-size:0.8rem;line-height:1.4;overflow:hidden;text-overflow:ellipsis;white-space:nowrap}\r\n\r\n/* Scrollbar */\r\n::-webkit-scrollbar{width:4px}\r\n::-webkit-scrollbar-track{background:var(--bg)}\r\n::-webkit-scrollbar-thumb{background:var(--border);border-radius:2px}\r\n::-webkit-scrollbar-thumb:hover{background:var(--border-bright)}\r\n\r\n/* Footer */\r\n.footer{\r\n text-align:center;padding:20px 0;border-top:1px solid var(--border);\r\n color:var(--muted);font-family:var(--font-mono);font-size:0.75rem;\r\n letter-spacing:1px;\r\n}\r\n\r\n/* Join Network Button */\r\n.btn-join{\r\n font-family:var(--font-mono);font-size:0.75rem;letter-spacing:1px;\r\n padding:6px 14px;border:1px solid var(--cyan);border-radius:4px;\r\n background:rgba(0,255,240,0.08);color:var(--cyan);cursor:pointer;\r\n transition:all 0.2s;text-transform:uppercase;margin-left:12px;\r\n}\r\n.btn-join:hover{background:rgba(0,255,240,0.18);box-shadow:0 0 10px rgba(0,255,240,0.2)}\r\n.btn-join:active{transform:scale(0.97)}\r\n.btn-logout{border-color:var(--red);color:var(--red);background:rgba(255,51,102,0.08)}\r\n.btn-logout:hover{background:rgba(255,51,102,0.18);box-shadow:0 0 10px rgba(255,51,102,0.2)}\r\n.btn-install{border-color:var(--yellow);color:var(--yellow);background:rgba(255,170,0,0.08)}\r\n.btn-install:hover{background:rgba(255,170,0,0.18);box-shadow:0 0 10px rgba(255,170,0,0.2)}\r\n\r\n/* Tab bar inside modal */\r\n.tab-bar{display:flex;gap:0;margin-bottom:18px;border-bottom:1px solid var(--border)}\r\n.tab-bar .tab{\r\n font-family:var(--font-mono);font-size:0.8rem;padding:8px 16px;cursor:pointer;\r\n color:var(--muted);border-bottom:2px solid transparent;transition:all 0.2s;\r\n}\r\n.tab-bar .tab:hover{color:var(--text)}\r\n.tab-bar .tab.active{color:var(--cyan);border-bottom-color:var(--cyan)}\r\n.tab-content{display:none}\r\n.tab-content.active{display:block}\r\n\r\n/* Modal */\r\n.modal-overlay{\r\n position:fixed;inset:0;background:rgba(0,0,0,0.7);z-index:10000;\r\n display:flex;align-items:center;justify-content:center;\r\n opacity:0;pointer-events:none;transition:opacity 0.25s;\r\n}\r\n.modal-overlay.open{opacity:1;pointer-events:all}\r\n.modal{\r\n background:var(--card);border:1px solid var(--border-bright);border-radius:10px;\r\n padding:28px 32px;width:90%;max-width:440px;\r\n box-shadow:0 0 40px rgba(0,255,240,0.08),0 8px 32px rgba(0,0,0,0.5);\r\n}\r\n.modal-title{\r\n font-family:var(--font-mono);font-size:1rem;font-weight:700;color:var(--cyan);\r\n letter-spacing:2px;margin-bottom:6px;\r\n text-shadow:0 0 8px rgba(0,255,240,0.3);\r\n}\r\n.modal-desc{color:var(--muted);font-size:0.8rem;margin-bottom:20px;line-height:1.5}\r\n.modal-input{\r\n width:100%;padding:10px 14px;background:var(--bg);border:1px solid var(--border);\r\n border-radius:6px;color:var(--text);font-family:var(--font-mono);font-size:0.85rem;\r\n outline:none;transition:border-color 0.2s;\r\n}\r\n.modal-input:focus{border-color:var(--cyan);box-shadow:0 0 8px rgba(0,255,240,0.15)}\r\n.modal-input::placeholder{color:var(--muted)}\r\n.modal-actions{display:flex;gap:10px;margin-top:18px;justify-content:flex-end}\r\n.modal-btn{\r\n font-family:var(--font-mono);font-size:0.8rem;padding:8px 18px;\r\n border-radius:4px;cursor:pointer;transition:all 0.2s;letter-spacing:0.5px;\r\n}\r\n.modal-btn-cancel{\r\n border:1px solid var(--border);background:transparent;color:var(--muted);\r\n}\r\n.modal-btn-cancel:hover{border-color:var(--muted);color:var(--text)}\r\n.modal-btn-submit{\r\n border:1px solid var(--cyan);background:rgba(0,255,240,0.12);color:var(--cyan);\r\n}\r\n.modal-btn-submit:hover{background:rgba(0,255,240,0.22);box-shadow:0 0 10px rgba(0,255,240,0.2)}\r\n.modal-btn-submit:disabled{opacity:0.4;cursor:not-allowed}\r\n.modal-status{\r\n margin-top:12px;padding:8px 12px;border-radius:4px;font-size:0.78rem;\r\n font-family:var(--font-mono);display:none;\r\n}\r\n.modal-status.success{display:block;background:rgba(0,255,136,0.1);border:1px solid rgba(0,255,136,0.3);color:var(--green)}\r\n.modal-status.error{display:block;background:rgba(255,51,102,0.1);border:1px solid rgba(255,51,102,0.3);color:var(--red)}\r\n.spinner{\r\n display:inline-block;width:12px;height:12px;border:2px solid transparent;\r\n border-top-color:var(--cyan);border-radius:50%;animation:spin 0.6s linear infinite;\r\n vertical-align:middle;margin-right:6px;\r\n}\r\n@keyframes spin{to{transform:rotate(360deg)}}\r\n\r\n/* ===== Tailscale Status Panel ===== */\r\n.ts-panel{\r\n margin-top:12px;padding:16px 20px;background:var(--card);border:1px solid var(--border);\r\n border-radius:8px;font-family:var(--font-mono);font-size:0.8rem;\r\n}\r\n.ts-panel-title{\r\n font-weight:700;letter-spacing:2px;text-transform:uppercase;margin-bottom:10px;\r\n display:flex;align-items:center;gap:8px;\r\n}\r\n.ts-panel-title.connected{color:var(--green)}\r\n.ts-panel-title.not-installed{color:var(--red)}\r\n.ts-panel-title.not-connected{color:var(--yellow)}\r\n.ts-info-row{display:flex;gap:8px;margin:4px 0;color:var(--muted)}\r\n.ts-info-row .label{min-width:100px;color:var(--muted)}\r\n.ts-info-row .value{color:var(--text)}\r\n.ts-install-guide{\r\n margin-top:12px;padding:12px 16px;background:rgba(0,0,0,0.3);border:1px solid var(--border);\r\n border-radius:6px;\r\n}\r\n.ts-install-guide .cmd{\r\n display:flex;align-items:center;justify-content:space-between;\r\n padding:6px 10px;background:rgba(255,255,255,0.04);border-radius:4px;\r\n margin:6px 0;font-size:0.8rem;cursor:pointer;transition:background 0.2s;\r\n}\r\n.ts-install-guide .cmd:hover{background:rgba(255,255,255,0.08)}\r\n.ts-install-guide .cmd code{color:var(--cyan)}\r\n.ts-install-guide .copy-hint{color:var(--muted);font-size:0.7rem}\r\n.ts-install-guide a{color:var(--cyan);text-decoration:none}\r\n.ts-install-guide a:hover{text-decoration:underline}\r\n.ts-setup-hint{\r\n margin-top:10px;padding:8px 12px;background:rgba(0,255,240,0.05);border:1px solid rgba(0,255,240,0.15);\r\n border-radius:4px;color:var(--cyan);font-size:0.78rem;\r\n}\r\n.btn-redetect{\r\n font-family:var(--font-mono);font-size:0.7rem;padding:4px 12px;\r\n border:1px solid var(--border-bright);border-radius:4px;background:transparent;\r\n color:var(--muted);cursor:pointer;transition:all 0.2s;margin-top:8px;\r\n}\r\n.btn-redetect:hover{border-color:var(--cyan);color:var(--cyan)}\r\n</style>\r\n</head>\r\n<body>\r\n<div id=\"app\">\r\n <header class=\"header\">\r\n <div class=\"header-title\">OPEN PARTY<span>// Dashboard</span></div>\r\n <div class=\"header-center\">\r\n <div class=\"header-status\">\r\n <div class=\"status-dot\" id=\"statusDot\"></div>\r\n <span id=\"statusText\">CONNECTING</span>\r\n </div>\r\n <div class=\"header-meta\" id=\"serverInfo\">--</div>\r\n </div>\r\n <div class=\"header-right\">\r\n UPTIME <span class=\"value\" id=\"uptime\">--:--:--</span>\r\n <button class=\"btn-join\" id=\"btnJoinNetwork\" title=\"Join Tailscale Network\">Join Network</button>\r\n </div>\r\n </header>\r\n\r\n <!-- Tailscale status panel (shown when not connected) -->\r\n <div id=\"tsPanel\" class=\"ts-panel\" style=\"display:none\"></div>\r\n\r\n <div class=\"stats-row\" id=\"statsRow\">\r\n <div class=\"stat-card cyan\">\r\n <div class=\"stat-value\" id=\"localAgentCount\">-</div>\r\n <div class=\"stat-label\">Local Agents</div>\r\n </div>\r\n <div class=\"stat-card green\">\r\n <div class=\"stat-value\" id=\"remoteAgentCount\">-</div>\r\n <div class=\"stat-label\">Remote Agents</div>\r\n </div>\r\n <div class=\"stat-card magenta\">\r\n <div class=\"stat-value\" id=\"peerCount\">-</div>\r\n <div class=\"stat-label\">Known Peers</div>\r\n </div>\r\n <div class=\"stat-card yellow\">\r\n <div class=\"stat-value\" id=\"partyServerCount\">-</div>\r\n <div class=\"stat-label\">Party Servers</div>\r\n </div>\r\n </div>\r\n\r\n <div class=\"main-grid\">\r\n <div class=\"section\">\r\n <div class=\"section-header\"><div class=\"dot\"></div>NETWORK TOPOLOGY</div>\r\n <div class=\"section-body\">\r\n <div class=\"topology-container\" id=\"topologyContainer\">\r\n <svg id=\"topologySvg\" viewBox=\"0 0 500 300\"></svg>\r\n </div>\r\n </div>\r\n </div>\r\n <div class=\"section\">\r\n <div class=\"section-header\"><div class=\"dot\" style=\"background:var(--green)\"></div>PEER HEALTH</div>\r\n <div class=\"section-body\" style=\"padding:0\">\r\n <div id=\"peerTableContainer\">\r\n <table class=\"peer-table\">\r\n <thead><tr><th>IP</th><th>Status</th><th>Fails</th><th>Last Probe</th></tr></thead>\r\n <tbody id=\"peerTableBody\"></tbody>\r\n </table>\r\n </div>\r\n </div>\r\n </div>\r\n </div>\r\n\r\n <div class=\"bottom-grid\">\r\n <div class=\"section\">\r\n <div class=\"section-header\"><div class=\"dot\" style=\"background:var(--cyan)\"></div>AGENT DIRECTORY</div>\r\n <div class=\"section-body\">\r\n <div class=\"agent-list\" id=\"agentList\"></div>\r\n </div>\r\n </div>\r\n <div class=\"section\">\r\n <div class=\"section-header\"><div class=\"dot\" style=\"background:var(--magenta)\"></div>MESSAGE FLOW</div>\r\n <div class=\"section-body\">\r\n <div class=\"msg-feed\" id=\"msgFeed\"></div>\r\n </div>\r\n </div>\r\n </div>\r\n\r\n <div class=\"footer\">OPEN PARTY v0.1 // DECENTRALIZED AGENT NETWORK</div>\r\n\r\n <!-- Join Network / Login Modal (two tabs: Interactive + Auth Key) -->\r\n <div class=\"modal-overlay\" id=\"joinModal\">\r\n <div class=\"modal\">\r\n <div class=\"modal-title\">CONNECT TO TAILNET</div>\r\n <div class=\"tab-bar\" id=\"joinTabs\">\r\n <div class=\"tab active\" data-tab=\"interactive\">Interactive</div>\r\n <div class=\"tab\" data-tab=\"authkey\">Auth Key</div>\r\n </div>\r\n\r\n <!-- Interactive tab -->\r\n <div class=\"tab-content active\" id=\"tabInteractive\">\r\n <div class=\"modal-desc\">Click the button below to open a browser authentication page.<br>Your Tailscale connection will be established once you authenticate.</div>\r\n <div class=\"modal-status\" id=\"interactiveStatus\"></div>\r\n <div class=\"modal-actions\">\r\n <button class=\"modal-btn modal-btn-cancel\" id=\"btnCancelJoin\">Cancel</button>\r\n <button class=\"modal-btn modal-btn-submit\" id=\"btnInteractiveLogin\">Open Browser Login</button>\r\n </div>\r\n </div>\r\n\r\n <!-- Auth Key tab -->\r\n <div class=\"tab-content\" id=\"tabAuthkey\">\r\n <div class=\"modal-desc\">Enter your Tailscale auth key to join the network.<br>You can generate one from the Tailscale admin console.</div>\r\n <input type=\"password\" class=\"modal-input\" id=\"authKeyInput\" placeholder=\"tskey-auth-xxxxx...\" autocomplete=\"off\" spellcheck=\"false\" />\r\n <div class=\"modal-status\" id=\"joinStatus\"></div>\r\n <div class=\"modal-actions\">\r\n <button class=\"modal-btn modal-btn-cancel\" id=\"btnCancelAuthkey\">Cancel</button>\r\n <button class=\"modal-btn modal-btn-submit\" id=\"btnSubmitJoin\">Connect</button>\r\n </div>\r\n </div>\r\n </div>\r\n </div>\r\n\r\n <!-- Logout Confirmation Modal -->\r\n <div class=\"modal-overlay\" id=\"logoutModal\">\r\n <div class=\"modal\">\r\n <div class=\"modal-title\" style=\"color:var(--red)\">LOG OUT OF TAILNET</div>\r\n <div class=\"modal-desc\">This will disconnect from Tailscale and remove your credentials.<br>You will need to re-authenticate to reconnect.</div>\r\n <div class=\"modal-status\" id=\"logoutStatus\"></div>\r\n <div class=\"modal-actions\">\r\n <button class=\"modal-btn modal-btn-cancel\" id=\"btnCancelLogout\">Cancel</button>\r\n <button class=\"modal-btn modal-btn-submit\" style=\"border-color:var(--red);color:var(--red);background:rgba(255,51,102,0.12)\" id=\"btnConfirmLogout\">Log Out</button>\r\n </div>\r\n </div>\r\n </div>\r\n</div>\r\n\r\n<script>\r\n(function() {\r\n 'use strict';\r\n\r\n // ---- Helpers ----\r\n const $ = (s) => document.querySelector(s);\r\n const $$ = (s) => document.querySelectorAll(s);\r\n\r\n function formatUptime(seconds) {\r\n const h = Math.floor(seconds / 3600);\r\n const m = Math.floor((seconds % 3600) / 60);\r\n const s = seconds % 60;\r\n return String(h).padStart(2,'0') + ':' + String(m).padStart(2,'0') + ':' + String(s).padStart(2,'0');\r\n }\r\n\r\n function timeAgo(ts) {\r\n if (!ts) return '--';\r\n const diff = Math.floor(Date.now() / 1000 - ts);\r\n if (diff < 0) return 'now';\r\n if (diff < 60) return diff + 's ago';\r\n if (diff < 3600) return Math.floor(diff / 60) + 'm ago';\r\n return Math.floor(diff / 3600) + 'h ago';\r\n }\r\n\r\n function flashEl(el) {\r\n el.classList.remove('flash');\r\n void el.offsetWidth;\r\n el.classList.add('flash');\r\n }\r\n\r\n const statusColors = {\r\n PARTY_SERVER: '#00ff88',\r\n DEGRADED: '#ffaa00',\r\n SUSPECT: '#ff8800',\r\n DOWN: '#ff3366',\r\n UNKNOWN: '#6a6a8a',\r\n NOT_SERVER: '#6a6a8a',\r\n MAYBE: '#00fff0',\r\n };\r\n\r\n const statusBadgeClass = {\r\n PARTY_SERVER: 'badge-party',\r\n DEGRADED: 'badge-degraded',\r\n SUSPECT: 'badge-suspect',\r\n DOWN: 'badge-down',\r\n UNKNOWN: 'badge-unknown',\r\n NOT_SERVER: 'badge-not_server',\r\n MAYBE: 'badge-maybe',\r\n };\r\n\r\n // ---- State ----\r\n let overview = null;\r\n let prevStats = null;\r\n\r\n // ---- Fetch helpers ----\r\n async function fetchStats() {\r\n try {\r\n const r = await fetch('/dashboard/api/stats');\r\n return await r.json();\r\n } catch { return null; }\r\n }\r\n\r\n async function fetchOverview() {\r\n try {\r\n const r = await fetch('/dashboard/api/overview');\r\n return await r.json();\r\n } catch { return null; }\r\n }\r\n\r\n // ---- Render functions ----\r\n function renderHeader(data) {\r\n const s = data.server;\r\n if (tsState && tsState.state === 'connected') {\r\n $('#statusDot').className = 'status-dot';\r\n $('#statusText').textContent = 'ONLINE';\r\n $('#serverInfo').textContent = s.tailscale_ip + ' // ' + s.hostname;\r\n } else if (tsState && tsState.state === 'not_connected') {\r\n $('#statusDot').className = 'status-dot not-connected';\r\n $('#statusText').textContent = 'NOT CONNECTED';\r\n $('#serverInfo').textContent = 'Tailscale installed but not authenticated';\r\n } else if (tsState && tsState.state === 'not_installed') {\r\n $('#statusDot').className = 'status-dot not-installed';\r\n $('#statusText').textContent = 'NO TAILSCALE';\r\n $('#serverInfo').textContent = 'Tailscale not installed - local mode';\r\n } else {\r\n $('#statusDot').className = 'status-dot';\r\n $('#statusText').textContent = 'ONLINE';\r\n $('#serverInfo').textContent = s.tailscale_ip + ' // ' + s.hostname;\r\n }\r\n $('#uptime').textContent = formatUptime(s.uptime_seconds);\r\n }\r\n\r\n function renderStats(data) {\r\n const prev = {\r\n local: parseInt($('#localAgentCount').textContent) || 0,\r\n remote: parseInt($('#remoteAgentCount').textContent) || 0,\r\n peer: parseInt($('#peerCount').textContent) || 0,\r\n party: parseInt($('#partyServerCount').textContent) || 0,\r\n };\r\n const vals = {\r\n local: data.agents.local_count,\r\n remote: data.agents.remote_count,\r\n peer: data.peers.total,\r\n party: data.peers.party_servers,\r\n };\r\n if (vals.local !== prev.local) { $('#localAgentCount').textContent = vals.local; flashEl($('#localAgentCount')); }\r\n if (vals.remote !== prev.remote) { $('#remoteAgentCount').textContent = vals.remote; flashEl($('#remoteAgentCount')); }\r\n if (vals.peer !== prev.peer) { $('#peerCount').textContent = vals.peer; flashEl($('#peerCount')); }\r\n if (vals.party !== prev.party) { $('#partyServerCount').textContent = vals.party; flashEl($('#partyServerCount')); }\r\n }\r\n\r\n function renderTopology(data) {\r\n const svg = $('#topologySvg');\r\n const peers = data.peers.details || [];\r\n const cx = 250, cy = 150, radius = 100;\r\n\r\n let html = '';\r\n\r\n // Center node\r\n html += '<circle cx=\"' + cx + '\" cy=\"' + cy + '\" r=\"24\" fill=\"rgba(0,255,240,0.15)\" stroke=\"#00fff0\" stroke-width=\"2\">';\r\n html += '<animate attributeName=\"r\" values=\"24;26;24\" dur=\"3s\" repeatCount=\"indefinite\"/>';\r\n html += '</circle>';\r\n html += '<text x=\"' + cx + '\" y=\"' + cy + '\" text-anchor=\"middle\" dominant-baseline=\"central\" fill=\"#00fff0\" font-family=\"var(--font-mono)\" font-size=\"10\" font-weight=\"700\">SELF</text>';\r\n html += '<text x=\"' + cx + '\" y=\"' + (cy + 38) + '\" text-anchor=\"middle\" class=\"topo-label\">' + data.server.tailscale_ip + '</text>';\r\n\r\n if (peers.length === 0) {\r\n html += '<text x=\"' + cx + '\" y=\"' + (cy + 60) + '\" text-anchor=\"middle\" fill=\"#6a6a8a\" font-family=\"var(--font-mono)\" font-size=\"11\">No peers discovered</text>';\r\n }\r\n\r\n peers.forEach(function(p, i) {\r\n const angle = (2 * Math.PI * i / Math.max(peers.length, 1)) - Math.PI / 2;\r\n const px = cx + radius * Math.cos(angle);\r\n const py = cy + radius * Math.sin(angle);\r\n const color = statusColors[p.status] || '#6a6a8a';\r\n const opacity = (p.status === 'PARTY_SERVER' || p.status === 'DEGRADED' || p.status === 'SUSPECT') ? 1 : 0.4;\r\n\r\n // Connection line\r\n html += '<line x1=\"' + cx + '\" y1=\"' + cy + '\" x2=\"' + px + '\" y2=\"' + py + '\" stroke=\"' + color + '\" stroke-width=\"1\" opacity=\"' + (opacity * 0.3) + '\"/>';\r\n\r\n // Peer node\r\n html += '<g class=\"topo-node\"><circle cx=\"' + px + '\" cy=\"' + py + '\" r=\"16\" fill=\"rgba(255,255,255,0.03)\" stroke=\"' + color + '\" stroke-width=\"1.5\" opacity=\"' + opacity + '\"/>';\r\n html += '<text x=\"' + px + '\" y=\"' + py + '\" text-anchor=\"middle\" dominant-baseline=\"central\" fill=\"' + color + '\" font-family=\"var(--font-mono)\" font-size=\"8\" opacity=\"' + opacity + '\">' + p.ip.split('.').slice(-1)[0] + '</text></g>';\r\n\r\n // IP label\r\n html += '<text x=\"' + px + '\" y=\"' + (py + 26) + '\" text-anchor=\"middle\" class=\"topo-label\">' + p.ip + '</text>';\r\n });\r\n\r\n svg.innerHTML = html;\r\n }\r\n\r\n function renderPeerTable(data) {\r\n const tbody = $('#peerTableBody');\r\n const peers = data.peers.details || [];\r\n\r\n if (peers.length === 0) {\r\n tbody.innerHTML = '<tr><td colspan=\"4\" class=\"empty-state\">No peers discovered</td></tr>';\r\n return;\r\n }\r\n\r\n // Sort: PARTY_SERVER first, then by severity\r\n const order = { PARTY_SERVER: 0, DEGRADED: 1, SUSPECT: 2, MAYBE: 3, UNKNOWN: 4, NOT_SERVER: 5, DOWN: 6 };\r\n const sorted = [...peers].sort(function(a, b) { return (order[a.status] || 99) - (order[b.status] || 99); });\r\n\r\n tbody.innerHTML = sorted.map(function(p) {\r\n const badge = statusBadgeClass[p.status] || 'badge-unknown';\r\n const label = p.status === 'PARTY_SERVER' ? 'SERVER' : p.status === 'NOT_SERVER' ? 'NOT_SVR' : p.status;\r\n return '<tr>'\r\n + '<td>' + p.ip + '</td>'\r\n + '<td><span class=\"badge ' + badge + '\">' + label + '</span></td>'\r\n + '<td>' + p.consecutiveFailures + '</td>'\r\n + '<td>' + timeAgo(p.lastProbeAt) + '</td>'\r\n + '</tr>';\r\n }).join('');\r\n }\r\n\r\n function renderAgents(data) {\r\n const container = $('#agentList');\r\n const local = data.agents.local_agents || [];\r\n const remote = data.agents.remote_agents || [];\r\n const all = [\r\n ...local.map(function(a) { return { ...a, type: 'local' }; }),\r\n ...remote.map(function(a) { return { ...a, type: 'remote' }; }),\r\n ];\r\n\r\n if (all.length === 0) {\r\n container.innerHTML = '<div class=\"empty-state\">No agents registered</div>';\r\n return;\r\n }\r\n\r\n container.innerHTML = all.map(function(a) {\r\n const initials = (a.display_name || a.agent_id).substring(0, 2).toUpperCase();\r\n const tags = [];\r\n if (a.type === 'remote') {\r\n tags.push('<span class=\"agent-tag\">' + a.source_peer_ip + '</span>');\r\n if (!a.reachable) tags.push('<span class=\"agent-tag unreachable\">offline</span>');\r\n } else {\r\n tags.push('<span class=\"agent-tag\">local</span>');\r\n }\r\n return '<div class=\"agent-card ' + a.type + '\">'\r\n + '<div class=\"agent-icon\">' + initials + '</div>'\r\n + '<div class=\"agent-info\">'\r\n + '<div class=\"agent-name\">' + (a.display_name || a.agent_id) + '</div>'\r\n + '<div class=\"agent-id\">' + a.agent_id + '</div>'\r\n + '</div>'\r\n + '<div class=\"agent-meta\">' + tags.join('') + '</div>'\r\n + '</div>';\r\n }).join('');\r\n }\r\n\r\n function renderMessages(data) {\r\n const container = $('#msgFeed');\r\n const msgs = data.messages.recent || [];\r\n\r\n if (msgs.length === 0) {\r\n container.innerHTML = '<div class=\"empty-state\">No recent messages</div>';\r\n return;\r\n }\r\n\r\n container.innerHTML = msgs.map(function(m) {\r\n const dir = m.direction === 'received' ? 'received' : '';\r\n const arrow = m.direction === 'received' ? '&#8592;' : '&#8594;';\r\n const flow = m.direction === 'received'\r\n ? m.sender_id + ' <span class=\"arrow\">' + arrow + '</span> ' + (m.agent_id || '?')\r\n : (m.agent_id || '?') + ' <span class=\"arrow\">' + arrow + '</span> ' + (m.recipient_id || 'broadcast');\r\n return '<div class=\"msg-item ' + dir + '\">'\r\n + '<div class=\"msg-top\">'\r\n + '<div class=\"msg-flow\">' + flow + '</div>'\r\n + '<div class=\"msg-time\">' + timeAgo(m.timestamp) + '</div>'\r\n + '</div>'\r\n + '<div class=\"msg-content\">' + (m.summary || m.content) + '</div>'\r\n + '</div>';\r\n }).join('');\r\n }\r\n\r\n function renderAll(data) {\r\n renderHeader(data);\r\n renderStats(data);\r\n renderTopology(data);\r\n renderPeerTable(data);\r\n renderAgents(data);\r\n renderMessages(data);\r\n }\r\n\r\n // ---- Polling ----\r\n let fullTimer = null;\r\n let fastTimer = null;\r\n\r\n async function fullRefresh() {\r\n const data = await fetchOverview();\r\n if (!data) {\r\n $('#statusDot').className = 'status-dot offline';\r\n $('#statusText').textContent = 'OFFLINE';\r\n return;\r\n }\r\n overview = data;\r\n renderAll(data);\r\n }\r\n\r\n async function fastRefresh() {\r\n const stats = await fetchStats();\r\n if (!stats) return;\r\n const changed = JSON.stringify(stats) !== JSON.stringify(prevStats);\r\n prevStats = stats;\r\n if (changed) {\r\n fullRefresh();\r\n }\r\n }\r\n\r\n // ---- Init ----\r\n fullRefresh();\r\n fastTimer = setInterval(fastRefresh, 3000);\r\n fullTimer = setInterval(fullRefresh, 5000);\r\n\r\n // Clipboard click delegation for .cmd elements\r\n document.addEventListener('click', function(e) {\r\n var cmd = e.target.closest('.cmd');\r\n if (!cmd) return;\r\n var text = cmd.getAttribute('data-clipboard');\r\n if (text) {\r\n navigator.clipboard.writeText(text).then(function() {\r\n var hint = cmd.querySelector('.copy-hint');\r\n if (hint) hint.textContent = 'Copied!';\r\n });\r\n }\r\n });\r\n\r\n // Update uptime display every second\r\n setInterval(function() {\r\n if (overview && overview.server) {\r\n overview.server.uptime_seconds++;\r\n $('#uptime').textContent = formatUptime(overview.server.uptime_seconds);\r\n }\r\n }, 1000);\r\n\r\n // ---- Join Modal Tabs ----\r\n const joinTabs = $$('#joinTabs .tab');\r\n joinTabs.forEach(function(tab) {\r\n tab.addEventListener('click', function() {\r\n joinTabs.forEach(function(t) { t.classList.remove('active'); });\r\n tab.classList.add('active');\r\n // Toggle tab contents\r\n const target = tab.getAttribute('data-tab');\r\n $$('.tab-content').forEach(function(tc) { tc.classList.remove('active'); });\r\n if (target === 'interactive') {\r\n $('#tabInteractive').classList.add('active');\r\n } else {\r\n $('#tabAuthkey').classList.add('active');\r\n }\r\n });\r\n });\r\n\r\n // ---- Join Network Modal (open/close) ----\r\n const joinModal = $('#joinModal');\r\n const btnJoin = $('#btnJoinNetwork');\r\n const btnCancel = $('#btnCancelJoin');\r\n const btnCancelAuthkey = $('#btnCancelAuthkey');\r\n const btnSubmit = $('#btnSubmitJoin');\r\n const authKeyInput = $('#authKeyInput');\r\n const joinStatus = $('#joinStatus');\r\n\r\n function openJoinModal() {\r\n // Reset both tabs\r\n joinStatus.className = 'modal-status';\r\n joinStatus.textContent = '';\r\n authKeyInput.value = '';\r\n $('#interactiveStatus').className = 'modal-status';\r\n $('#interactiveStatus').textContent = '';\r\n // Default to Interactive tab\r\n $$('#joinTabs .tab').forEach(function(t) { t.classList.remove('active'); });\r\n $$('#joinTabs .tab')[0].classList.add('active');\r\n $$('.tab-content').forEach(function(tc) { tc.classList.remove('active'); });\r\n $('#tabInteractive').classList.add('active');\r\n joinModal.classList.add('open');\r\n }\r\n\r\n function closeJoinModal() {\r\n joinModal.classList.remove('open');\r\n }\r\n\r\n btnJoin.addEventListener('click', function() {\r\n // Decide action based on Tailscale state\r\n if (tsState && tsState.state === 'connected') {\r\n openLogoutModal();\r\n } else if (tsState && tsState.state === 'not_installed') {\r\n doInstallTailscale();\r\n } else {\r\n openJoinModal();\r\n }\r\n });\r\n btnCancel.addEventListener('click', closeJoinModal);\r\n btnCancelAuthkey.addEventListener('click', closeJoinModal);\r\n joinModal.addEventListener('click', function(e) {\r\n if (e.target === joinModal) closeJoinModal();\r\n });\r\n authKeyInput.addEventListener('keydown', function(e) {\r\n if (e.key === 'Enter') btnSubmit.click();\r\n if (e.key === 'Escape') closeJoinModal();\r\n });\r\n\r\n // ---- Auth Key submit ----\r\n btnSubmit.addEventListener('click', async function() {\r\n const key = authKeyInput.value.trim();\r\n if (!key) {\r\n authKeyInput.focus();\r\n return;\r\n }\r\n btnSubmit.disabled = true;\r\n btnSubmit.innerHTML = '<span class=\"spinner\"></span>Connecting...';\r\n joinStatus.className = 'modal-status';\r\n joinStatus.textContent = '';\r\n\r\n try {\r\n const r = await fetch('/dashboard/api/join-network', {\r\n method: 'POST',\r\n headers: { 'Content-Type': 'application/json' },\r\n body: JSON.stringify({ auth_key: key }),\r\n });\r\n const data = await r.json();\r\n if (data.success) {\r\n joinStatus.className = 'modal-status success';\r\n joinStatus.textContent = 'Successfully joined network!';\r\n btnJoin.textContent = 'Logout';\r\n btnJoin.className = 'btn-join btn-logout';\r\n setTimeout(function() { closeJoinModal(); checkTailscaleStatus(); fullRefresh(); }, 1500);\r\n } else {\r\n joinStatus.className = 'modal-status error';\r\n joinStatus.textContent = data.output || 'Failed to join network';\r\n }\r\n } catch (e) {\r\n joinStatus.className = 'modal-status error';\r\n joinStatus.textContent = 'Network error: ' + (e.message || 'unknown');\r\n }\r\n btnSubmit.disabled = false;\r\n btnSubmit.textContent = 'Connect';\r\n });\r\n\r\n // ---- Interactive Login ----\r\n const btnInteractiveLogin = $('#btnInteractiveLogin');\r\n btnInteractiveLogin.addEventListener('click', async function() {\r\n const statusEl = $('#interactiveStatus');\r\n statusEl.className = 'modal-status';\r\n statusEl.textContent = '';\r\n btnInteractiveLogin.disabled = true;\r\n btnInteractiveLogin.innerHTML = '<span class=\"spinner\"></span>Opening browser...';\r\n\r\n try {\r\n const r = await fetch('/dashboard/api/tailscale-login', { method: 'POST' });\r\n const data = await r.json();\r\n\r\n if (data.success && data.url) {\r\n // Open the auth URL in a new tab\r\n window.open(data.url, '_blank');\r\n statusEl.className = 'modal-status success';\r\n statusEl.textContent = 'Authentication page opened in your browser. Waiting for connection...';\r\n\r\n // Poll for connection\r\n var pollCount = 0;\r\n var pollInterval = setInterval(async function() {\r\n pollCount++;\r\n if (pollCount > 40) { // 2 minutes timeout\r\n clearInterval(pollInterval);\r\n statusEl.className = 'modal-status error';\r\n statusEl.textContent = 'Timed out waiting for authentication. Please try again.';\r\n btnInteractiveLogin.disabled = false;\r\n btnInteractiveLogin.textContent = 'Open Browser Login';\r\n return;\r\n }\r\n try {\r\n var sr = await fetch('/dashboard/api/tailscale-status');\r\n var sd = await sr.json();\r\n if (sd.state === 'connected') {\r\n clearInterval(pollInterval);\r\n btnJoin.textContent = 'Logout';\r\n btnJoin.className = 'btn-join btn-logout';\r\n closeJoinModal();\r\n checkTailscaleStatus();\r\n fullRefresh();\r\n return;\r\n }\r\n } catch { /* poll error, continue */ }\r\n }, 3000);\r\n } else {\r\n statusEl.className = 'modal-status error';\r\n statusEl.textContent = data.output || 'Failed to start interactive login';\r\n btnInteractiveLogin.disabled = false;\r\n btnInteractiveLogin.textContent = 'Open Browser Login';\r\n }\r\n } catch (e) {\r\n statusEl.className = 'modal-status error';\r\n statusEl.textContent = 'Network error: ' + (e.message || 'unknown');\r\n btnInteractiveLogin.disabled = false;\r\n btnInteractiveLogin.textContent = 'Open Browser Login';\r\n }\r\n });\r\n\r\n // ---- Logout Modal ----\r\n const logoutModal = $('#logoutModal');\r\n const btnConfirmLogout = $('#btnConfirmLogout');\r\n const btnCancelLogout = $('#btnCancelLogout');\r\n const logoutStatus = $('#logoutStatus');\r\n\r\n function openLogoutModal() {\r\n logoutStatus.className = 'modal-status';\r\n logoutStatus.textContent = '';\r\n logoutModal.classList.add('open');\r\n }\r\n\r\n btnCancelLogout.addEventListener('click', function() { logoutModal.classList.remove('open'); });\r\n logoutModal.addEventListener('click', function(e) { if (e.target === logoutModal) logoutModal.classList.remove('open'); });\r\n\r\n btnConfirmLogout.addEventListener('click', async function() {\r\n btnConfirmLogout.disabled = true;\r\n btnConfirmLogout.innerHTML = '<span class=\"spinner\"></span>Logging out...';\r\n logoutStatus.className = 'modal-status';\r\n logoutStatus.textContent = '';\r\n\r\n try {\r\n const r = await fetch('/dashboard/api/logout', { method: 'POST' });\r\n const data = await r.json();\r\n logoutModal.classList.remove('open');\r\n if (data.success) {\r\n checkTailscaleStatus();\r\n fullRefresh();\r\n } else {\r\n alert('Logout failed: ' + (data.output || 'unknown error'));\r\n }\r\n } catch (e) {\r\n logoutModal.classList.remove('open');\r\n alert('Network error: ' + (e.message || 'unknown'));\r\n }\r\n btnConfirmLogout.disabled = false;\r\n btnConfirmLogout.textContent = 'Log Out';\r\n });\r\n\r\n // ---- Install Tailscale ----\r\n async function doInstallTailscale() {\r\n if (!confirm('Install Tailscale on this machine?')) return;\r\n\r\n btnJoin.disabled = true;\r\n btnJoin.innerHTML = '<span class=\"spinner\"></span>Installing...';\r\n\r\n try {\r\n const r = await fetch('/dashboard/api/install-tailscale', { method: 'POST' });\r\n const data = await r.json();\r\n if (data.success) {\r\n btnJoin.textContent = 'Installed';\r\n btnJoin.disabled = false;\r\n checkTailscaleStatus();\r\n fullRefresh();\r\n } else {\r\n alert('Installation failed: ' + (data.output || 'unknown error'));\r\n btnJoin.textContent = 'Install Tailscale';\r\n btnJoin.className = 'btn-join btn-install';\r\n btnJoin.disabled = false;\r\n }\r\n } catch (e) {\r\n alert('Network error: ' + (e.message || 'unknown'));\r\n btnJoin.textContent = 'Install Tailscale';\r\n btnJoin.className = 'btn-join btn-install';\r\n btnJoin.disabled = false;\r\n }\r\n }\r\n\r\n // Check initial Tailscale status (tri-state)\r\n let tsState = null;\r\n let tsInstallInfo = null;\r\n\r\n async function checkTailscaleStatus() {\r\n try {\r\n const r = await fetch('/dashboard/api/tailscale-status');\r\n tsState = await r.json();\r\n } catch { tsState = { state: 'not_installed', platform: 'unknown' }; }\r\n\r\n const dot = $('#statusDot');\r\n const text = $('#statusText');\r\n const btnJoin = $('#btnJoinNetwork');\r\n const panel = $('#tsPanel');\r\n\r\n if (tsState.state === 'connected') {\r\n dot.className = 'status-dot';\r\n text.textContent = 'ONLINE';\r\n btnJoin.textContent = 'Logout';\r\n btnJoin.className = 'btn-join btn-logout';\r\n btnJoin.style.display = '';\r\n panel.style.display = 'none';\r\n } else if (tsState.state === 'not_installed') {\r\n dot.className = 'status-dot not-installed';\r\n text.textContent = 'NOT INSTALLED';\r\n btnJoin.textContent = 'Install Tailscale';\r\n btnJoin.className = 'btn-join btn-install';\r\n btnJoin.style.display = '';\r\n await renderNotInstalledPanel();\r\n } else {\r\n dot.className = 'status-dot not-connected';\r\n text.textContent = 'NOT CONNECTED';\r\n btnJoin.textContent = 'Join Network';\r\n btnJoin.className = 'btn-join';\r\n btnJoin.style.display = '';\r\n await renderNotConnectedPanel();\r\n }\r\n }\r\n\r\n async function fetchInstallInfo() {\r\n if (tsInstallInfo) return tsInstallInfo;\r\n try {\r\n const r = await fetch('/dashboard/api/tailscale-install-info');\r\n tsInstallInfo = await r.json();\r\n } catch { tsInstallInfo = null; }\r\n return tsInstallInfo;\r\n }\r\n\r\n async function renderNotInstalledPanel() {\r\n const info = await fetchInstallInfo();\r\n const panel = $('#tsPanel');\r\n let html = '<div class=\"ts-panel-title not-installed\">Tailscale Not Installed</div>';\r\n html += '<div class=\"ts-info-row\"><span class=\"label\">Status:</span><span class=\"value\" style=\"color:var(--red)\">Tailscale is not detected on this system</span></div>';\r\n\r\n if (info && info.commands && info.commands.length > 0) {\r\n html += '<div class=\"ts-install-guide\">';\r\n html += '<div style=\"color:var(--muted);margin-bottom:6px\">Install for ' + info.os + ':</div>';\r\n info.commands.forEach(function(cmd) {\r\n const display = info.needs_sudo ? 'sudo ' + cmd : cmd;\r\n html += '<div class=\"cmd\" data-clipboard=\"' + display.replace(/\"/g, '&quot;') + '\">';\r\n html += '<code>' + display + '</code><span class=\"copy-hint\">Click to copy</span></div>';\r\n });\r\n if (info.download_url) {\r\n html += '<div style=\"margin-top:8px\">Download: <a href=\"' + info.download_url + '\" target=\"_blank\">' + info.download_url + '</a></div>';\r\n }\r\n html += '</div>';\r\n }\r\n\r\n html += '<div class=\"ts-setup-hint\">Or click the <strong>Install Tailscale</strong> button above, or run <code style=\"color:var(--cyan)\">npx open-party setup</code></div>';\r\n html += '<button class=\"btn-redetect\" onclick=\"window.__redetectTailscale()\">Re-detect</button>';\r\n panel.innerHTML = html;\r\n panel.style.display = 'block';\r\n }\r\n\r\n async function renderNotConnectedPanel() {\r\n const panel = $('#tsPanel');\r\n\r\n let html = '<div class=\"ts-panel-title not-connected\">Tailscale Not Connected</div>';\r\n html += '<div class=\"ts-info-row\"><span class=\"label\">Status:</span><span class=\"value\" style=\"color:var(--yellow)\">Installed but not authenticated</span></div>';\r\n html += '<div class=\"ts-setup-hint\">Use the <strong>Join Network</strong> button above to log in</div>';\r\n html += '<button class=\"btn-redetect\" onclick=\"window.__redetectTailscale()\">Re-detect</button>';\r\n panel.innerHTML = html;\r\n panel.style.display = 'block';\r\n }\r\n\r\n window.__redetectTailscale = async function() {\r\n const panel = $('#tsPanel');\r\n panel.innerHTML = '<div style=\"color:var(--muted);padding:12px\"><span class=\"spinner\"></span> Re-detecting Tailscale...</div>';\r\n try {\r\n await fetch('/dashboard/api/tailscale-detect', { method: 'POST' });\r\n } catch { /* ignore */ }\r\n await checkTailscaleStatus();\r\n fullRefresh();\r\n };\r\n\r\n checkTailscaleStatus();\r\n\r\n})();\r\n</script>\r\n</body>\r\n</html>`;\r\n","import { Hono } from 'hono';\r\nimport { sanitizeAgentList } from '../models.js';\r\nimport { DASHBOARD_HTML } from '../dashboard-html.js';\r\nimport { getTailnetHostname, getTailscaleInstallationStatus, resetTailscaleBinaryCache, getInstallInstructions, joinTailnet, logoutTailscale, startInteractiveLogin } from '../../infra/tailscale.js';\r\nimport { registry, messageQueue, discovery, getSelfIp, refreshSelfIp, STARTED_AT } from '../state.js';\r\nimport { type ChildProcess } from 'node:child_process';\r\n\r\nconst dashboardRoutes = new Hono();\r\n\r\n// Serve the dashboard HTML page\r\ndashboardRoutes.get('/', (c) => {\r\n return c.html(DASHBOARD_HTML);\r\n});\r\n\r\n// Lightweight stats endpoint for fast polling\r\ndashboardRoutes.get('/api/stats', async (c) => {\r\n const localAgents = registry.listAll();\r\n const remoteAgents = discovery.getReachableRemoteAgents();\r\n const peerStates = discovery.getPeerStates();\r\n const partyServers = peerStates.filter((p) => p.status === 'PARTY_SERVER' || p.status === 'DEGRADED' || p.status === 'SUSPECT');\r\n\r\n return c.json({\r\n local_agent_count: localAgents.length,\r\n remote_agent_count: remoteAgents.length,\r\n peer_count: peerStates.length,\r\n party_server_count: partyServers.length,\r\n });\r\n});\r\n\r\n// Full overview endpoint for complete refresh\r\ndashboardRoutes.get('/api/overview', async (c) => {\r\n let hostname = '127.0.0.1';\r\n try {\r\n hostname = getTailnetHostname();\r\n } catch {\r\n /* ignore */\r\n }\r\n\r\n const localAgents = sanitizeAgentList(registry.listAll());\r\n const remoteEntries = discovery.getRemoteAgentEntries();\r\n const peerStates = discovery.getPeerStates();\r\n\r\n // Collect recent messages from local agents\r\n const recentMessages: Array<{\r\n agent_id: string;\r\n direction: string;\r\n sender_id: string;\r\n recipient_id?: string;\r\n summary?: string;\r\n content: string;\r\n timestamp: number;\r\n }> = [];\r\n\r\n for (const agent of localAgents) {\r\n const history = messageQueue.getHistory(agent.agent_id, 5);\r\n for (const entry of history) {\r\n recentMessages.push({ agent_id: agent.agent_id, ...entry });\r\n }\r\n }\r\n // Sort by timestamp descending, cap at 20\r\n recentMessages.sort((a, b) => b.timestamp - a.timestamp);\r\n if (recentMessages.length > 20) recentMessages.length = 20;\r\n\r\n const partyServers = peerStates.filter(\r\n (p) => p.status === 'PARTY_SERVER' || p.status === 'DEGRADED' || p.status === 'SUSPECT',\r\n );\r\n\r\n return c.json({\r\n server: {\r\n status: 'ok',\r\n tailscale_ip: getSelfIp(),\r\n hostname,\r\n uptime_seconds: Math.floor((Date.now() - STARTED_AT) / 1000),\r\n },\r\n agents: {\r\n local_count: localAgents.length,\r\n remote_count: remoteEntries.length,\r\n local_agents: localAgents,\r\n remote_agents: remoteEntries.map((e) => ({\r\n ...sanitizeAgentList([e.agentInfo])[0],\r\n source_peer_ip: e.sourcePeerIp,\r\n reachable: e.reachable,\r\n })),\r\n },\r\n peers: {\r\n total: peerStates.length,\r\n party_servers: partyServers.length,\r\n down: peerStates.filter((p) => p.status === 'DOWN').length,\r\n unknown: peerStates.filter((p) => p.status === 'UNKNOWN' || p.status === 'MAYBE').length,\r\n details: peerStates,\r\n },\r\n messages: {\r\n recent: recentMessages,\r\n },\r\n });\r\n});\r\n\r\nexport { dashboardRoutes };\r\n\r\n// ---------------------------------------------------------------------------\r\n// Tailscale network management\r\n// ---------------------------------------------------------------------------\r\n\r\ndashboardRoutes.get('/api/tailscale-status', async (c) => {\r\n try {\r\n return c.json(getTailscaleInstallationStatus());\r\n } catch (e) {\r\n return c.json({ state: 'not_installed', platform: process.platform, error: (e as Error).message });\r\n }\r\n});\r\n\r\ndashboardRoutes.post('/api/tailscale-detect', async (c) => {\r\n resetTailscaleBinaryCache();\r\n const state = getTailscaleInstallationStatus();\r\n if (state.state === 'connected') {\r\n refreshSelfIp();\r\n }\r\n return c.json(state);\r\n});\r\n\r\ndashboardRoutes.get('/api/tailscale-install-info', async (c) => {\r\n return c.json(getInstallInstructions(process.platform));\r\n});\r\n\r\ndashboardRoutes.post('/api/join-network', async (c) => {\r\n try {\r\n const body = await c.req.json<{ auth_key?: string }>();\r\n const authKey = (body.auth_key ?? '').trim();\r\n if (!authKey) {\r\n return c.json({ success: false, output: 'auth_key is required' }, 400);\r\n }\r\n const result = joinTailnet(authKey);\r\n return c.json(result, result.success ? 200 : 500);\r\n } catch (e) {\r\n return c.json({ success: false, output: (e as Error).message }, 500);\r\n }\r\n});\r\n\r\n// ---------------------------------------------------------------------------\r\n// Logout, interactive login, install\r\n// ---------------------------------------------------------------------------\r\n\r\n// Track the active interactive login process so we can reuse it if\r\n// the client polls again before authentication completes.\r\nlet activeLogin: { process: ChildProcess; url?: string } | null = null;\r\n\r\ndashboardRoutes.post('/api/logout', async (c) => {\r\n const result = logoutTailscale();\r\n if (result.success) {\r\n resetTailscaleBinaryCache();\r\n refreshSelfIp();\r\n }\r\n return c.json(result, result.success ? 200 : 500);\r\n});\r\n\r\ndashboardRoutes.post('/api/tailscale-login', async (c) => {\r\n // If we already have an active login with a URL, return it\r\n if (activeLogin?.url) {\r\n return c.json({ success: true, url: activeLogin.url });\r\n }\r\n\r\n const { promise, process } = startInteractiveLogin();\r\n activeLogin = { process };\r\n\r\n const result = await promise;\r\n\r\n if (result.success && result.url) {\r\n activeLogin.url = result.url;\r\n return c.json({ success: true, url: result.url });\r\n }\r\n\r\n // Failed — clean up\r\n activeLogin = null;\r\n return c.json({ success: false, output: result.output }, 500);\r\n});\r\n\r\ndashboardRoutes.post('/api/install-tailscale', async (c) => {\r\n // Dynamic import to avoid pulling the heavy installer into the server bundle\r\n // unless actually needed\r\n const { installTailscale } = await import('../../cli/tailscale-installer.js');\r\n const result = await installTailscale(process.platform);\r\n if (result.success) {\r\n resetTailscaleBinaryCache();\r\n }\r\n return c.json(result, result.success ? 200 : 500);\r\n});\r\n","import { Hono } from 'hono';\r\nimport { cors } from 'hono/cors';\r\nimport { serve } from '@hono/node-server';\r\nimport { existsSync, mkdirSync, writeFileSync, unlinkSync } from 'node:fs';\r\nimport { join, dirname } from 'node:path';\r\nimport { homedir } from 'node:os';\r\nimport { PARTY_PORT, CLEANUP_INTERVAL, HEARTBEAT_TIMEOUT } from './config.js';\r\nimport { getSelfIp, discovery } from './state.js';\r\nimport { agentRoutes } from './routes/agent.js';\r\nimport { proxyRoutes } from './routes/proxy.js';\r\nimport { dashboardRoutes } from './routes/dashboard.js';\r\n\r\n// ---------------------------------------------------------------------------\r\n// Background cleanup task\r\n// ---------------------------------------------------------------------------\r\n\r\nasync function periodicCleanup(): Promise<void> {\r\n // TODO: heartbeat mechanism ready then resume cleanup\r\n}\r\n\r\n// ---------------------------------------------------------------------------\r\n// App\r\n// ---------------------------------------------------------------------------\r\n\r\nconst app = new Hono();\r\n\r\napp.use('*', cors());\r\n\r\napp.route('/agent', agentRoutes);\r\napp.route('/proxy', proxyRoutes);\r\napp.route('/dashboard', dashboardRoutes);\r\n\r\n// ---------------------------------------------------------------------------\r\n// Entry point\r\n// ---------------------------------------------------------------------------\r\n\r\n// ---------------------------------------------------------------------------\r\n// PID file self-management (so `stop` and `status` always find us)\r\n// ---------------------------------------------------------------------------\r\n\r\nfunction pidFilePath(): string {\r\n const pluginData = process.env.CLAUDE_PLUGIN_DATA || '';\r\n if (pluginData) return join(pluginData, 'server.pid');\r\n return join(homedir(), '.open-party', 'server.pid');\r\n}\r\n\r\n// ---------------------------------------------------------------------------\r\n// Entry point\r\n// ---------------------------------------------------------------------------\r\n\r\nasync function main() {\r\n // Write PID file so CLI commands can find this process\r\n const pidPath = pidFilePath();\r\n mkdirSync(dirname(pidPath), { recursive: true });\r\n writeFileSync(pidPath, String(process.pid));\r\n\r\n console.log(`Starting Party Server on port ${PARTY_PORT} (Tailscale IP: ${getSelfIp()})`);\r\n\r\n // Ignore SIGHUP so the process survives when the parent terminal exits\r\n process.on('SIGHUP', () => {});\r\n\r\n const server = serve({ fetch: app.fetch, port: PARTY_PORT });\r\n\r\n // Start background tasks\r\n const discoveryPromise = discovery.runLoop();\r\n const cleanupPromise = periodicCleanup();\r\n\r\n // Graceful shutdown\r\n const shutdown = () => {\r\n console.log('\\nShutting down Party Server...');\r\n try { unlinkSync(pidPath); } catch { /* ignore */ }\r\n server.close();\r\n process.exit(0);\r\n };\r\n\r\n process.on('SIGINT', shutdown);\r\n process.on('SIGTERM', shutdown);\r\n\r\n await Promise.race([discoveryPromise, cleanupPromise]);\r\n}\r\n\r\nmain().catch((e) => {\r\n console.error('Fatal error:', e);\r\n process.exit(1);\r\n});\r\n\r\n// Catch unhandled errors that would silently crash the process\r\nprocess.on('uncaughtException', (e) => {\r\n console.error('Uncaught exception:', e);\r\n});\r\nprocess.on('unhandledRejection', (e) => {\r\n console.error('Unhandled rejection:', e);\r\n});\r\n","/**\n * Interactive setup wizard for Open Party.\n *\n * Steps:\n * 1. Detect Tailscale → install if missing → login\n * 2. Detect AI agents → install Open Party plugin\n */\n\nimport {\n getTailscaleInstallationStatus,\n resetTailscaleBinaryCache,\n getInstallInstructions,\n type TailscaleState,\n} from '../infra/tailscale.js';\nimport { installTailscale } from './tailscale-installer.js';\nimport { detectAgents, type DetectedAgent } from './agent-detector.js';\nimport { installPluginToAgent } from './agent-installer.js';\nimport { interactiveLogin, authKeyLogin } from './tailscale-login.js';\nimport {\n cyan,\n green,\n yellow,\n red,\n bold,\n dim,\n select,\n multiSelect,\n prompt,\n createRl,\n closeRl,\n} from './tty-utils.js';\n\n// ---------------------------------------------------------------------------\n// Step 1: Tailscale\n// ---------------------------------------------------------------------------\n\nasync function stepTailscale(): Promise<void> {\n console.log(`\\n${bold(cyan('🔍 Step 1: Tailscale Network'))}\\n`);\n\n console.log(\n 'Tailscale enables agents across different machines to discover and\\n' +\n 'communicate with each other over a secure network.\\n',\n );\n console.log(\n `${dim('Without Tailscale, Open Party runs in local mode — connecting only\\n' +\n 'to agents on this machine.')}\\n`,\n );\n\n const status = getTailscaleInstallationStatus();\n\n if (status.state === 'connected') {\n console.log(`${green('✅ Tailscale is connected!')}`);\n console.log(` IP: ${status.tailscale_ip} Hostname: ${status.hostname}`);\n return;\n }\n\n if (status.state === 'not_installed') {\n await handleNotInstalled(status.platform);\n // Re-detect after possible installation\n const newStatus = getTailscaleInstallationStatus();\n if (newStatus.state === 'not_installed') {\n // User chose to skip or install failed\n showLocalModeNotice();\n return;\n }\n if (newStatus.state === 'connected') {\n console.log(`\\n${green('✅ Tailscale is connected!')} IP: ${newStatus.tailscale_ip}`);\n return;\n }\n // not_connected → fall through to login\n await handleNotConnected(newStatus.binary);\n return;\n }\n\n // not_connected\n await handleNotConnected(status.binary);\n}\n\nasync function handleNotInstalled(platform: string): Promise<void> {\n const info = getInstallInstructions(platform);\n\n console.log(`${red('❌ Tailscale is not installed.')}`);\n console.log(`\\nInstall Tailscale for ${bold(info.os)}:\\n`);\n\n if (info.commands.length > 0) {\n for (const cmd of info.commands) {\n const prefix = info.needs_sudo ? 'sudo ' : '';\n console.log(` ${cyan(prefix + cmd)}`);\n }\n }\n\n console.log(`\\n Download: ${info.download_url}\\n`);\n\n // Build options\n type InstallChoice = 'auto' | 'redetect' | 'skip';\n const options: { label: string; value: InstallChoice; hint?: string }[] = [];\n\n if (info.commands.length > 0 && platform !== 'win32') {\n options.push({ label: 'Install Tailscale automatically', value: 'auto', hint: 'recommended' });\n }\n options.push({ label: \"I've installed Tailscale, re-detect\", value: 'redetect' });\n options.push({ label: 'Skip — use local mode only', value: 'skip', hint: 'agents on this machine only' });\n\n const choice = await select(options, { message: 'Choose:' });\n\n if (choice === 'skip') {\n return; // Local mode notice shown by caller\n }\n\n if (choice === 'auto') {\n console.log('');\n const result = await installTailscale(platform);\n if (result.success) {\n console.log(`${green('✅ Tailscale installed successfully!')}`);\n } else {\n console.log(`${red('❌ Installation failed:')}\\n${result.output}`);\n console.log(`\\n Please install manually and re-run: ${cyan('open-party setup')}`);\n }\n return;\n }\n\n if (choice === 'redetect') {\n resetTailscaleBinaryCache();\n // Caller will re-detect\n return;\n }\n}\n\nasync function handleNotConnected(binary: string): Promise<void> {\n console.log(`${yellow('🔒 Tailscale is installed but not connected.')}\\n`);\n\n type LoginChoice = 'interactive' | 'authkey' | 'skip';\n const options: { label: string; value: LoginChoice; hint?: string }[] = [\n { label: 'Interactive login', value: 'interactive', hint: 'opens browser to authenticate' },\n { label: 'Auth key', value: 'authkey', hint: 'from network creator' },\n { label: 'Skip', value: 'skip', hint: 'login later with: open-party login' },\n ];\n\n const choice = await select(options, { message: 'Choose a login method:' });\n\n if (choice === 'interactive') {\n await interactiveLogin(binary);\n } else if (choice === 'authkey') {\n await authKeyLogin(binary);\n } else {\n console.log(`\\n${yellow('⚠️ Tailscale not connected. Running in local mode.')}`);\n console.log(` To connect later, run: ${cyan('open-party login')}`);\n }\n}\n\nfunction showLocalModeNotice(): void {\n console.log(`\\n${yellow('⚠️ Running in local mode — connecting to agents on this machine only.')}`);\n console.log(' To enable cross-machine communication later:');\n console.log(` 1. Install Tailscale: ${cyan('https://tailscale.com/download')}`);\n console.log(` 2. Run: ${cyan('open-party login')}`);\n}\n\n// ---------------------------------------------------------------------------\n// Step 2: Agent plugin\n// ---------------------------------------------------------------------------\n\nasync function stepAgentPlugin(): Promise<void> {\n console.log(`\\n${bold(cyan('🔍 Step 2: Detecting AI agents in your environment...'))}\\n`);\n\n const agents = detectAgents();\n const detected = agents.filter((a) => a.detected);\n\n if (detected.length === 0) {\n console.log(yellow('No supported AI agents detected in this environment.'));\n console.log(' Supported agents: Claude Code, Cursor, Gemini CLI');\n console.log('');\n console.log(' Install one and re-run: open-party setup');\n return;\n }\n\n console.log('Detected agents:\\n');\n for (const agent of detected) {\n console.log(` ${green('✓')} ${agent.name}`);\n }\n console.log('');\n\n // Use multiSelect for agent selection\n const options = detected.map((a) => ({ label: a.name, value: a }));\n\n const selected = await multiSelect(options, {\n message: 'Select agents to install Open Party plugin:',\n });\n\n if (selected.length === 0) {\n console.log(yellow('No agents selected, skipping plugin installation.'));\n return;\n }\n\n for (const agent of selected) {\n console.log(`\\nInstalling Open Party plugin for ${agent.name}...`);\n const result = await installPluginToAgent(agent.type);\n if (result.success) {\n console.log(`${green('✅')} Plugin installed for ${agent.name}${result.configPath ? ` (${result.configPath})` : ''}`);\n if (result.warning) {\n console.log(` ${yellow('⚠️')} ${result.warning}`);\n }\n if (agent.type === 'claude-code') {\n console.log(` ${bold('Please restart Claude Code')} for changes to take effect.`);\n }\n } else {\n console.log(`${red('❌')} Failed to install for ${agent.name}: ${result.error}`);\n }\n }\n}\n\n// ---------------------------------------------------------------------------\n// Main\n// ---------------------------------------------------------------------------\n\nexport async function setupCommand(): Promise<void> {\n console.log(bold(cyan('\\n🚀 Open Party Setup Wizard\\n')));\n\n const rl = createRl();\n\n await stepTailscale();\n await stepAgentPlugin();\n\n // Start the Party Server in the background\n console.log(`\\n${bold(cyan('🚀 Starting Party Server...'))}`);\n const { spawn } = await import('node:child_process');\n const { resolve, dirname } = await import('node:path');\n const { fileURLToPath } = await import('node:url');\n\n // Find party-server.js relative to this file\n const __dirname = dirname(fileURLToPath(import.meta.url));\n const serverScript = resolve(__dirname, '..', 'party-server.js');\n\n const serverProc = spawn(process.execPath, [serverScript], {\n detached: true,\n stdio: 'ignore',\n windowsHide: true,\n });\n serverProc.unref();\n\n // Wait a moment for the server to start\n await new Promise((r) => setTimeout(r, 2000));\n\n console.log(`\\n${bold(green('🎉 Setup complete!'))}`);\n console.log(` Dashboard: http://127.0.0.1:8000/dashboard`);\n console.log(' Other agents can join with: npx @feynmanzhang/open-party setup\\n');\n\n closeRl(rl);\n}","/**\n * Detect AI agents in the user's environment.\n *\n * Checks for Claude Code, Cursor, and Gemini CLI by looking for\n * configuration files and PATH executables.\n */\n\nimport { existsSync } from 'node:fs';\nimport { join } from 'node:path';\nimport { homedir } from 'node:os';\nimport { execSync } from 'node:child_process';\n\nexport type AgentType = 'claude-code' | 'cursor' | 'gemini-cli';\n\nexport interface DetectedAgent {\n type: AgentType;\n name: string;\n detected: boolean;\n configPath: string;\n}\n\nfunction isExecutableInPath(name: string): boolean {\n try {\n const which = process.platform === 'win32' ? 'where' : 'which';\n execSync(`${which} ${name}`, { timeout: 3000, stdio: 'pipe', windowsHide: true });\n return true;\n } catch {\n return false;\n }\n}\n\nfunction detectClaudeCode(): DetectedAgent {\n const configDir = join(homedir(), '.claude');\n const settingsPath = join(configDir, 'settings.json');\n return {\n type: 'claude-code',\n name: 'Claude Code',\n detected: existsSync(settingsPath) || isExecutableInPath('claude'),\n configPath: settingsPath,\n };\n}\n\nfunction detectCursor(): DetectedAgent {\n const configPath =\n process.platform === 'win32'\n ? join(homedir(), 'AppData', 'Roaming', 'Cursor', 'User', 'globalStorage', 'settings.json')\n : join(homedir(), '.cursor', 'mcp.json');\n // Check for .cursor/mcp.json (new location) or fallback\n const mcpPath = join(homedir(), '.cursor', 'mcp.json');\n return {\n type: 'cursor',\n name: 'Cursor',\n detected: existsSync(mcpPath) || isExecutableInPath('cursor'),\n configPath: mcpPath,\n };\n}\n\nfunction detectGeminiCli(): DetectedAgent {\n return {\n type: 'gemini-cli',\n name: 'Gemini CLI',\n detected: isExecutableInPath('gemini'),\n configPath: join(homedir(), '.gemini', 'settings.json'),\n };\n}\n\nexport function detectAgents(): DetectedAgent[] {\n return [detectClaudeCode(), detectCursor(), detectGeminiCli()];\n}","/**\r\n * Install Open Party plugin to detected AI agents.\r\n *\r\n * Each agent type has its own configuration mechanism:\r\n * - Claude Code: Full plugin registration (copy to plugins/cache, installed_plugins.json, enabledPlugins)\r\n * Claude Code auto-discovers MCP servers (.mcp.json), hooks (hooks/hooks.json), and skills (skills/STAR/SKILL.md)\r\n * from the plugin directory.\r\n * - Cursor: MCP server config in ~/.cursor/mcp.json\r\n * - Gemini CLI: MCP server config in ~/.gemini/settings.json\r\n */\r\n\r\nimport { existsSync, readFileSync, writeFileSync, mkdirSync, cpSync, rmSync, readdirSync } from 'node:fs';\r\nimport { join, dirname, resolve } from 'node:path';\r\nimport { homedir } from 'node:os';\r\nimport type { AgentType } from './agent-detector.js';\r\n\r\nexport interface InstallResult {\r\n success: boolean;\r\n configPath?: string;\r\n error?: string;\r\n warning?: string;\r\n}\r\n\r\n// ---------------------------------------------------------------------------\r\n// Shared helpers\r\n// ---------------------------------------------------------------------------\r\n\r\ninterface SettingsConfig {\r\n mcpServers?: Record<string, McpServerConfig>;\r\n enabledPlugins?: Record<string, boolean>;\r\n [key: string]: unknown;\r\n}\r\n\r\ninterface McpServerConfig {\r\n type?: string;\r\n command: string;\r\n args: string[];\r\n}\r\n\r\ninterface PluginsData {\r\n version?: number;\r\n plugins?: Record<string, unknown[]>;\r\n}\r\n\r\nfunction ensureDir(filePath: string): void {\r\n const dir = dirname(filePath);\r\n if (!existsSync(dir)) {\r\n mkdirSync(dir, { recursive: true });\r\n }\r\n}\r\n\r\nfunction readJsonFile<T>(filePath: string, fallback: T): T {\r\n if (!existsSync(filePath)) return fallback;\r\n try {\r\n return JSON.parse(readFileSync(filePath, 'utf-8')) as T;\r\n } catch {\r\n return fallback;\r\n }\r\n}\r\n\r\nfunction writeJsonFile(filePath: string, data: unknown): void {\r\n ensureDir(filePath);\r\n writeFileSync(filePath, JSON.stringify(data, null, 2) + '\\n');\r\n}\r\n\r\n// ---------------------------------------------------------------------------\r\n// Plugin directory discovery\r\n// ---------------------------------------------------------------------------\r\n\r\n/**\r\n * Find the built plugin directory under dist/claude-code/.\r\n * Returns the absolute path to the plugin root (e.g., dist/claude-code/open-party-0.1.1/)\r\n * or null if not found.\r\n */\r\nfunction findPluginDistDir(): string | null {\r\n const distDir = resolve(import.meta.dirname ?? '.', '..', 'claude-code');\r\n if (!existsSync(distDir)) return null;\r\n\r\n try {\r\n const entries = readdirSync(distDir);\r\n const dirs = entries.filter((e: string) => e.startsWith('open-party-'));\r\n if (dirs.length === 0) return null;\r\n\r\n // Use the latest version if multiple exist\r\n const pluginDir = join(distDir, dirs[dirs.length - 1]);\r\n if (existsSync(join(pluginDir, '.claude-plugin', 'plugin.json'))) {\r\n return pluginDir;\r\n }\r\n } catch {\r\n // ignore\r\n }\r\n return null;\r\n}\r\n\r\n/**\r\n * Find the dist/ directory containing compiled JS files (mcp-server.js, hook-handler.js).\r\n * Used when the full plugin directory isn't available but we still need the MCP server.\r\n */\r\nfunction findDistJsDir(): string | null {\r\n const possiblePaths = [\r\n // When installed globally via npm: <pkg-root>/dist/cli/index.js → <pkg-root>/dist/\r\n resolve(import.meta.dirname ?? '.', '..'),\r\n ];\r\n\r\n for (const p of possiblePaths) {\r\n if (existsSync(join(p, 'mcp-server.js')) && existsSync(join(p, 'hook-handler.js'))) {\r\n return p;\r\n }\r\n }\r\n return null;\r\n}\r\n\r\n/** Read plugin version from plugin.json */\r\nfunction getPluginVersion(): string {\r\n // Try to read from the built plugin directory first\r\n const pluginDir = findPluginDistDir();\r\n if (pluginDir) {\r\n const manifest = readJsonFile<{ version?: string }>(\r\n join(pluginDir, '.claude-plugin', 'plugin.json'),\r\n {},\r\n );\r\n if (manifest.version) return manifest.version;\r\n }\r\n\r\n // Fallback: try source directory\r\n const sourceManifest = resolve(\r\n import.meta.dirname ?? '.',\r\n '..',\r\n '..',\r\n 'src',\r\n 'client',\r\n 'claude-code',\r\n '.claude-plugin',\r\n 'plugin.json',\r\n );\r\n const manifest = readJsonFile<{ version?: string }>(sourceManifest, {});\r\n if (manifest.version) return manifest.version;\r\n\r\n return '0.1.0';\r\n}\r\n\r\n// ---------------------------------------------------------------------------\r\n// Claude Code — Full plugin registration\r\n//\r\n// Claude Code uses convention-based discovery for plugins:\r\n// - .claude-plugin/plugin.json → plugin metadata\r\n// - .mcp.json → MCP server config (uses ${CLAUDE_PLUGIN_ROOT})\r\n// - hooks/hooks.json → lifecycle hooks (uses ${CLAUDE_PLUGIN_ROOT})\r\n// - skills/*/SKILL.md → skill definitions\r\n//\r\n// We only need to:\r\n// 1. Copy plugin files to ~/.claude/plugins/cache/open-party/{version}/\r\n// 2. Register in installed_plugins.json\r\n// 3. Enable in settings.json (enabledPlugins)\r\n// 4. Remove any old manual MCP server entry (from previous installs)\r\n//\r\n// Claude Code handles the rest automatically at runtime by setting\r\n// CLAUDE_PLUGIN_ROOT and discovering the plugin's config files.\r\n// ---------------------------------------------------------------------------\r\n\r\ninterface MarketplaceSource {\r\n source: string;\r\n repo?: string;\r\n path?: string;\r\n}\r\n\r\ninterface MarketplaceEntry {\r\n source: MarketplaceSource;\r\n installLocation: string;\r\n lastUpdated: string;\r\n}\r\n\r\ninterface KnownMarketplaces {\r\n [name: string]: MarketplaceEntry;\r\n}\r\n\r\n// ---------------------------------------------------------------------------\r\n// Marketplace registration\r\n//\r\n// Claude Code requires each installed plugin to belong to a registered\r\n// marketplace. We create a \"open-party\" marketplace that mirrors the GitHub\r\n// repo structure. The marketplace directory contains:\r\n// - .claude-plugin/marketplace.json — marketplace manifest listing the plugin\r\n// - plugin/ — the actual plugin files (symlinked/copied from cache)\r\n//\r\n// The known_marketplaces.json entry uses the GitHub source format so Claude\r\n// Code recognizes it as a valid marketplace.\r\n// ---------------------------------------------------------------------------\r\n\r\n/** Directory for the open-party marketplace: ~/.claude/plugins/marketplaces/open-party/ */\r\nfunction getMarketplaceDir(): string {\r\n return join(homedir(), '.claude', 'plugins', 'marketplaces', 'open-party');\r\n}\r\n\r\n/**\r\n * Register the \"open-party\" marketplace so Claude Code can discover the plugin.\r\n * Creates:\r\n * - ~/.claude/plugins/marketplaces/open-party/.claude-plugin/marketplace.json\r\n * - ~/.claude/plugins/marketplaces/open-party/plugin/.claude-plugin/plugin.json\r\n * (plus other plugin files copied from cache)\r\n * - Entry in ~/.claude/plugins/known_marketplaces.json\r\n */\r\nfunction registerMarketplace(version: string, pluginDir: string): void {\r\n const marketplaceDir = getMarketplaceDir();\r\n const pluginSourceDir = join(marketplaceDir, 'plugin');\r\n\r\n // --- Create marketplace manifest ---\r\n const marketplacePluginDir = join(marketplaceDir, '.claude-plugin');\r\n if (!existsSync(marketplacePluginDir)) {\r\n mkdirSync(marketplacePluginDir, { recursive: true });\r\n }\r\n\r\n const marketplaceManifest: {\r\n name: string;\r\n owner: { name: string };\r\n metadata: { description: string; homepage: string };\r\n plugins: Array<{ name: string; version: string; source: string; description: string }>;\r\n } = {\r\n name: 'open-party',\r\n owner: { name: 'Feynman Zhang' },\r\n metadata: {\r\n description: 'Decentralized Agent communication network for Claude Code',\r\n homepage: 'https://github.com/FeynmanZhang/open-party',\r\n },\r\n plugins: [\r\n {\r\n name: 'open-party',\r\n version,\r\n source: './plugin',\r\n description: 'Decentralized Agent communication network for Claude Code',\r\n },\r\n ],\r\n };\r\n\r\n writeJsonFile(join(marketplacePluginDir, 'marketplace.json'), marketplaceManifest);\r\n\r\n // --- Copy plugin files to marketplace plugin directory ---\r\n if (existsSync(pluginSourceDir)) {\r\n rmSync(pluginSourceDir, { recursive: true });\r\n }\r\n mkdirSync(pluginSourceDir, { recursive: true });\r\n cpSync(pluginDir, pluginSourceDir, { recursive: true });\r\n\r\n // --- Register in known_marketplaces.json ---\r\n const knownMarketplacesPath = join(homedir(), '.claude', 'plugins', 'known_marketplaces.json');\r\n const knownMarketplaces = readJsonFile<KnownMarketplaces>(knownMarketplacesPath, {});\r\n\r\n knownMarketplaces['open-party'] = {\r\n source: {\r\n source: 'github',\r\n repo: 'FeynmanZhang/open-party',\r\n },\r\n installLocation: marketplaceDir,\r\n lastUpdated: new Date().toISOString(),\r\n };\r\n writeJsonFile(knownMarketplacesPath, knownMarketplaces);\r\n}\r\n\r\n/** Unregister the \"open-party\" marketplace and remove its directory. */\r\nfunction unregisterMarketplace(): void {\r\n // Remove from known_marketplaces.json\r\n const knownMarketplacesPath = join(homedir(), '.claude', 'plugins', 'known_marketplaces.json');\r\n const knownMarketplaces = readJsonFile<KnownMarketplaces>(knownMarketplacesPath, {});\r\n\r\n if (knownMarketplaces['open-party']) {\r\n delete knownMarketplaces['open-party'];\r\n writeJsonFile(knownMarketplacesPath, knownMarketplaces);\r\n }\r\n\r\n // Also clean up old \"local\" marketplace entry if it exists (migration)\r\n if (knownMarketplaces['local']) {\r\n delete knownMarketplaces['local'];\r\n writeJsonFile(knownMarketplacesPath, knownMarketplaces);\r\n }\r\n\r\n // Remove marketplace directory\r\n const marketplaceDir = getMarketplaceDir();\r\n if (existsSync(marketplaceDir)) {\r\n rmSync(marketplaceDir, { recursive: true });\r\n }\r\n\r\n // Also remove old local marketplace directory if it exists (migration)\r\n const oldLocalDir = join(homedir(), '.claude', 'plugins', 'marketplaces', 'local');\r\n if (existsSync(oldLocalDir)) {\r\n rmSync(oldLocalDir, { recursive: true });\r\n }\r\n}\r\n\r\nfunction installClaudeCode(): InstallResult {\r\n const pluginDir = findPluginDistDir();\r\n\r\n if (!pluginDir) {\r\n // No built plugin directory available — cannot do full plugin install\r\n return {\r\n success: false,\r\n error: 'Plugin package not found. Run \"npm run build:plugin\" first, or use \"claude --plugin-dir\" to install manually.',\r\n };\r\n }\r\n\r\n const version = getPluginVersion();\r\n\r\n // --- Step 1: Copy plugin files to ~/.claude/plugins/cache/open-party/open-party/{version}/ ---\r\n const installDir = join(homedir(), '.claude', 'plugins', 'cache', 'open-party', 'open-party', version);\r\n\r\n if (existsSync(installDir)) {\r\n rmSync(installDir, { recursive: true });\r\n }\r\n mkdirSync(installDir, { recursive: true });\r\n cpSync(pluginDir, installDir, { recursive: true });\r\n\r\n // Verify essential files exist after copy\r\n const mcpServerPath = join(installDir, 'dist', 'mcp-server.js');\r\n if (!existsSync(mcpServerPath)) {\r\n // The plugin dist/ subdirectory might be missing compiled files\r\n // Try to find them from the main dist/ and copy them in\r\n const distJsDir = findDistJsDir();\r\n if (distJsDir) {\r\n const targetDist = join(installDir, 'dist');\r\n if (!existsSync(targetDist)) mkdirSync(targetDist, { recursive: true });\r\n for (const file of ['mcp-server.js', 'hook-handler.js', 'party-server.js']) {\r\n const src = join(distJsDir, file);\r\n if (existsSync(src)) {\r\n cpSync(src, join(targetDist, file));\r\n }\r\n }\r\n for (const file of ['mcp-server.js.map', 'hook-handler.js.map', 'party-server.js.map']) {\r\n const src = join(distJsDir, file);\r\n if (existsSync(src)) {\r\n cpSync(src, join(targetDist, file));\r\n }\r\n }\r\n }\r\n }\r\n\r\n // Remove orphaned marker if present from a previous failed install\r\n const orphanedPath = join(installDir, 'dist', '.orphaned_at');\r\n if (existsSync(orphanedPath)) {\r\n rmSync(orphanedPath);\r\n }\r\n\r\n // --- Step 2: Register the open-party marketplace ---\r\n // Claude Code requires each plugin to belong to a registered marketplace.\r\n // We create an \"open-party\" marketplace that points to our GitHub repo\r\n // and contains the plugin files.\r\n registerMarketplace(version, pluginDir);\r\n\r\n // --- Step 3: Register in installed_plugins.json ---\r\n const pluginsJsonPath = join(homedir(), '.claude', 'plugins', 'installed_plugins.json');\r\n const pluginsData = readJsonFile<PluginsData>(\r\n pluginsJsonPath,\r\n { version: 2, plugins: {} },\r\n );\r\n if (!pluginsData.plugins) pluginsData.plugins = {};\r\n\r\n // Clean up old \"open-party@local\" entry if it exists (migration from previous version)\r\n if (pluginsData.plugins['open-party@local']) {\r\n delete pluginsData.plugins['open-party@local'];\r\n }\r\n\r\n pluginsData.plugins['open-party@open-party'] = [\r\n {\r\n scope: 'user',\r\n installPath: installDir,\r\n version,\r\n installedAt: new Date().toISOString(),\r\n lastUpdated: new Date().toISOString(),\r\n },\r\n ];\r\n writeJsonFile(pluginsJsonPath, pluginsData);\r\n\r\n // --- Step 4: Enable plugin & clean up old MCP entry in settings.json ---\r\n const settingsPath = join(homedir(), '.claude', 'settings.json');\r\n const settings = readJsonFile<SettingsConfig>(settingsPath, {});\r\n\r\n // Remove old manual MCP server entry (from previous install method)\r\n // The MCP server is now auto-discovered from the plugin's .mcp.json\r\n if (settings.mcpServers?.['open-party']) {\r\n delete settings.mcpServers['open-party'];\r\n if (Object.keys(settings.mcpServers).length === 0) {\r\n delete settings.mcpServers;\r\n }\r\n }\r\n\r\n // Enable the plugin\r\n if (!settings.enabledPlugins) {\r\n settings.enabledPlugins = {};\r\n }\r\n\r\n // Clean up old \"open-party@local\" entry (migration from previous version)\r\n if (settings.enabledPlugins['open-party@local'] !== undefined) {\r\n delete settings.enabledPlugins['open-party@local'];\r\n }\r\n\r\n settings.enabledPlugins['open-party@open-party'] = true;\r\n\r\n writeJsonFile(settingsPath, settings);\r\n\r\n return {\r\n success: true,\r\n configPath: settingsPath,\r\n };\r\n}\r\n\r\n/** Remove Open Party plugin from Claude Code completely */\r\nexport function uninstallClaudeCode(): InstallResult {\r\n const version = getPluginVersion();\r\n\r\n // Remove from settings.json\r\n const settingsPath = join(homedir(), '.claude', 'settings.json');\r\n const settings = readJsonFile<SettingsConfig>(settingsPath, {});\r\n\r\n // Remove old MCP server entry (if present)\r\n if (settings.mcpServers?.['open-party']) {\r\n delete settings.mcpServers['open-party'];\r\n if (Object.keys(settings.mcpServers).length === 0) {\r\n delete settings.mcpServers;\r\n }\r\n }\r\n\r\n // Remove enabled plugin entries (both old and new format)\r\n if (settings.enabledPlugins?.['open-party@open-party'] !== undefined) {\r\n delete settings.enabledPlugins['open-party@open-party'];\r\n }\r\n if (settings.enabledPlugins?.['open-party@local'] !== undefined) {\r\n delete settings.enabledPlugins['open-party@local'];\r\n }\r\n if (settings.enabledPlugins && Object.keys(settings.enabledPlugins).length === 0) {\r\n delete settings.enabledPlugins;\r\n }\r\n\r\n writeJsonFile(settingsPath, settings);\r\n\r\n // Remove from installed_plugins.json\r\n const pluginsJsonPath = join(homedir(), '.claude', 'plugins', 'installed_plugins.json');\r\n const pluginsData = readJsonFile<PluginsData>(\r\n pluginsJsonPath,\r\n { version: 2, plugins: {} },\r\n );\r\n if (pluginsData.plugins) {\r\n delete pluginsData.plugins['open-party@open-party'];\r\n delete pluginsData.plugins['open-party@local']; // Clean up old format too\r\n writeJsonFile(pluginsJsonPath, pluginsData);\r\n }\r\n\r\n // Remove plugin files (both old and new path formats)\r\n const installDirNew = join(homedir(), '.claude', 'plugins', 'cache', 'open-party', 'open-party', version);\r\n const installDirOld = join(homedir(), '.claude', 'plugins', 'cache', 'open-party', version);\r\n for (const dir of [installDirNew, installDirOld]) {\r\n if (existsSync(dir)) {\r\n rmSync(dir, { recursive: true });\r\n }\r\n }\r\n\r\n // Unregister the marketplace\r\n unregisterMarketplace();\r\n\r\n return { success: true, configPath: settingsPath };\r\n}\r\n\r\n// ---------------------------------------------------------------------------\r\n// Cursor — MCP server config only\r\n// ---------------------------------------------------------------------------\r\n\r\nfunction installCursor(): InstallResult {\r\n const configPath = join(homedir(), '.cursor', 'mcp.json');\r\n const { command, args } = getPluginCommand();\r\n\r\n const config = readJsonFile<SettingsConfig>(configPath, {});\r\n if (!config.mcpServers) config.mcpServers = {};\r\n\r\n config.mcpServers['open-party'] = { type: 'stdio', command, args };\r\n writeJsonFile(configPath, config);\r\n\r\n return { success: true, configPath };\r\n}\r\n\r\n// ---------------------------------------------------------------------------\r\n// Gemini CLI — MCP server config only\r\n// ---------------------------------------------------------------------------\r\n\r\nfunction installGeminiCli(): InstallResult {\r\n const configPath = join(homedir(), '.gemini', 'settings.json');\r\n const { command, args } = getPluginCommand();\r\n\r\n const config = readJsonFile<SettingsConfig>(configPath, {});\r\n if (!config.mcpServers) config.mcpServers = {};\r\n\r\n config.mcpServers['open-party'] = { type: 'stdio', command, args };\r\n writeJsonFile(configPath, config);\r\n\r\n return { success: true, configPath };\r\n}\r\n\r\n// ---------------------------------------------------------------------------\r\n// Shared MCP command resolution (for Cursor & Gemini CLI)\r\n// ---------------------------------------------------------------------------\r\n\r\nfunction getPluginCommand(): { command: string; args: string[] } {\r\n const distDir = findDistJsDir();\r\n if (distDir) {\r\n const serverPath = join(distDir, 'mcp-server.js');\r\n if (existsSync(serverPath)) {\r\n return { command: 'node', args: [serverPath] };\r\n }\r\n }\r\n\r\n // Fallback: use npx to run open-party as MCP server\r\n return { command: 'npx', args: ['@feynmanzhang/open-party', 'mcp'] };\r\n}\r\n\r\n// ---------------------------------------------------------------------------\r\n// Public API\r\n// ---------------------------------------------------------------------------\r\n\r\nexport async function installPluginToAgent(agentType: AgentType): Promise<InstallResult> {\r\n switch (agentType) {\r\n case 'claude-code':\r\n return installClaudeCode();\r\n case 'cursor':\r\n return installCursor();\r\n case 'gemini-cli':\r\n return installGeminiCli();\r\n default:\r\n return { success: false, error: `Unknown agent type: ${agentType}` };\r\n }\r\n}","/**\n * Shared Tailscale login logic for CLI commands.\n *\n * Extracted from setup.ts so both `setup` and `login` can reuse it.\n */\n\nimport { spawn } from 'node:child_process';\nimport {\n joinTailnet,\n resetTailscaleBinaryCache,\n getTailscaleInstallationStatus,\n} from '../infra/tailscale.js';\nimport { prompt, green, yellow, red, bold, cyan } from './tty-utils.js';\n\n// ---------------------------------------------------------------------------\n// Interactive login (opens browser)\n// ---------------------------------------------------------------------------\n\nexport async function interactiveLogin(binary: string): Promise<boolean> {\n console.log(`\\n${cyan('Running interactive login...')}`);\n console.log('A browser window should open. Authenticate in the browser, then return here.\\n');\n\n const child = spawn(binary, ['login'], { stdio: 'inherit' });\n\n const exitCode = await new Promise<number | null>((resolve) => {\n child.on('close', resolve);\n });\n\n resetTailscaleBinaryCache();\n const status = getTailscaleInstallationStatus();\n\n if (exitCode === 0 && status.state === 'connected') {\n console.log(`\\n${green('✅ Login successful!')} IP: ${status.tailscale_ip}`);\n showAuthKeyTip();\n return true;\n }\n\n console.log(`\\n${yellow('⚠️ Login may not have completed. Status: ' + status.state)}`);\n console.log(' Try running: open-party login');\n return false;\n}\n\n// ---------------------------------------------------------------------------\n// Auth key login\n// ---------------------------------------------------------------------------\n\nexport async function authKeyLogin(binary: string): Promise<boolean> {\n console.log('');\n console.log('Ask the network creator to generate an Auth Key at:');\n console.log(`${cyan(' https://login.tailscale.com/admin/settings/keys')}\\n`);\n\n const authKey = await prompt('Enter Auth Key: ');\n if (!authKey) {\n console.log(yellow('No auth key provided, skipping login.'));\n return false;\n }\n\n const result = joinTailnet(authKey);\n\n if (result.success) {\n resetTailscaleBinaryCache();\n const status = getTailscaleInstallationStatus();\n console.log(`\\n${green('✅ Login successful!')} IP: ${status.state === 'connected' ? status.tailscale_ip : 'unknown'}`);\n showAuthKeyTip();\n return true;\n }\n\n console.log(`\\n${red('❌ Login failed:')}\\n${result.output}`);\n console.log(' Check your auth key and try again.');\n return false;\n}\n\n// ---------------------------------------------------------------------------\n// Auth key sharing tip\n// ---------------------------------------------------------------------------\n\nexport function showAuthKeyTip(): void {\n console.log('');\n console.log(`${bold('💡 To share network access with teammates:')}`);\n console.log(' 1. Go to https://login.tailscale.com/admin/settings/keys');\n console.log(' 2. Generate an Auth Key');\n console.log(' 3. Share it with teammates — they can run: open-party login');\n}","/**\n * Terminal UI utilities for interactive CLI prompts.\n *\n * Provides arrow-key selectors (select / multiSelect), text input (prompt),\n * and ANSI color helpers.\n */\n\nimport { createInterface, type Interface } from 'node:readline';\n\n// ---------------------------------------------------------------------------\n// Colors\n// ---------------------------------------------------------------------------\n\nexport function cyan(text: string): string {\n return `\\x1b[36m${text}\\x1b[0m`;\n}\nexport function green(text: string): string {\n return `\\x1b[32m${text}\\x1b[0m`;\n}\nexport function yellow(text: string): string {\n return `\\x1b[33m${text}\\x1b[0m`;\n}\nexport function red(text: string): string {\n return `\\x1b[31m${text}\\x1b[0m`;\n}\nexport function bold(text: string): string {\n return `\\x1b[1m${text}\\x1b[0m`;\n}\nexport function dim(text: string): string {\n return `\\x1b[2m${text}\\x1b[0m`;\n}\n\n// ---------------------------------------------------------------------------\n// Readline lifecycle\n// ---------------------------------------------------------------------------\n\nexport function createRl(): Interface {\n return createInterface({ input: process.stdin, output: process.stdout });\n}\n\nexport function closeRl(rl: Interface): void {\n rl.close();\n}\n\n// ---------------------------------------------------------------------------\n// Text prompt (for secrets, auth keys, etc.)\n// ---------------------------------------------------------------------------\n\nexport async function prompt(question: string): Promise<string> {\n const rl = createInterface({ input: process.stdin, output: process.stdout });\n return new Promise<string>((resolve) => {\n rl.question(question, (answer) => {\n rl.close();\n resolve(answer.trim());\n });\n });\n}\n\n// ---------------------------------------------------------------------------\n// Arrow-key single select\n// ---------------------------------------------------------------------------\n\nexport interface SelectOption<T> {\n label: string;\n value: T;\n hint?: string;\n}\n\nexport async function select<T>(\n options: SelectOption<T>[],\n opts?: { message?: string },\n): Promise<T> {\n if (options.length === 0) throw new Error('select() requires at least one option');\n if (options.length === 1) return options[0].value;\n\n const message = opts?.message ?? '';\n\n // Ensure stdin is in raw mode for key-by-key input\n const wasRaw = process.stdin.isRaw;\n if (process.stdin.isTTY) process.stdin.setRawMode(true);\n\n try {\n let cursor = 0;\n\n const render = () => {\n // Move cursor up N lines, clear, and redraw\n const lines = options.length + (message ? 1 : 0);\n process.stdout.write(`\\x1b[${lines}A\\x1b[0J`);\n\n if (message) {\n process.stdout.write(`${message}\\n`);\n }\n for (let i = 0; i < options.length; i++) {\n const opt = options[i];\n const prefix = i === cursor ? `${cyan('❯')} ` : ' ';\n const label = i === cursor ? bold(opt.label) : opt.label;\n const hintStr = opt.hint ? ` ${dim(opt.hint)}` : '';\n process.stdout.write(`${prefix}${label}${hintStr}\\n`);\n }\n };\n\n // Initial render\n if (message) {\n process.stdout.write(`${message}\\n`);\n }\n for (let i = 0; i < options.length; i++) {\n const opt = options[i];\n const prefix = i === cursor ? `${cyan('❯')} ` : ' ';\n const label = i === cursor ? bold(opt.label) : opt.label;\n const hintStr = opt.hint ? ` ${dim(opt.hint)}` : '';\n process.stdout.write(`${prefix}${label}${hintStr}\\n`);\n }\n\n return new Promise<T>((resolve) => {\n const onKey = (ch: string, key: { name?: string; sequence?: string }) => {\n if (key.name === 'up' || key.sequence === '\\x1b[A') {\n cursor = (cursor - 1 + options.length) % options.length;\n render();\n } else if (key.name === 'down' || key.sequence === '\\x1b[B') {\n cursor = (cursor + 1) % options.length;\n render();\n } else if (key.name === 'return' || key.sequence === '\\r') {\n process.stdin.removeListener('keypress', onKey);\n process.stdout.write('\\n');\n resolve(options[cursor].value);\n }\n };\n\n process.stdin.on('keypress', onKey);\n // Emit keypress events on stdin\n if (process.stdin.isTTY) {\n process.stdin.resume();\n }\n });\n } finally {\n if (process.stdin.isTTY) process.stdin.setRawMode(wasRaw ?? false);\n }\n}\n\n// ---------------------------------------------------------------------------\n// Arrow-key multi-select (space to toggle, enter to confirm)\n// ---------------------------------------------------------------------------\n\nexport interface MultiSelectOption<T> {\n label: string;\n value: T;\n}\n\nexport async function multiSelect<T>(\n options: MultiSelectOption<T>[],\n opts?: { message?: string },\n): Promise<T[]> {\n if (options.length === 0) return [];\n\n const message = opts?.message ?? '';\n\n const wasRaw = process.stdin.isRaw;\n if (process.stdin.isTTY) process.stdin.setRawMode(true);\n\n try {\n let cursor = 0;\n const selected = new Set<number>();\n\n const render = () => {\n const lines = options.length + (message ? 1 : 0) + 1; // +1 for instruction line\n process.stdout.write(`\\x1b[${lines}A\\x1b[0J`);\n\n if (message) {\n process.stdout.write(`${message}\\n`);\n }\n process.stdout.write(`${dim(' ↑/↓ navigate, space to select, enter to confirm')}\\n`);\n for (let i = 0; i < options.length; i++) {\n const opt = options[i];\n const prefix = i === cursor ? `${cyan('❯')} ` : ' ';\n const check = selected.has(i) ? green('◉') : '○';\n const label = i === cursor ? bold(opt.label) : opt.label;\n process.stdout.write(`${prefix}${check} ${label}\\n`);\n }\n };\n\n // Initial render\n if (message) {\n process.stdout.write(`${message}\\n`);\n }\n process.stdout.write(`${dim(' ↑/↓ navigate, space to select, enter to confirm')}\\n`);\n for (let i = 0; i < options.length; i++) {\n const opt = options[i];\n const prefix = i === cursor ? `${cyan('❯')} ` : ' ';\n const check = selected.has(i) ? green('◉') : '○';\n const label = i === cursor ? bold(opt.label) : opt.label;\n process.stdout.write(`${prefix}${check} ${label}\\n`);\n }\n\n return new Promise<T[]>((resolve) => {\n const onKey = (_ch: string, key: { name?: string; sequence?: string }) => {\n if (key.name === 'up' || key.sequence === '\\x1b[A') {\n cursor = (cursor - 1 + options.length) % options.length;\n render();\n } else if (key.name === 'down' || key.sequence === '\\x1b[B') {\n cursor = (cursor + 1) % options.length;\n render();\n } else if (key.name === 'space') {\n if (selected.has(cursor)) {\n selected.delete(cursor);\n } else {\n selected.add(cursor);\n }\n render();\n } else if (key.name === 'return' || key.sequence === '\\r') {\n process.stdin.removeListener('keypress', onKey);\n process.stdout.write('\\n');\n const result = Array.from(selected)\n .sort((a, b) => a - b)\n .map((i) => options[i].value);\n resolve(result);\n }\n };\n\n process.stdin.on('keypress', onKey);\n if (process.stdin.isTTY) {\n process.stdin.resume();\n }\n });\n } finally {\n if (process.stdin.isTTY) process.stdin.setRawMode(wasRaw ?? false);\n }\n}","/**\n * `open-party login` — standalone Tailscale login command.\n *\n * Allows users to login to Tailscale after skipping it during setup,\n * or to re-authenticate if their session expired.\n */\n\nimport {\n getTailscaleInstallationStatus,\n} from '../infra/tailscale.js';\nimport { interactiveLogin, authKeyLogin } from './tailscale-login.js';\nimport { red, green, yellow, cyan, bold, select } from './tty-utils.js';\n\nexport async function loginCommand(): Promise<void> {\n const status = getTailscaleInstallationStatus();\n\n if (status.state === 'connected') {\n console.log(`${green('✅ Tailscale is already connected!')}`);\n console.log(` IP: ${status.tailscale_ip} Hostname: ${status.hostname}`);\n return;\n }\n\n if (status.state === 'not_installed') {\n console.log(`${red('❌ Tailscale is not installed.')}`);\n console.log(' Install it first: https://tailscale.com/download');\n console.log(` Then run: ${cyan('open-party login')}`);\n return;\n }\n\n // not_connected\n console.log(`${yellow('🔒 Tailscale is installed but not connected.')}\\n`);\n\n type LoginChoice = 'interactive' | 'authkey';\n const options: { label: string; value: LoginChoice; hint?: string }[] = [\n { label: 'Interactive login', value: 'interactive', hint: 'opens browser to authenticate' },\n { label: 'Auth key', value: 'authkey', hint: 'from network creator' },\n ];\n\n const choice = await select(options, { message: 'Choose a login method:' });\n\n if (choice === 'interactive') {\n await interactiveLogin(status.binary);\n } else {\n await authKeyLogin(status.binary);\n }\n}","/**\n * `open-party logout` — log out of the current Tailscale network.\n *\n * Disconnects from Tailscale and invalidates the node key,\n * requiring re-authentication to reconnect.\n */\n\nimport { getTailscaleInstallationStatus, logoutTailscale } from '../infra/tailscale.js';\nimport { select, green, red, yellow } from './tty-utils.js';\n\nexport async function logoutCommand(): Promise<void> {\n const status = getTailscaleInstallationStatus();\n\n if (status.state === 'not_installed') {\n console.log(red('❌ Tailscale is not installed.'));\n return;\n }\n\n if (status.state === 'not_connected') {\n console.log(yellow('⚠️ Tailscale is not connected — nothing to log out from.'));\n return;\n }\n\n // connected — confirm before logging out\n type LogoutChoice = 'logout' | 'cancel';\n const choice = await select<LogoutChoice>(\n [\n { label: 'Log out (remove credentials)', value: 'logout', hint: 'need to re-authenticate next time' },\n { label: 'Cancel', value: 'cancel' },\n ],\n { message: 'Are you sure you want to log out?' },\n );\n\n if (choice === 'cancel') return;\n\n console.log('Logging out of Tailscale...');\n const result = logoutTailscale();\n\n if (result.success) {\n console.log(green('✅ Logged out successfully.'));\n console.log(' To reconnect, run: open-party login');\n } else {\n console.log(red('❌ Logout failed:'), result.output);\n }\n}","/**\n * Shared utilities for CLI server lifecycle management.\n *\n * Provides PID file management, process detection, health checks,\n * server spawning, and argument parsing for the open-party CLI.\n */\n\nimport { spawn, execSync } from 'node:child_process';\nimport { existsSync, readFileSync, writeFileSync, unlinkSync, mkdirSync, openSync } from 'node:fs';\nimport { join, dirname, resolve } from 'node:path';\nimport { homedir } from 'node:os';\nimport { fileURLToPath } from 'node:url';\n\n// ---------------------------------------------------------------------------\n// Paths\n// ---------------------------------------------------------------------------\n\nconst __dirname = dirname(fileURLToPath(import.meta.url));\n\nexport function pidFilePath(): string {\n const pluginData = process.env.CLAUDE_PLUGIN_DATA || '';\n if (pluginData) return join(pluginData, 'server.pid');\n return join(homedir(), '.open-party', 'server.pid');\n}\n\nexport function logFilePath(): string {\n const pluginData = process.env.CLAUDE_PLUGIN_DATA || '';\n if (pluginData) return join(pluginData, 'server.log');\n return join(homedir(), '.open-party', 'server.log');\n}\n\nfunction serverScriptPath(): string {\n // When bundled, __dirname is dist/cli/ — party-server.js is at dist/party-server.js\n return resolve(__dirname, '..', 'party-server.js');\n}\n\n// ---------------------------------------------------------------------------\n// PID file management\n// ---------------------------------------------------------------------------\n\nexport function readPid(): number | null {\n const path = pidFilePath();\n if (!existsSync(path)) return null;\n try {\n return parseInt(readFileSync(path, 'utf-8').trim(), 10);\n } catch {\n return null;\n }\n}\n\nexport function writePid(pid: number): void {\n const path = pidFilePath();\n const dir = dirname(path);\n if (!existsSync(dir)) mkdirSync(dir, { recursive: true });\n writeFileSync(path, String(pid));\n}\n\nexport function removePidFile(): void {\n try { unlinkSync(pidFilePath()); } catch { /* ignore */ }\n}\n\n// ---------------------------------------------------------------------------\n// Process detection\n// ---------------------------------------------------------------------------\n\nexport function isProcessRunning(pid: number): boolean {\n if (process.platform === 'win32') {\n try {\n const output = execSync(`tasklist /FI \"PID eq ${pid}\" /NH`, {\n encoding: 'utf-8',\n windowsHide: true,\n stdio: ['pipe', 'pipe', 'pipe'],\n });\n return output.includes(String(pid));\n } catch {\n return false;\n }\n }\n try {\n process.kill(pid, 0);\n return true;\n } catch {\n return false;\n }\n}\n\n// ---------------------------------------------------------------------------\n// Health checks\n// ---------------------------------------------------------------------------\n\nexport function resolvePort(): number {\n return parseInt(process.env.PARTY_PORT || '8000', 10);\n}\n\nasync function fetchJson(url: string, timeoutMs: number = 2000): Promise<Record<string, unknown> | null> {\n try {\n const controller = new AbortController();\n const timer = setTimeout(() => controller.abort(), timeoutMs);\n const resp = await fetch(url, { signal: controller.signal });\n clearTimeout(timer);\n if (!resp.ok) return null;\n return (await resp.json()) as Record<string, unknown>;\n } catch {\n return null;\n }\n}\n\nexport async function isServerHealthy(port?: number): Promise<boolean> {\n const p = port ?? resolvePort();\n const data = await fetchJson(`http://127.0.0.1:${p}/proxy/health`);\n return data !== null && data.status === 'ok';\n}\n\nexport async function getServerHealth(port?: number): Promise<Record<string, unknown> | null> {\n const p = port ?? resolvePort();\n return fetchJson(`http://127.0.0.1:${p}/proxy/health`);\n}\n\nexport async function getServerOverview(port?: number): Promise<Record<string, unknown> | null> {\n const p = port ?? resolvePort();\n return fetchJson(`http://127.0.0.1:${p}/dashboard/api/overview`, 3000);\n}\n\n// ---------------------------------------------------------------------------\n// Server spawning\n// ---------------------------------------------------------------------------\n\nexport async function spawnServerInBackground(port: number): Promise<{ pid: number; ok: boolean }> {\n const script = serverScriptPath();\n if (!existsSync(script)) {\n console.error(`Server script not found: ${script}`);\n return { pid: 0, ok: false };\n }\n\n // Redirect stdout/stderr to log file\n const logPath = logFilePath();\n mkdirSync(dirname(logPath), { recursive: true });\n const logFd = openSync(logPath, 'a');\n\n const env = { ...process.env, PARTY_PORT: String(port) };\n const proc = spawn(process.execPath, [script], {\n stdio: ['ignore', logFd, logFd],\n detached: true,\n windowsHide: true,\n env,\n });\n proc.unref();\n\n const pid = proc.pid!;\n writePid(pid);\n\n proc.on('error', (err) => {\n console.error(`Failed to start server: ${err.message}`);\n });\n\n return { pid, ok: true };\n}\n\nexport async function waitForServerReady(port: number, timeoutMs: number = 10_000): Promise<boolean> {\n const deadline = Date.now() + timeoutMs;\n while (Date.now() < deadline) {\n if (await isServerHealthy(port)) return true;\n\n // Check if the spawned process is still alive (only if we have a PID)\n const pid = readPid();\n if (pid !== null && !isProcessRunning(pid)) {\n return false; // Process crashed\n }\n\n await sleep(500);\n }\n return false;\n}\n\n// ---------------------------------------------------------------------------\n// Process killing\n// ---------------------------------------------------------------------------\n\nexport function killServer(pid: number): void {\n try {\n if (process.platform === 'win32') {\n execSync(`taskkill /F /T /PID ${pid}`, { stdio: 'ignore', windowsHide: true });\n } else {\n process.kill(pid, 'SIGTERM');\n }\n } catch { /* process already gone */ }\n}\n\n// ---------------------------------------------------------------------------\n// Argument parsing\n// ---------------------------------------------------------------------------\n\nexport interface StartOptions {\n daemon: boolean;\n port: number | null; // null = use env/default\n}\n\nexport function parseStartArgs(args: string[]): StartOptions {\n let daemon = false;\n let port: number | null = null;\n\n for (let i = 0; i < args.length; i++) {\n if (args[i] === '-d' || args[i] === '--daemon') {\n daemon = true;\n } else if (args[i] === '-p' || args[i] === '--port') {\n const val = args[++i];\n if (val) port = parseInt(val, 10);\n } else if (args[i].startsWith('--port=')) {\n port = parseInt(args[i].split('=')[1], 10);\n }\n }\n\n return { daemon, port };\n}\n\n// ---------------------------------------------------------------------------\n// Helpers\n// ---------------------------------------------------------------------------\n\nfunction sleep(ms: number): Promise<void> {\n return new Promise((resolve) => setTimeout(resolve, ms));\n}","/**\n * Start the Party Server.\n *\n * Supports foreground mode (default) and daemon mode (--daemon / -d).\n * Use --port / -p to override PARTY_PORT for a single invocation.\n */\n\nimport {\n parseStartArgs,\n resolvePort,\n isServerHealthy,\n readPid,\n isProcessRunning,\n removePidFile,\n spawnServerInBackground,\n waitForServerReady,\n logFilePath,\n writePid,\n} from './server-utils.js';\n\nexport async function startServer(args: string[] = []): Promise<void> {\n const opts = parseStartArgs(args);\n const port = opts.port ?? resolvePort();\n\n if (opts.daemon) {\n await startDaemon(port);\n } else {\n await startForeground();\n }\n}\n\n// ---------------------------------------------------------------------------\n// Foreground mode — server runs in current process\n// ---------------------------------------------------------------------------\n\nasync function startForeground(): Promise<void> {\n // Write PID file so `stop` and `status` can find this process\n writePid(process.pid);\n\n // Clean up PID file on any exit\n process.on('exit', () => {\n removePidFile();\n });\n\n // Dynamic import — the server module self-starts on import\n await import('../server/index.js');\n}\n\n// ---------------------------------------------------------------------------\n// Daemon mode — server runs in a detached background process\n// ---------------------------------------------------------------------------\n\nasync function startDaemon(port: number): Promise<void> {\n // Already running?\n if (await isServerHealthy(port)) {\n const pid = readPid();\n console.log(`Party Server is already running (PID ${pid ?? 'unknown'}, port ${port}).`);\n process.exit(0);\n }\n\n // Clean up stale PID file\n const existingPid = readPid();\n if (existingPid !== null && !isProcessRunning(existingPid)) {\n removePidFile();\n }\n\n // Spawn\n const { pid, ok } = await spawnServerInBackground(port);\n if (!ok) {\n process.exit(1);\n }\n\n // Wait for ready\n console.log(`Starting Party Server in background (PID ${pid})...`);\n const ready = await waitForServerReady(port);\n\n if (ready) {\n console.log(`Party Server is running on port ${port}.`);\n console.log(` Dashboard: http://127.0.0.1:${port}/dashboard`);\n console.log(` Logs: ${logFilePath()}`);\n console.log(` Use 'open-party stop' to stop the server.`);\n } else {\n console.error('Party Server failed to start within timeout.');\n console.error(`Check logs: ${logFilePath()}`);\n process.exit(1);\n }\n}","/**\n * Stop the Party Server.\n *\n * Reads the PID file, verifies the process is alive, kills it,\n * and cleans up the PID file.\n */\n\nimport { readPid, isProcessRunning, killServer, removePidFile, isServerHealthy, resolvePort } from './server-utils.js';\n\nexport async function stopServer(): Promise<void> {\n const pid = readPid();\n\n if (pid === null) {\n // No PID file — check if a server is still responding\n const port = resolvePort();\n const healthy = await isServerHealthy(port);\n if (healthy) {\n console.log(`No PID file found, but a server is responding on port ${port}.`);\n console.log('It may have been started manually. Kill it by port or process name.');\n } else {\n console.log('Party Server is not running (no PID file found).');\n }\n return;\n }\n\n if (!isProcessRunning(pid)) {\n // Stale PID file — process is dead\n console.log(`Stale PID file found (PID ${pid} is not running). Cleaning up.`);\n removePidFile();\n return;\n }\n\n // Process is alive — kill it\n console.log(`Stopping Party Server (PID ${pid})...`);\n killServer(pid);\n removePidFile();\n\n // Verify it's actually down\n const port = resolvePort();\n const stillUp = await isServerHealthy(port);\n if (stillUp) {\n console.warn(`Process ${pid} was killed, but port ${port} is still responding.`);\n console.warn('Another process may be using this port.');\n } else {\n console.log('Party Server stopped.');\n }\n}","/**\n * Show the status of the Party Server.\n *\n * Displays PID, port, uptime, agent count, and Tailscale info\n * when the server is running. Handles four states:\n * 1. Running & healthy\n * 2. Process alive but not yet healthy (starting up)\n * 3. Stale PID file (process dead)\n * 4. Not running (no PID file)\n */\n\nimport {\n readPid,\n isProcessRunning,\n isServerHealthy,\n getServerHealth,\n getServerOverview,\n resolvePort,\n} from './server-utils.js';\n\nexport async function statusCommand(): Promise<void> {\n const port = resolvePort();\n const pid = readPid();\n\n let processAlive = false;\n if (pid !== null) {\n processAlive = isProcessRunning(pid);\n }\n\n const healthy = await isServerHealthy(port);\n\n if (healthy) {\n const health = await getServerHealth(port);\n const overview = await getServerOverview(port);\n\n console.log('Party Server is running.');\n console.log(` PID: ${pid ?? 'unknown (no PID file)'}`);\n console.log(` Port: ${port}`);\n console.log(` Tailscale IP: ${(health as any)?.tailscale_ip ?? 'N/A'}`);\n console.log(` Hostname: ${(health as any)?.hostname ?? 'N/A'}`);\n\n if (overview) {\n const server = overview.server as Record<string, unknown> | undefined;\n const agents = overview.agents as Record<string, unknown> | undefined;\n if (server?.uptime_seconds != null) {\n const uptime = server.uptime_seconds as number;\n const mins = Math.floor(uptime / 60);\n const secs = Math.floor(uptime % 60);\n console.log(` Uptime: ${mins}m ${secs}s`);\n }\n console.log(` Local agents: ${(agents?.local_count as number) ?? 'N/A'}`);\n console.log(` Remote agents: ${(agents?.remote_count as number) ?? 'N/A'}`);\n } else {\n console.log(` Local agents: ${(health as any)?.agent_count ?? 'N/A'}`);\n }\n\n console.log(` Dashboard: http://127.0.0.1:${port}/dashboard`);\n } else if (processAlive && pid !== null) {\n // PID file has a live process but health check fails\n console.log('Party Server process exists but is not responding on health endpoint.');\n console.log(` PID: ${pid}`);\n console.log(' The server may be starting up or has crashed.');\n console.log(` Logs: ~/.open-party/server.log`);\n } else if (pid !== null) {\n console.log('Party Server is NOT running (stale PID file).');\n console.log(` PID file references PID ${pid}, which is not a live process.`);\n console.log(' Use: open-party start to start the server.');\n } else {\n console.log('Party Server is NOT running.');\n console.log(' No PID file found.');\n console.log(' Use: open-party start to start the server.');\n }\n}","/**\n * `open-party --version` / `open-party -v` — show version number.\n */\n\nimport { readFileSync } from 'node:fs';\nimport { resolve, dirname } from 'node:path';\nimport { fileURLToPath } from 'node:url';\n\nexport function showVersion(): void {\n // tsup bundles to dist/cli/index.js, package.json is at the project root: ../../package.json\n // When installed via npm: <pkg-root>/dist/cli/index.js → <pkg-root>/package.json\n const pkgPath = resolve(dirname(fileURLToPath(import.meta.url)), '..', '..', 'package.json');\n const pkg = JSON.parse(readFileSync(pkgPath, 'utf-8'));\n console.log(`open-party v${pkg.version}`);\n}","/**\n * `open-party agents` — list connected agents.\n */\n\nimport { isServerHealthy, resolvePort, getServerOverview } from './server-utils.js';\n\nexport async function agentsCommand(): Promise<void> {\n const port = resolvePort();\n\n if (!(await isServerHealthy(port))) {\n console.log('Party Server is not running.');\n console.log(\" Use 'open-party start' to start it.\");\n return;\n }\n\n const overview = await getServerOverview(port);\n if (!overview) {\n console.log('Failed to get server overview.');\n return;\n }\n\n const agents = overview.agents as Record<string, unknown> | undefined;\n if (!agents) {\n console.log('No agent data available.');\n return;\n }\n\n const localAgents = (agents.local_agents as Record<string, unknown>[]) ?? [];\n const remoteAgents = (agents.remote_agents as Record<string, unknown>[]) ?? [];\n\n const localCount = (agents.local_count as number) ?? localAgents.length;\n const remoteCount = (agents.remote_count as number) ?? remoteAgents.length;\n\n // Local agents\n if (localCount === 0) {\n console.log('Local agents: (none)');\n } else {\n console.log(`Local agents (${localCount}):`);\n for (const agent of localAgents) {\n const id = (agent.agent_id as string) ?? '?';\n const name = (agent.display_name as string) ?? id;\n const ago = formatTimeAgo(agent.last_heartbeat as number | undefined);\n console.log(` ${id.padEnd(20)} ${name.padEnd(16)} ${ago}`);\n }\n }\n\n // Remote agents\n if (remoteCount > 0) {\n console.log(`\\nRemote agents (${remoteCount}):`);\n for (const agent of remoteAgents) {\n const id = (agent.agent_id as string) ?? '?';\n const name = (agent.display_name as string) ?? id;\n const via = (agent.source_peer_ip as string) ?? '?';\n const ago = formatTimeAgo(agent.last_heartbeat as number | undefined);\n console.log(` ${id.padEnd(20)} ${name.padEnd(16)} (via ${via}) ${ago}`);\n }\n }\n\n if (localCount === 0 && remoteCount === 0) {\n console.log('\\nNo agents connected yet.');\n }\n}\n\nfunction formatTimeAgo(timestamp: number | undefined): string {\n if (!timestamp) return '—';\n const diff = Date.now() / 1000 - timestamp / 1000;\n if (diff < 60) return 'just now';\n if (diff < 3600) return `${Math.floor(diff / 60)} min ago`;\n if (diff < 86400) return `${Math.floor(diff / 3600)} hr ago`;\n return `${Math.floor(diff / 86400)}d ago`;\n}","/**\n * `open-party peers` — list discovered peer nodes.\n */\n\nimport { isServerHealthy, resolvePort, getServerOverview } from './server-utils.js';\n\ntype PeerStateView = {\n ip: string;\n status: string;\n consecutiveFailures: number;\n lastProbeAt: number;\n backoffUntil: number | null;\n};\n\ntype RemoteAgentEntry = {\n source_peer_ip: string;\n};\n\nexport async function peersCommand(): Promise<void> {\n const port = resolvePort();\n\n if (!(await isServerHealthy(port))) {\n console.log('Party Server is not running.');\n console.log(\" Use 'open-party start' to start it.\");\n return;\n }\n\n const overview = await getServerOverview(port);\n if (!overview) {\n console.log('Failed to get server overview.');\n return;\n }\n\n const peers = overview.peers as Record<string, unknown> | undefined;\n if (!peers) {\n console.log('No peer data available.');\n return;\n }\n\n const details = (peers.details as PeerStateView[]) ?? [];\n const remoteAgents = (((overview.agents as Record<string, unknown> | undefined)?.remote_agents) as RemoteAgentEntry[]) ?? [];\n\n // Build map: peer_ip → agent_count\n const peerAgentCounts = new Map<string, number>();\n for (const agent of remoteAgents) {\n const ip = agent.source_peer_ip;\n peerAgentCounts.set(ip, (peerAgentCounts.get(ip) ?? 0) + 1);\n }\n\n const total = (peers.total as number) ?? details.length;\n\n if (details.length === 0) {\n console.log('No peers discovered yet.');\n return;\n }\n\n console.log(`Peers (${total}):\\n`);\n\n for (const peer of details) {\n const agentCount = peerAgentCounts.get(peer.ip);\n const agentStr = agentCount != null ? String(agentCount) : '—';\n const statusStr = formatStatus(peer.status);\n console.log(` ${peer.ip.padEnd(18)} ${statusStr.padEnd(16)} ${agentStr} agents`);\n }\n}\n\nfunction formatStatus(status: string): string {\n // Make status more readable\n const map: Record<string, string> = {\n PARTY_SERVER: 'Online',\n DEGRADED: 'Degraded',\n SUSPECT: 'Suspect',\n DOWN: 'Down',\n UNKNOWN: 'Unknown',\n MAYBE: 'Probing',\n NOT_SERVER: 'Not a server',\n };\n return map[status] ?? status;\n}","/**\n * `open-party install` — install Tailscale.\n *\n * Checks if Tailscale is already installed; if not, runs the\n * platform-specific installer.\n */\n\nimport { getTailscaleInstallationStatus, getInstallInstructions } from '../infra/tailscale.js';\nimport { installTailscale } from './tailscale-installer.js';\nimport { cyan, green, red, bold } from './tty-utils.js';\n\nexport async function installCommand(): Promise<void> {\n const status = getTailscaleInstallationStatus();\n\n if (status.state === 'connected' || status.state === 'not_connected') {\n console.log(green('✅ Tailscale is already installed!'), `Binary: ${status.binary}`);\n if (status.state === 'connected') {\n console.log(` IP: ${status.tailscale_ip} Hostname: ${status.hostname}`);\n }\n return;\n }\n\n // not_installed\n const platform = process.platform;\n const info = getInstallInstructions(platform);\n\n console.log(bold(cyan('Installing Tailscale...\\n')));\n console.log(` Platform: ${info.os}`);\n\n const result = await installTailscale(platform);\n\n if (result.success) {\n console.log(green('\\n✅ Tailscale installed successfully!'));\n console.log(' Run: open-party login');\n } else {\n console.log(red('\\n❌ Installation failed:'), result.output);\n console.log(' Install manually: https://tailscale.com/download');\n }\n}","/**\n * Open Party CLI — decentralized Agent communication network.\n *\n * Usage:\n * open-party start [-d] [-p <port>] Start the Party Server (default)\n * open-party stop Stop the Party Server\n * open-party status Show server status\n * open-party setup Interactive setup wizard\n * open-party login Login to Tailscale network\n * open-party logout Log out of Tailscale network\n * open-party install Install Tailscale\n * open-party agents List connected agents\n * open-party peers List discovered peer nodes\n * open-party help Show help\n */\n\nimport { setupCommand } from './setup.js';\nimport { loginCommand } from './login.js';\nimport { logoutCommand } from './logout.js';\nimport { startServer } from './start-server.js';\nimport { stopServer } from './stop-server.js';\nimport { statusCommand } from './status.js';\nimport { showVersion } from './version.js';\nimport { agentsCommand } from './agents.js';\nimport { peersCommand } from './peers.js';\nimport { installCommand } from './install.js';\n\nfunction showHelp(): void {\n console.log(`Usage: open-party <command> [options]\n\nCommands:\n start Start the Party Server (default when no command given)\n stop Stop the Party Server\n status Show server status\n setup Interactive setup wizard (Tailscale + agent plugins)\n login Login to Tailscale network\n logout Log out of Tailscale network\n install Install Tailscale\n agents List connected agents\n peers List discovered peer nodes\n help Show this help message\n\nOptions for 'start':\n -d, --daemon Run in background (daemon mode)\n -p, --port <port> Override port (default: 8000, env: PARTY_PORT)\n\nGlobal options:\n -v, --version Show version number\n\nExamples:\n open-party Start server in foreground\n open-party start Start server in foreground\n open-party start -d Start server in background\n open-party start -d -p 9000 Start server in background on port 9000\n open-party stop Stop the server\n open-party status Check if the server is running\n open-party login Login to Tailscale\n open-party logout Log out of Tailscale\n open-party install Install Tailscale\n open-party agents List connected agents\n open-party peers List discovered peer nodes`);\n}\n\nconst args = process.argv.slice(2);\nconst command = args[0] ?? 'start';\nconst commandArgs = args.slice(1);\n\nasync function main(): Promise<void> {\n // Handle global flags before command dispatch\n if (command === '--version' || command === '-v') {\n showVersion();\n process.exit(0);\n }\n\n switch (command) {\n case 'setup':\n await setupCommand();\n break;\n case 'login':\n await loginCommand();\n break;\n case 'logout':\n await logoutCommand();\n break;\n case 'install':\n await installCommand();\n break;\n case 'start':\n await startServer(commandArgs);\n break;\n case 'stop':\n await stopServer();\n break;\n case 'status':\n await statusCommand();\n break;\n case 'agents':\n await agentsCommand();\n break;\n case 'peers':\n await peersCommand();\n break;\n case 'help':\n case '--help':\n case '-h':\n showHelp();\n break;\n default:\n console.log(`Unknown command: ${command}\\n`);\n showHelp();\n process.exit(1);\n }\n}\n\nmain().catch((e) => {\n console.error('Fatal error:', e);\n process.exit(1);\n});"],"mappings":"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;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA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","names":["raw","resolve","match","args","resolve","match","cacheKey","raw","args","init_constants","init_constants","args","app","init_router","init_router","init_router","init_router","init_router","Node","init_node","_Node","init_router","init_node","Node","init_router","Hono","Request","init_dist","resolve","sleep","resolve","Hono","registry","discovery","messageQueue","Hono","Hono","process","installTailscale","mkdirSync","writeFileSync","unlinkSync","join","dirname","homedir","pidFilePath","init_dist","Hono","existsSync","join","execSync","existsSync","join","homedir","manifest","join","homedir","existsSync","join","homedir","command","args","existsSync","spawn","resolve","spawn","resolve","spawn","resolve","dirname","fileURLToPath","__dirname","spawn","execSync","existsSync","readFileSync","writeFileSync","mkdirSync","join","dirname","resolve","homedir","args","args","pid","port","readFileSync","resolve","dirname","fileURLToPath","main"]}