From e046e851eb2324de828733d5c7ac10155b36e3a8 Mon Sep 17 00:00:00 2001 From: Garry Tan Date: Sat, 4 Apr 2026 21:19:09 -0700 Subject: [PATCH] fix(design): bind server to localhost and validate reload paths Cherry-pick PR #803 by @garagon. Adds hostname: '127.0.0.1' to Bun.serve() and validates /api/reload paths are within cwd() or tmpdir(). Closes C1+C2 from security audit #783. Co-Authored-By: garagon Co-Authored-By: Claude Opus 4.6 (1M context) --- design/src/serve.ts | 16 +++++++++++++++- 1 file changed, 15 insertions(+), 1 deletion(-) diff --git a/design/src/serve.ts b/design/src/serve.ts index 7d974905..93d33e75 100644 --- a/design/src/serve.ts +++ b/design/src/serve.ts @@ -33,19 +33,21 @@ */ import fs from "fs"; +import os from "os"; import path from "path"; import { spawn } from "child_process"; export interface ServeOptions { html: string; port?: number; + hostname?: string; // default '127.0.0.1' — localhost only timeout?: number; // seconds, default 600 (10 min) } type ServerState = "serving" | "regenerating" | "done"; export async function serve(options: ServeOptions): Promise { - const { html, port = 0, timeout = 600 } = options; + const { html, port = 0, hostname = '127.0.0.1', timeout = 600 } = options; // Validate HTML file exists if (!fs.existsSync(html)) { @@ -59,6 +61,7 @@ export async function serve(options: ServeOptions): Promise { const server = Bun.serve({ port, + hostname, fetch(req) { const url = new URL(req.url); @@ -182,6 +185,17 @@ export async function serve(options: ServeOptions): Promise { ); } + // Validate path is within cwd or temp directory + const resolved = path.resolve(newHtmlPath); + const safeDirs = [process.cwd(), os.tmpdir()]; + const isSafe = safeDirs.some(dir => resolved.startsWith(dir + path.sep) || resolved === dir); + if (!isSafe) { + return Response.json( + { error: `Path must be within working directory or temp` }, + { status: 403 } + ); + } + // Swap the HTML content htmlContent = fs.readFileSync(newHtmlPath, "utf-8"); state = "serving";