import type { AddressInfo } from "node:net"; import fs from "node:fs/promises"; import path from "node:path"; import { afterAll, beforeAll, describe, expect, it, vi } from "vitest"; const MEDIA_DIR = path.join(process.cwd(), "tmp-media-test"); const cleanOldMedia = vi.fn().mockResolvedValue(undefined); vi.mock("./store.js", async (importOriginal) => { const actual = await importOriginal(); return { ...actual, getMediaDir: () => MEDIA_DIR, cleanOldMedia, }; }); const { startMediaServer } = await import("./server.js"); const { MEDIA_MAX_BYTES } = await import("./store.js"); const waitForFileRemoval = async (file: string, timeoutMs = 200) => { const start = Date.now(); while (Date.now() - start < timeoutMs) { try { await fs.stat(file); } catch { return; } await new Promise((resolve) => setTimeout(resolve, 5)); } throw new Error(`timed out waiting for ${file} removal`); }; describe("media server", () => { beforeAll(async () => { await fs.rm(MEDIA_DIR, { recursive: true, force: true }); await fs.mkdir(MEDIA_DIR, { recursive: true }); }); afterAll(async () => { await fs.rm(MEDIA_DIR, { recursive: true, force: true }); }); it("serves media and cleans up after send", async () => { const file = path.join(MEDIA_DIR, "file1"); await fs.writeFile(file, "hello"); const server = await startMediaServer(0, 5_000); const port = (server.address() as AddressInfo).port; const res = await fetch(`http://localhost:${port}/media/file1`); expect(res.status).toBe(200); expect(await res.text()).toBe("hello"); await waitForFileRemoval(file); await new Promise((r) => server.close(r)); }); it("expires old media", async () => { const file = path.join(MEDIA_DIR, "old"); await fs.writeFile(file, "stale"); const past = Date.now() - 10_000; await fs.utimes(file, past / 1000, past / 1000); const server = await startMediaServer(0, 1_000); const port = (server.address() as AddressInfo).port; const res = await fetch(`http://localhost:${port}/media/old`); expect(res.status).toBe(410); await expect(fs.stat(file)).rejects.toThrow(); await new Promise((r) => server.close(r)); }); it("blocks path traversal attempts", async () => { const server = await startMediaServer(0, 5_000); const port = (server.address() as AddressInfo).port; // URL-encoded "../" to bypass client-side path normalization const res = await fetch(`http://localhost:${port}/media/%2e%2e%2fpackage.json`); expect(res.status).toBe(400); expect(await res.text()).toBe("invalid path"); await new Promise((r) => server.close(r)); }); it("blocks symlink escaping outside media dir", async () => { const target = path.join(process.cwd(), "package.json"); // outside MEDIA_DIR const link = path.join(MEDIA_DIR, "link-out"); await fs.symlink(target, link); const server = await startMediaServer(0, 5_000); const port = (server.address() as AddressInfo).port; const res = await fetch(`http://localhost:${port}/media/link-out`); expect(res.status).toBe(400); expect(await res.text()).toBe("invalid path"); await new Promise((r) => server.close(r)); }); it("rejects invalid media ids", async () => { const file = path.join(MEDIA_DIR, "file2"); await fs.writeFile(file, "hello"); const server = await startMediaServer(0, 5_000); const port = (server.address() as AddressInfo).port; const res = await fetch(`http://localhost:${port}/media/invalid%20id`); expect(res.status).toBe(400); expect(await res.text()).toBe("invalid path"); await new Promise((r) => server.close(r)); }); it("rejects oversized media files", async () => { const file = path.join(MEDIA_DIR, "big"); await fs.writeFile(file, ""); await fs.truncate(file, MEDIA_MAX_BYTES + 1); const server = await startMediaServer(0, 5_000); const port = (server.address() as AddressInfo).port; const res = await fetch(`http://localhost:${port}/media/big`); expect(res.status).toBe(413); expect(await res.text()).toBe("too large"); await new Promise((r) => server.close(r)); }); });