Skip to main content
Glama
sharp.test.ts20.9 kB
import { describe, it, expect, vi, beforeEach, type Mock } from 'vitest'; import { resizeImageTool, resizeImageSchema } from '../../src/tools/sharp'; import { ErrorCode, McpError } from '@modelcontextprotocol/sdk/types.js'; import { z } from 'zod'; // Mock dependencies vi.mock('fs', () => ({ default: { readFileSync: vi.fn(), writeFileSync: vi.fn(), }, })); // A more comprehensive mock for sharp's fluent interface const mockSharpInstance = { metadata: vi.fn().mockReturnThis(), resize: vi.fn().mockReturnThis(), jpeg: vi.fn().mockReturnThis(), png: vi.fn().mockReturnThis(), webp: vi.fn().mockReturnThis(), avif: vi.fn().mockReturnThis(), rotate: vi.fn().mockReturnThis(), flip: vi.fn().mockReturnThis(), flop: vi.fn().mockReturnThis(), grayscale: vi.fn().mockReturnThis(), blur: vi.fn().mockReturnThis(), sharpen: vi.fn().mockReturnThis(), gamma: vi.fn().mockReturnThis(), negate: vi.fn().mockReturnThis(), normalize: vi.fn().mockReturnThis(), threshold: vi.fn().mockReturnThis(), trim: vi.fn().mockReturnThis(), toBuffer: vi.fn(), }; // Mock the sharp constructor to return our mock instance vi.mock('sharp', () => ({ default: vi.fn(() => mockSharpInstance), })); vi.mock('../../src/utils.js', () => ({ fetchImageFromUrl: vi.fn(), base64ToBuffer: vi.fn(), bufferToBase64: vi.fn((buffer: Buffer) => `mockBase64-${buffer.toString()}`), // Simple mock for base64 conversion isValidInputFormat: vi.fn(), normalizeFilePath: vi.fn((path: string) => path), // Passthrough mock })); // Helper to create a minimal valid args object export type ResizeImageArgs = z.infer<z.ZodObject<typeof resizeImageSchema>>; const createValidArgs = (overrides: Partial<ResizeImageArgs> = {}): ResizeImageArgs => ({ imagePath: 'test.jpg', // Default to imagePath outputImage: true, ...overrides, }); // A simple mock buffer const mockImageBuffer = Buffer.from('mockImageData'); const mockOutputBuffer = Buffer.from('mockOutputData'); describe('resizeImageTool', () => { beforeEach(async () => { // Reset mocks before each test vi.clearAllMocks(); // Default implementations for mocks const fsMockFromSetup = await import('fs'); // fs is mocked globally (fsMockFromSetup.default.readFileSync as Mock).mockReturnValue(mockImageBuffer); (fsMockFromSetup.default.writeFileSync as Mock).mockClear(); const sharpModule = await import('sharp'); // Only try to setup mockReturnValue if sharp.default is actually a mock function // This allows beforeEach to coexist with tests that unmock 'sharp' if (vi.isMockFunction(sharpModule.default)) { (sharpModule.default as unknown as Mock).mockReturnValue(mockSharpInstance); } // Reset calls for individual sharp instance methods on our shared mockSharpInstance Object.values(mockSharpInstance).forEach(mockFn => { if (typeof mockFn.mockClear === 'function') { mockFn.mockClear(); } }); mockSharpInstance.metadata.mockResolvedValue({ format: 'jpeg', width: 100, height: 100 }); mockSharpInstance.toBuffer.mockResolvedValue(mockOutputBuffer); const utilsActual = await import('../../src/utils.js'); if (vi.isMockFunction(utilsActual.fetchImageFromUrl)) { (utilsActual.fetchImageFromUrl as Mock).mockResolvedValue(mockImageBuffer); } if (vi.isMockFunction(utilsActual.base64ToBuffer)) { (utilsActual.base64ToBuffer as Mock).mockReturnValue(mockImageBuffer); } if (vi.isMockFunction(utilsActual.isValidInputFormat)) { (utilsActual.isValidInputFormat as Mock).mockReturnValue(true); } if (vi.isMockFunction(utilsActual.bufferToBase64)) { (utilsActual.bufferToBase64 as Mock).mockImplementation((buffer: Buffer) => `mockBase64-${buffer.toString()}`); } if (vi.isMockFunction(utilsActual.normalizeFilePath)) { (utilsActual.normalizeFilePath as Mock).mockImplementation((path: string) => path); } }); it('should throw McpError if no image source is provided', async () => { const args = createValidArgs({ imagePath: undefined, imageUrl: undefined, base64Image: undefined }); await expect(resizeImageTool(args)).rejects.toThrow( new McpError(ErrorCode.InvalidParams, 'One of imagePath, imageUrl, or base64Image must be provided'), ); }); it('should process image from imagePath successfully', async () => { const fsActual = await import('fs'); const args = createValidArgs({ imagePath: 'input.jpg', outputPath: 'output.jpg', width: 50, height: 50 }); const result = await resizeImageTool(args); expect(fsActual.default.readFileSync).toHaveBeenCalledWith('input.jpg'); expect(mockSharpInstance.resize).toHaveBeenCalledWith(expect.objectContaining({ width: 50, height: 50 })); expect(mockSharpInstance.jpeg).toHaveBeenCalledWith({ quality: 80 }); // Assuming default quality expect(mockSharpInstance.toBuffer).toHaveBeenCalled(); expect(fsActual.default.writeFileSync).toHaveBeenCalledWith('output.jpg', mockOutputBuffer); expect(result.isError).toBeUndefined(); expect(result.content).toBeDefined(); expect(result.content).toHaveLength(1); if (!result.content || typeof result.content[0].text !== 'string') { throw new Error('Invalid result content'); } const resultContent = JSON.parse(result.content[0].text); expect(resultContent.image).toBe('mockBase64-mockOutputData'); expect(resultContent.format).toBe('jpeg'); // Default output format if not specified and input is jpeg expect(resultContent.savedTo).toBe('output.jpg'); expect(resultContent.source).toBe('file'); }); it('should process image from imageUrl successfully', async () => { const utilsActual = await import('../../src/utils.js'); const args = createValidArgs({ imagePath: undefined, imageUrl: 'http://example.com/image.png', format: 'png' }); mockSharpInstance.metadata.mockResolvedValue({ format: 'png', width: 200, height: 150 }); const result = await resizeImageTool(args); expect(utilsActual.fetchImageFromUrl).toHaveBeenCalledWith('http://example.com/image.png'); expect(mockSharpInstance.png).toHaveBeenCalledWith({ quality: 80 }); expect(mockSharpInstance.toBuffer).toHaveBeenCalled(); expect(result.isError).toBeUndefined(); expect(result.content).toBeDefined(); expect(result.content).toHaveLength(1); if (!result.content || typeof result.content[0].text !== 'string') { throw new Error('Invalid result content'); } const resultContent = JSON.parse(result.content[0].text); expect(resultContent.image).toBe('mockBase64-mockOutputData'); expect(resultContent.format).toBe('png'); expect(resultContent.source).toBe('url'); }); it('should process image from base64Image successfully', async () => { const utilsActual = await import('../../src/utils.js'); const base64String = ''; // Minimal webp const args = createValidArgs({ imagePath: undefined, base64Image: base64String, format: 'webp' }); mockSharpInstance.metadata.mockResolvedValue({ format: 'webp', width: 10, height: 10 }); const result = await resizeImageTool(args); expect(utilsActual.base64ToBuffer).toHaveBeenCalledWith(base64String); expect(mockSharpInstance.webp).toHaveBeenCalledWith({ quality: 80 }); expect(mockSharpInstance.toBuffer).toHaveBeenCalled(); expect(result.isError).toBeUndefined(); expect(result.content).toBeDefined(); expect(result.content).toHaveLength(1); if (!result.content || typeof result.content[0].text !== 'string') { throw new Error('Invalid result content'); } const resultContent = JSON.parse(result.content[0].text); expect(resultContent.image).toBe('mockBase64-mockOutputData'); expect(resultContent.format).toBe('webp'); expect(resultContent.source).toBe('base64'); }); it('should throw McpError for unsupported input format', async () => { const utilsActual = await import('../../src/utils.js'); (utilsActual.isValidInputFormat as Mock).mockReturnValue(false); mockSharpInstance.metadata.mockResolvedValue({ format: 'bmp', width: 100, height: 100 }); const args = createValidArgs({ imagePath: 'input.bmp' }); await expect(resizeImageTool(args)).rejects.toThrow( new McpError(ErrorCode.InvalidParams, 'Unsupported input format: bmp'), ); }); it('should throw McpError if fs.readFileSync fails', async () => { const fsActual = await import('fs'); (fsActual.default.readFileSync as Mock).mockImplementation(() => { throw new Error('File not found'); }); const args = createValidArgs({ imagePath: 'nonexistent.jpg' }); await expect(resizeImageTool(args)).rejects.toThrow( new McpError(ErrorCode.InvalidParams, 'Failed to read image from path: nonexistent.jpg. File not found'), ); }); it('should throw McpError if fs.writeFileSync fails', async () => { const fsActual = await import('fs'); (fsActual.default.writeFileSync as Mock).mockImplementation(() => { throw new Error('Disk full'); }); const args = createValidArgs({ imagePath: 'input.jpg', outputPath: 'output.jpg' }); await expect(resizeImageTool(args)).rejects.toThrow( new McpError(ErrorCode.InternalError, 'Failed to save image to output.jpg: Disk full'), ); }); it('should apply various transformations like rotate and grayscale', async () => { const args = createValidArgs({ imagePath: 'input.jpg', rotate: 90, grayscale: true }); await resizeImageTool(args); expect(mockSharpInstance.rotate).toHaveBeenCalledWith(90); expect(mockSharpInstance.grayscale).toHaveBeenCalled(); }); it('should handle aspect ratio correctly when only width is provided', async () => { const args = createValidArgs({ imagePath: 'input.jpg', width: 100 }); await resizeImageTool(args); expect(mockSharpInstance.resize).toHaveBeenCalledWith(expect.objectContaining({ width: 100, height: undefined })); }); it('should handle aspect ratio correctly when only height is provided', async () => { const args = createValidArgs({ imagePath: 'input.jpg', height: 100 }); await resizeImageTool(args); expect(mockSharpInstance.resize).toHaveBeenCalledWith(expect.objectContaining({ width: undefined, height: 100 })); }); it('should use default dimensions and "contain" fit if none are provided', async () => { // This test needs to ensure constants are mocked correctly *before* the module under test is imported, // or it needs to re-import the module after mocking constants if they are top-level. // For simplicity, we'll assume constants are used as imported. // If constants were dynamic, this test would be more complex. // We are testing the logic within resizeImageTool that uses these defaults. // Reset modules to ensure a fresh import of constants for this test vi.resetModules(); // Mock constants specifically for this test case vi.doMock('../../src/constants.js', async () => { // It's important to import the original AFTER resetModules and within the factory // if you need to spread original values, but here we are fully defining them. return { DEFAULT_WIDTH: 300, DEFAULT_HEIGHT: 200, DEFAULT_QUALITY: 80, // Ensure this is also covered SUPPORTED_OUTPUT_FORMATS: ['jpeg', 'png', 'webp', 'avif', 'jpg'], // Match original or ensure it's sufficient // Add other constants if they are used by the module and need specific values or existence }; }); // Dynamically import the module under test AFTER mocks are set up const { resizeImageTool: toolWithMockedConstants } = await import('../../src/tools/sharp.js'); const args = createValidArgs({ imagePath: 'input.jpg' }); // No width, height, or fit await toolWithMockedConstants(args); expect(mockSharpInstance.resize).toHaveBeenCalledWith(expect.objectContaining({ width: 300, height: 200, fit: 'contain' })); // Clean up the mock for constants.js so it doesn't affect other tests vi.doUnmock('../../src/constants.js'); // Reset modules again to ensure subsequent tests get a clean state if they also import constants.js vi.resetModules(); }); it('should process HEIC image successfully using actual libheif-js and sharp', async () => { // Unmock sharp, libheif-js, and utils for this integration test vi.doUnmock('sharp'); vi.doUnmock('libheif-js'); vi.doUnmock('../../src/utils.js'); // Ensure actual utils are used vi.resetModules(); // Important to clear cache and re-import actual modules // Dynamically import the tool to get the version with actual sharp/libheif/utils const { resizeImageTool: actualResizeImageTool } = await import('../../src/tools/sharp.js'); const path = await vi.importActual<typeof import('path')>('path'); const nodeFs = await vi.importActual<typeof import('fs')>('fs'); const heicFileName = 'image1.heic'; const heicFilePath = path.resolve(__dirname, '../assets', heicFileName); const outputJpegFileName = 'output.heic.integration.test.jpg'; const outputJpegPath = path.resolve(__dirname, '../assets', outputJpegFileName); // Ensure the output file from previous runs is cleaned up if (nodeFs.existsSync(outputJpegPath)) { nodeFs.unlinkSync(outputJpegPath); } const heicInputBuffer = nodeFs.readFileSync(heicFilePath); if (!heicInputBuffer || heicInputBuffer.length === 0) { throw new Error(`Failed to read actual HEIC file or file is empty: ${heicFilePath}`); } // The 'fs' module used by the tool is mocked globally. // We need its readFileSync to return the actual HEIC buffer for this specific path. const mockedFsModule = await import('fs'); const originalReadFileSyncMock = mockedFsModule.default.readFileSync as Mock; (mockedFsModule.default.readFileSync as Mock).mockImplementation((p: string) => { if (p === heicFilePath) { return heicInputBuffer; } // Fallback to original mock behavior for other paths if necessary // For this test, we only expect heicFilePath to be read by the tool. // If other tests rely on the default mockImageBuffer, this could be: // return mockImageBuffer; // However, to be strict for this test: throw new Error(`Unexpected readFileSync call in HEIC test: ${p}`); }); // Ensure writeFileSync is a mock we can inspect and control for this test const mockWriteFileSync = mockedFsModule.default.writeFileSync as Mock; let capturedBufferForActualWrite: Buffer | undefined; mockWriteFileSync.mockImplementation((filePath: string, data: Buffer) => { if (filePath === outputJpegPath) { capturedBufferForActualWrite = data; // Capture buffer } // Do not throw, simulate successful mock write }); const args = { imagePath: heicFilePath, // Tool will use this path format: 'jpeg', width: 60, // Different dimensions for testing height: 40, quality: 85, outputImage: true, outputPath: outputJpegPath, }; const result = await actualResizeImageTool(args as any); // Restore original readFileSync mock for other tests (mockedFsModule.default.readFileSync as Mock).mockImplementation(originalReadFileSyncMock); expect(result.isError).toBeUndefined(); expect(result.content).toBeDefined(); expect(result.content).toHaveLength(1); if (!result.content || typeof result.content[0].text !== 'string') { throw new Error('Invalid result content for HEIC test'); } const resultContent = JSON.parse(result.content[0].text); expect(resultContent.format).toBe('jpeg'); expect(resultContent.width).toBe(60); expect(resultContent.height).toBe(40); expect(resultContent.source).toBe('file'); expect(resultContent.savedTo).toBe(outputJpegPath); expect(resultContent.image).toBeDefined(); expect(typeof resultContent.image).toBe('string'); expect(resultContent.image.startsWith('data:image/jpeg;base64,')).toBe(true); expect(resultContent.image.length).toBeGreaterThan(50); // Basic check for non-empty base64 // Verify the mock writeFileSync was called by the tool expect(mockWriteFileSync).toHaveBeenCalledTimes(1); expect(mockWriteFileSync).toHaveBeenCalledWith(outputJpegPath, capturedBufferForActualWrite); // Now, perform the actual write using the captured buffer for verification purposes expect(capturedBufferForActualWrite).toBeInstanceOf(Buffer); if (!capturedBufferForActualWrite) throw new Error('Buffer to write was not captured'); nodeFs.writeFileSync(outputJpegPath, capturedBufferForActualWrite); // Actual write // Read the actually written file and verify its properties with actual sharp expect(nodeFs.existsSync(outputJpegPath)).toBe(true); const writtenBuffer = nodeFs.readFileSync(outputJpegPath); // Actual read const sharpActual = (await import('sharp')).default; const metadata = await sharpActual(writtenBuffer).metadata(); expect(metadata.format).toBe('jpeg'); expect(metadata.width).toBe(60); expect(metadata.height).toBe(40); // Cleanup the created file if (nodeFs.existsSync(outputJpegPath)) { nodeFs.unlinkSync(outputJpegPath); } }); it('should process HEIC to PNG (base64 output) successfully using actual libheif-js and sharp', async () => { // Unmock sharp, libheif-js, and utils for this integration test vi.doUnmock('sharp'); vi.doUnmock('libheif-js'); vi.doUnmock('../../src/utils.js'); // Ensure actual utils are used vi.resetModules(); // Dynamically import the tool to get the version with actual sharp/libheif/utils const { resizeImageTool: actualResizeImageTool } = await import('../../src/tools/sharp.js'); const path = await vi.importActual<typeof import('path')>('path'); const nodeFs = await vi.importActual<typeof import('fs')>('fs'); const sharpActual = (await import('sharp')).default; // For verifying output const heicFileName = 'image1.heic'; const heicFilePath = path.resolve(__dirname, '../assets', heicFileName); const heicInputBuffer = nodeFs.readFileSync(heicFilePath); if (!heicInputBuffer || heicInputBuffer.length === 0) { throw new Error(`Failed to read actual HEIC file for PNG test or file is empty: ${heicFilePath}`); } const mockedFsModule = await import('fs'); const originalReadFileSyncMock = mockedFsModule.default.readFileSync as Mock; (mockedFsModule.default.readFileSync as Mock).mockImplementation((p: string) => { if (p === heicFilePath) { return heicInputBuffer; } throw new Error(`Unexpected readFileSync call in HEIC to PNG test: ${p}`); }); const mockWriteFileSync = mockedFsModule.default.writeFileSync as Mock; mockWriteFileSync.mockClear(); const targetWidth = 70; const targetHeight = 50; const args = { imagePath: heicFilePath, format: 'png', width: targetWidth, height: targetHeight, outputImage: true, quality: 90, // No outputPath, so it should return base64 only }; const result = await actualResizeImageTool(args as any); // Restore original readFileSync mock (mockedFsModule.default.readFileSync as Mock).mockImplementation(originalReadFileSyncMock); expect(result.isError).toBeUndefined(); expect(result.content).toBeDefined(); expect(result.content).toHaveLength(1); if (!result.content || typeof result.content[0].text !== 'string') { throw new Error('Invalid result content for HEIC to PNG test'); } const resultContent = JSON.parse(result.content[0].text); expect(resultContent.format).toBe('png'); expect(resultContent.width).toBe(targetWidth); expect(resultContent.height).toBe(targetHeight); expect(resultContent.source).toBe('file'); expect(resultContent.savedTo).toBeNull(); // Not saved to file expect(resultContent.image).toBeDefined(); expect(typeof resultContent.image).toBe('string'); expect(resultContent.image.startsWith('data:image/png;base64,')).toBe(true); expect(resultContent.image.length).toBeGreaterThan(50); // Verify writeFileSync was NOT called expect(mockWriteFileSync).not.toHaveBeenCalled(); // Verify the dimensions and format of the base64 output by decoding it const base64Data = resultContent.image.replace(/^data:image\/png;base64,/, ''); const outputBuffer = Buffer.from(base64Data, 'base64'); const metadata = await sharpActual(outputBuffer).metadata(); expect(metadata.format).toBe('png'); expect(metadata.width).toBe(targetWidth); expect(metadata.height).toBe(targetHeight); }); });

Latest Blog Posts

MCP directory API

We provide all the information about MCP servers via our MCP API.

curl -X GET 'https://glama.ai/api/mcp/v1/servers/BoomLinkAi/image-worker-mcp'

If you have feedback or need assistance with the MCP directory API, please join our Discord server