Files
openclaw/src/media/server.ts
Vincent Koc 01bf61fcfd fix(media): remove express from media host (#71436)
* fix(media): remove express from media host

* fix(media): harden media host responses

* fix(msteams): stage express runtime dependency

* fix(browser): align profile facade exports

* fix(msteams): keep setup entry narrow

* fix(types): satisfy extension setup gates

* fix(msteams): use generic setup config type
2026-04-25 01:39:42 -07:00

225 lines
6.2 KiB
TypeScript

import fs from "node:fs/promises";
import { createServer, type IncomingMessage, type Server, type ServerResponse } from "node:http";
import { danger } from "../globals.js";
import { defaultRuntime, type RuntimeEnv } from "../runtime.js";
import { detectMime } from "./mime.js";
import {
cleanOldMedia,
getMediaDir,
isSafeOpenError,
MEDIA_MAX_BYTES,
readFileWithinRoot,
} from "./server.runtime.js";
const DEFAULT_TTL_MS = 2 * 60 * 1000;
const MAX_MEDIA_ID_CHARS = 200;
const MEDIA_ID_PATTERN = /^[\p{L}\p{N}._-]+$/u;
const MAX_MEDIA_BYTES = MEDIA_MAX_BYTES;
const DEFAULT_MEDIA_CONTENT_TYPE = "application/octet-stream";
const ACTIVE_CONTENT_MIME_TYPES = new Set([
"application/xhtml+xml",
"application/xml",
"image/svg+xml",
"text/html",
"text/javascript",
"text/xml",
]);
const isValidMediaId = (id: string) => {
if (!id) {
return false;
}
if (id.length > MAX_MEDIA_ID_CHARS) {
return false;
}
if (id === "." || id === "..") {
return false;
}
return MEDIA_ID_PATTERN.test(id);
};
function sendText(res: ServerResponse, statusCode: number, body: string): void {
const data = Buffer.from(body);
res.statusCode = statusCode;
res.setHeader("Content-Type", "text/plain; charset=utf-8");
res.setHeader("Content-Length", String(data.byteLength));
res.end(data);
}
function resolveMediaId(req: IncomingMessage): {
routeMatched: boolean;
id?: string;
method?: string;
} {
if (req.method !== "GET" && req.method !== "HEAD") {
return { routeMatched: false };
}
const url = new URL(req.url ?? "/", "http://127.0.0.1");
const prefix = "/media/";
if (!url.pathname.startsWith(prefix)) {
return { routeMatched: false };
}
const encodedId = url.pathname.slice(prefix.length);
if (!encodedId || encodedId.includes("/")) {
return { routeMatched: false };
}
try {
return { routeMatched: true, id: decodeURIComponent(encodedId), method: req.method };
} catch {
return { routeMatched: true, id: "", method: req.method };
}
}
function isActiveContentMime(mime?: string): boolean {
const normalized = mime?.split(";")[0]?.trim().toLowerCase();
return normalized ? ACTIVE_CONTENT_MIME_TYPES.has(normalized) : false;
}
function sanitizeAttachmentFilename(id: string): string {
const name = id.replace(/["\\\r\n]/g, "_").trim();
return name || "media";
}
function setMediaHeaders(
res: ServerResponse,
params: { id: string; mime?: string; bytes: number },
): void {
const activeContent = isActiveContentMime(params.mime);
res.setHeader(
"Content-Type",
activeContent ? DEFAULT_MEDIA_CONTENT_TYPE : (params.mime ?? DEFAULT_MEDIA_CONTENT_TYPE),
);
res.setHeader("Content-Length", String(params.bytes));
if (activeContent) {
res.setHeader(
"Content-Disposition",
`attachment; filename="${sanitizeAttachmentFilename(params.id)}"`,
);
}
}
function scheduleMediaCleanup(realPath: string): void {
const cleanup = () => {
void fs.rm(realPath).catch(() => {});
};
if (process.env.VITEST || process.env.NODE_ENV === "test") {
queueMicrotask(cleanup);
return;
}
setTimeout(cleanup, 50);
}
function cleanupAfterGetResponse(res: ServerResponse, realPath: string): void {
let scheduled = false;
const scheduleOnce = () => {
if (scheduled) {
return;
}
scheduled = true;
scheduleMediaCleanup(realPath);
};
res.once("finish", scheduleOnce);
res.once("close", scheduleOnce);
res.once("error", scheduleOnce);
}
export function createMediaRequestHandler(ttlMs = DEFAULT_TTL_MS) {
const mediaDir = getMediaDir();
return (req: IncomingMessage, res: ServerResponse) => {
const route = resolveMediaId(req);
if (!route.routeMatched) {
sendText(res, 404, "not found");
return;
}
void (async () => {
res.setHeader("X-Content-Type-Options", "nosniff");
const id = route.id ?? "";
if (!isValidMediaId(id)) {
sendText(res, 400, "invalid path");
return;
}
try {
const {
buffer: data,
realPath,
stat,
} = await readFileWithinRoot({
rootDir: mediaDir,
relativePath: id,
maxBytes: MAX_MEDIA_BYTES,
});
if (Date.now() - stat.mtimeMs > ttlMs) {
await fs.rm(realPath).catch(() => {});
sendText(res, 410, "expired");
return;
}
const mime = await detectMime({ buffer: data, filePath: realPath });
setMediaHeaders(res, { id, mime, bytes: data.byteLength });
res.statusCode = 200;
if (route.method === "HEAD") {
res.end();
return;
}
cleanupAfterGetResponse(res, realPath);
if (req.aborted || res.destroyed || res.writableEnded) {
scheduleMediaCleanup(realPath);
return;
}
res.end(data);
} catch (err) {
if (isSafeOpenError(err)) {
if (err.code === "outside-workspace") {
sendText(res, 400, "file is outside workspace root");
return;
}
if (err.code === "invalid-path") {
sendText(res, 400, "invalid path");
return;
}
if (err.code === "not-found") {
sendText(res, 404, "not found");
return;
}
if (err.code === "too-large") {
sendText(res, 413, "too large");
return;
}
}
sendText(res, 404, "not found");
}
})().catch(() => {
if (!res.headersSent) {
sendText(res, 404, "not found");
} else {
res.destroy();
}
});
};
}
function startMediaCleanupInterval(ttlMs: number): void {
// periodic cleanup
setInterval(() => {
void cleanOldMedia(ttlMs, { recursive: false });
}, ttlMs).unref();
}
export async function startMediaServer(
port: number,
ttlMs = DEFAULT_TTL_MS,
runtime: RuntimeEnv = defaultRuntime,
): Promise<Server> {
const server = createServer(createMediaRequestHandler(ttlMs));
startMediaCleanupInterval(ttlMs);
return await new Promise((resolve, reject) => {
server.listen(port, "127.0.0.1");
server.once("listening", () => resolve(server));
server.once("error", (err) => {
runtime.error(danger(`Media server failed: ${String(err)}`));
reject(err);
});
});
}