Files
openclaw/src/media/mime.test.ts
2026-03-28 07:31:40 +00:00

194 lines
6.3 KiB
TypeScript

import JSZip from "jszip";
import { describe, expect, it } from "vitest";
import { mediaKindFromMime } from "./constants.js";
import {
detectMime,
extensionForMime,
imageMimeFromFormat,
isAudioFileName,
kindFromMime,
normalizeMimeType,
} from "./mime.js";
async function makeOoxmlZip(opts: { mainMime: string; partPath: string }): Promise<Buffer> {
const zip = new JSZip();
zip.file(
"[Content_Types].xml",
`<Types><Override PartName="${opts.partPath}" ContentType="${opts.mainMime}.main+xml"/></Types>`,
);
zip.file(opts.partPath.slice(1), "<xml/>");
return await zip.generateAsync({ type: "nodebuffer" });
}
describe("mime detection", () => {
async function expectDetectedMime(params: {
input: Parameters<typeof detectMime>[0];
expected: string;
}) {
expect(await detectMime(params.input)).toBe(params.expected);
}
it.each([
{ format: "jpg", expected: "image/jpeg" },
{ format: "jpeg", expected: "image/jpeg" },
{ format: "png", expected: "image/png" },
{ format: "webp", expected: "image/webp" },
{ format: "gif", expected: "image/gif" },
{ format: "unknown", expected: undefined },
])("maps $format image format", ({ format, expected }) => {
expect(imageMimeFromFormat(format)).toBe(expected);
});
it.each([
{
name: "detects docx from buffer",
mainMime: "application/vnd.openxmlformats-officedocument.wordprocessingml.document",
partPath: "/word/document.xml",
expected: "application/vnd.openxmlformats-officedocument.wordprocessingml.document",
},
{
name: "detects pptx from buffer",
mainMime: "application/vnd.openxmlformats-officedocument.presentationml.presentation",
partPath: "/ppt/presentation.xml",
expected: "application/vnd.openxmlformats-officedocument.presentationml.presentation",
},
] as const)("$name", async ({ mainMime, partPath, expected }) => {
await expectDetectedMime({
input: {
buffer: await makeOoxmlZip({ mainMime, partPath }),
filePath: "/tmp/file.bin",
},
expected,
});
});
it.each([
{
name: "prefers extension mapping over generic zip",
input: async () => {
const zip = new JSZip();
zip.file("hello.txt", "hi");
return {
buffer: await zip.generateAsync({ type: "nodebuffer" }),
filePath: "/tmp/file.xlsx",
};
},
expected: "application/vnd.openxmlformats-officedocument.spreadsheetml.sheet",
},
{
name: "uses extension mapping for JavaScript assets",
input: async () => ({
filePath: "/tmp/a2ui.bundle.js",
}),
expected: "text/javascript",
},
] as const)("$name", async ({ input, expected }) => {
await expectDetectedMime({
input: await input(),
expected,
});
});
});
describe("extensionForMime", () => {
function expectMimeExtensionCase(
mime: Parameters<typeof extensionForMime>[0],
expected: ReturnType<typeof extensionForMime>,
) {
expect(extensionForMime(mime)).toBe(expected);
}
it.each([
{ mime: "image/jpeg", expected: ".jpg" },
{ mime: "image/png", expected: ".png" },
{ mime: "image/webp", expected: ".webp" },
{ mime: "image/gif", expected: ".gif" },
{ mime: "image/heic", expected: ".heic" },
{ mime: "audio/mpeg", expected: ".mp3" },
{ mime: "audio/ogg", expected: ".ogg" },
{ mime: "audio/x-m4a", expected: ".m4a" },
{ mime: "audio/mp4", expected: ".m4a" },
{ mime: "video/mp4", expected: ".mp4" },
{ mime: "video/quicktime", expected: ".mov" },
{ mime: "application/pdf", expected: ".pdf" },
{ mime: "text/plain", expected: ".txt" },
{ mime: "text/markdown", expected: ".md" },
{ mime: "IMAGE/JPEG", expected: ".jpg" },
{ mime: "Audio/X-M4A", expected: ".m4a" },
{ mime: "Video/QuickTime", expected: ".mov" },
{ mime: "video/unknown", expected: undefined },
{ mime: "application/x-custom", expected: undefined },
{ mime: null, expected: undefined },
{ mime: undefined, expected: undefined },
] as const)("maps $mime to extension", ({ mime, expected }) => {
expectMimeExtensionCase(mime, expected);
});
});
describe("isAudioFileName", () => {
function expectAudioFileNameCase(fileName: string, expected: boolean) {
expect(isAudioFileName(fileName)).toBe(expected);
}
it.each([
{ fileName: "voice.mp3", expected: true },
{ fileName: "voice.caf", expected: true },
{ fileName: "voice.bin", expected: false },
] as const)("matches audio extension for $fileName", ({ fileName, expected }) => {
expectAudioFileNameCase(fileName, expected);
});
});
describe("normalizeMimeType", () => {
function expectNormalizedMimeCase(
input: Parameters<typeof normalizeMimeType>[0],
expected: ReturnType<typeof normalizeMimeType>,
) {
expect(normalizeMimeType(input)).toBe(expected);
}
it.each([
{ input: "Audio/MP4; codecs=mp4a.40.2", expected: "audio/mp4" },
{ input: " ", expected: undefined },
{ input: null, expected: undefined },
{ input: undefined, expected: undefined },
] as const)("normalizes $input", ({ input, expected }) => {
expectNormalizedMimeCase(input, expected);
});
});
describe("mediaKindFromMime", () => {
function expectMediaKindCase(
mime: Parameters<typeof mediaKindFromMime>[0],
expected: ReturnType<typeof mediaKindFromMime>,
) {
expect(mediaKindFromMime(mime)).toBe(expected);
}
function expectMimeKindCase(
mime: Parameters<typeof kindFromMime>[0],
expected: ReturnType<typeof kindFromMime>,
) {
expect(kindFromMime(mime)).toBe(expected);
}
it.each([
{ mime: "text/plain", expected: "document" },
{ mime: "text/csv", expected: "document" },
{ mime: "text/html; charset=utf-8", expected: "document" },
{ mime: "model/gltf+json", expected: undefined },
{ mime: null, expected: undefined },
{ mime: undefined, expected: undefined },
] as const)("classifies $mime", ({ mime, expected }) => {
expectMediaKindCase(mime, expected);
});
it.each([
{ mime: " Audio/Ogg; codecs=opus ", expected: "audio" },
{ mime: undefined, expected: undefined },
{ mime: "model/gltf+json", expected: undefined },
] as const)("maps kindFromMime($mime) => $expected", ({ mime, expected }) => {
expectMimeKindCase(mime, expected);
});
});