import sharp from "sharp" ;
import { describe, expect, it } from "vitest" ;
import { sanitizeContentBlocksImages, sanitizeImageBlocks } from "./tool-images.js" ;
describe("tool image sanitizing" , () => {
const getImageBlock = (
blocks: Awaited<ReturnType<typeof sanitizeContentBlocksImages>>,
): (typeof blocks)[number] & { type: "image" ; data: string; mimeType?: string } => {
const image = blocks.find((block) => block.type === "image" );
if (!image || image.type !== "image" ) {
throw new Error("expected image block" );
}
return image;
};
const createWidePng = async () => {
const width = 2600 ;
const height = 400 ;
const raw = Buffer.alloc(width * height * 3 , 0 x7f);
return sharp(raw, {
raw: { width, height, channels: 3 },
})
.png({ compressionLevel: 9 })
.toBuffer();
};
it("shrinks oversized images to the configured byte limit" , async () => {
const maxBytes = 128 * 1024 ;
const width = 900 ;
const height = 900 ;
const raw = Buffer.alloc(width * height * 3 , 0 xff);
const bigPng = await sharp(raw, {
raw: { width, height, channels: 3 },
})
.png({ compressionLevel: 0 })
.toBuffer();
expect(bigPng.byteLength).toBeGreaterThan(maxBytes);
const blocks = [
{
type: "image" as const ,
data: bigPng.toString("base64" ),
mimeType: "image/png" ,
},
];
const out = await sanitizeContentBlocksImages(blocks, "test" , { maxBytes });
const image = getImageBlock(out);
const size = Buffer.from(image.data, "base64" ).byteLength;
expect(size).toBeLessThanOrEqual(maxBytes);
expect(image.mimeType).toBe("image/jpeg" );
}, 20 _000 );
it("sanitizes image arrays and reports drops" , async () => {
const png = await createWidePng();
const images = [
{ type: "image" as const , data: png.toString("base64" ), mimeType: "image/png" },
];
const { images: out, dropped } = await sanitizeImageBlocks(images, "test" );
expect(dropped).toBe(0 );
expect(out.length).toBe(1 );
const meta = await sharp(Buffer.from(out[0 ].data, "base64" )).metadata();
expect(meta.width).toBeLessThanOrEqual(1200 );
expect(meta.height).toBeLessThanOrEqual(1200 );
}, 20 _000 );
it("shrinks images that exceed max dimension even if size is small" , async () => {
const png = await createWidePng();
const blocks = [
{
type: "image" as const ,
data: png.toString("base64" ),
mimeType: "image/png" ,
},
];
const out = await sanitizeContentBlocksImages(blocks, "test" );
const image = getImageBlock(out);
const meta = await sharp(Buffer.from(image.data, "base64" )).metadata();
expect(meta.width).toBeLessThanOrEqual(1200 );
expect(meta.height).toBeLessThanOrEqual(1200 );
expect(image.mimeType).toBe("image/jpeg" );
}, 20 _000 );
it("corrects mismatched jpeg mimeType" , async () => {
const jpeg = await sharp({
create: {
width: 10 ,
height: 10 ,
channels: 3 ,
background: { r: 255 , g: 0 , b: 0 },
},
})
.jpeg()
.toBuffer();
const blocks = [
{
type: "image" as const ,
data: jpeg.toString("base64" ),
mimeType: "image/png" ,
},
];
const out = await sanitizeContentBlocksImages(blocks, "test" );
const image = getImageBlock(out);
expect(image.mimeType).toBe("image/jpeg" );
});
it("drops malformed image base64 payloads" , async () => {
const blocks = [
{
type: "image" as const ,
data: 'iVBORw0KGgoAAAANSUhEUgAAAAEAAAABCAQAAAC1HAwCAAAAC0lEQVR42mP8/x8AAwMCAO2N4j8AAAAASUVORK5CYII=" onerror="alert(1)' ,
mimeType: "image/png" ,
},
];
const out = await sanitizeContentBlocksImages(blocks, "test" );
expect(out).toEqual([
{
type: "text" ,
text: "[test] omitted image payload: invalid base64" ,
},
]);
});
});
Messung V0.5 in Prozent C=100 H=100 G=100
¤ Dauer der Verarbeitung: 0.13 Sekunden
(vorverarbeitet am 2026-06-09)
¤
*© Formatika GbR, Deutschland