@wdio/image-comparison-core
Version:
Image comparison core module for @wdio/visual-service - WebdriverIO visual testing framework
1,228 lines (1,227 loc) • 61.9 kB
JavaScript
import { describe, it, expect, vi, beforeEach, afterEach } from 'vitest';
import { join } from 'node:path';
import { promises as fsPromises, readFileSync, writeFileSync } from 'node:fs';
import logger from '@wdio/logger';
import { DEFAULT_RESIZE_DIMENSIONS } from '../helpers/constants.js';
import { checkIfImageExists, removeDiffImageIfExists, checkBaselineImageExists, getRotatedImageIfNeeded, logDimensionWarning, getAdjustedAxis, handleIOSBezelCorners, cropAndConvertToDataURL, makeCroppedBase64Image, makeFullPageBase64Image, rotateBase64Image, takeResizedBase64Screenshot, } from './images.js';
const log = logger('test');
vi.mock('jimp', () => ({
Jimp: Object.assign(vi.fn().mockImplementation(() => ({
composite: vi.fn().mockReturnThis(),
getBase64: vi.fn().mockResolvedValue(''),
opacity: vi.fn().mockReturnThis(),
rotate: vi.fn().mockReturnThis(),
crop: vi.fn().mockReturnThis(),
})), {
read: vi.fn(),
MIME_PNG: 'image/png',
}),
JimpMime: {
png: 'image/png',
},
}));
vi.mock('@wdio/logger', () => import(join(process.cwd(), '__mocks__', '@wdio/logger')));
vi.mock('node:fs', async () => {
const actual = await vi.importActual('node:fs');
return {
...actual,
promises: {
access: vi.fn(),
unlink: vi.fn(),
},
readFileSync: vi.fn(),
writeFileSync: vi.fn(),
constants: {
R_OK: 4,
},
};
});
vi.mock('../helpers/utils.js', () => ({
getBase64ScreenshotSize: vi.fn(),
getAndCreatePath: vi.fn(),
getIosBezelImageNames: vi.fn(),
updateVisualBaseline: vi.fn(),
calculateDprData: vi.fn(),
}));
vi.mock('../helpers/constants.js', () => ({
DEFAULT_RESIZE_DIMENSIONS: { top: 0, right: 0, bottom: 0, left: 0 },
supportedIosBezelDevices: [
'iphonex', 'iphonexs', 'iphonexsmax', 'iphonexr', 'iphone11', 'iphone11pro', 'iphone11promax',
'iphone12', 'iphone12mini', 'iphone12pro', 'iphone12promax', 'iphone13', 'iphone13mini',
'iphone13pro', 'iphone13promax', 'iphone14', 'iphone14plus', 'iphone14pro', 'iphone14promax',
'iphone15', 'ipadmini', 'ipadair', 'ipadpro11', 'ipadpro129'
],
}));
vi.mock('./rectangles.js', () => ({
isWdioElement: vi.fn(),
determineStatusAddressToolBarRectangles: vi.fn(),
}));
vi.mock('./screenshots.js', () => ({
takeBase64Screenshot: vi.fn(),
}));
describe('checkIfImageExists', () => {
let accessSpy;
beforeEach(() => {
accessSpy = vi.spyOn(fsPromises, 'access');
});
afterEach(() => {
vi.clearAllMocks();
accessSpy.mockRestore();
});
it('should return true when file exists', async () => {
accessSpy.mockResolvedValue(undefined);
const result = await checkIfImageExists('/path/to/image.png');
expect(result).toMatchSnapshot();
});
it('should return false when file does not exist', async () => {
accessSpy.mockRejectedValue(new Error('File not found'));
const result = await checkIfImageExists('/path/to/image.png');
expect(result).toMatchSnapshot();
});
});
describe('removeDiffImageIfExists', () => {
let accessSpy;
let unlinkSpy;
let logInfoSpy;
beforeEach(() => {
accessSpy = vi.spyOn(fsPromises, 'access');
unlinkSpy = vi.spyOn(fsPromises, 'unlink');
logInfoSpy = vi.spyOn(log, 'info').mockImplementation(() => { });
});
afterEach(() => {
vi.clearAllMocks();
accessSpy.mockRestore();
unlinkSpy.mockRestore();
logInfoSpy.mockRestore();
});
// Note: We mock fsPromises.access here because removeDiffImageIfExists calls checkIfImageExists internally,
// which in turn calls fsPromises.access. This allows us to test the real integration between the functions
// without artificial mocking of internal dependencies.
it('should remove file when it exists', async () => {
accessSpy.mockResolvedValue(undefined);
unlinkSpy.mockResolvedValue(undefined);
await removeDiffImageIfExists('/path/to/diff.png');
expect(accessSpy).toHaveBeenCalledWith('/path/to/diff.png', 4);
expect(unlinkSpy).toHaveBeenCalledWith('/path/to/diff.png');
expect(logInfoSpy).toHaveBeenCalledWith('Successfully removed the diff image before comparing at /path/to/diff.png');
});
it('should do nothing when file does not exist', async () => {
accessSpy.mockRejectedValue(new Error('File not found'));
await removeDiffImageIfExists('/path/to/diff.png');
expect(accessSpy).toHaveBeenCalledWith('/path/to/diff.png', 4);
expect(unlinkSpy).not.toHaveBeenCalled();
expect(logInfoSpy).not.toHaveBeenCalled();
});
it('should throw error when file exists but cannot be removed', async () => {
accessSpy.mockResolvedValue(undefined);
const unlinkError = new Error('Permission denied');
unlinkSpy.mockRejectedValue(unlinkError);
await expect(removeDiffImageIfExists('/path/to/diff.png')).rejects.toThrow('Could not remove the diff image. The following error was thrown: Error: Permission denied');
expect(accessSpy).toHaveBeenCalledWith('/path/to/diff.png', 4);
expect(unlinkSpy).toHaveBeenCalledWith('/path/to/diff.png');
expect(logInfoSpy).not.toHaveBeenCalled();
});
});
describe('checkBaselineImageExists', () => {
let accessSpy;
let logInfoSpy;
beforeEach(() => {
accessSpy = vi.spyOn(fsPromises, 'access');
logInfoSpy = vi.spyOn(log, 'info').mockImplementation(() => { });
});
afterEach(() => {
vi.clearAllMocks();
accessSpy.mockRestore();
logInfoSpy.mockRestore();
});
it('should do nothing when baseline exists and no flags are set', async () => {
accessSpy.mockResolvedValue(undefined);
await checkBaselineImageExists({
actualFilePath: '/path/to/actual.png',
baselineFilePath: '/path/to/baseline.png'
});
expect(accessSpy).toHaveBeenCalledWith('/path/to/baseline.png', 4);
expect(logInfoSpy).not.toHaveBeenCalled();
});
it('should auto-save baseline when file does not exist and autoSaveBaseline is true', async () => {
accessSpy.mockRejectedValue(new Error('File not found'));
vi.mocked(readFileSync).mockReturnValue(Buffer.from('image data'));
vi.mocked(writeFileSync).mockImplementation(() => { });
await checkBaselineImageExists({
actualFilePath: '/path/to/actual.png',
baselineFilePath: '/path/to/baseline.png',
autoSaveBaseline: true
});
expect(accessSpy).toHaveBeenCalledWith('/path/to/baseline.png', 4);
expect(vi.mocked(readFileSync)).toHaveBeenCalledWith('/path/to/actual.png');
expect(vi.mocked(writeFileSync)).toHaveBeenCalledWith('/path/to/baseline.png', Buffer.from('image data'));
expect(logInfoSpy.mock.calls).toMatchSnapshot();
});
it('should update baseline when updateBaseline is true', async () => {
vi.mocked(readFileSync).mockReturnValue(Buffer.from('image data'));
vi.mocked(writeFileSync).mockImplementation(() => { });
await checkBaselineImageExists({
actualFilePath: '/path/to/actual.png',
baselineFilePath: '/path/to/baseline.png',
updateBaseline: true
});
expect(vi.mocked(readFileSync)).toHaveBeenCalledWith('/path/to/actual.png');
expect(vi.mocked(writeFileSync)).toHaveBeenCalledWith('/path/to/baseline.png', Buffer.from('image data'));
expect(logInfoSpy.mock.calls).toMatchSnapshot();
});
it('should use provided base64 when auto-saving baseline', async () => {
accessSpy.mockRejectedValue(new Error('File not found'));
const base64Actual = Buffer.from('image data').toString('base64');
vi.mocked(writeFileSync).mockImplementation(() => { });
await checkBaselineImageExists({
actualFilePath: '/path/to/actual.png',
baselineFilePath: '/path/to/baseline.png',
autoSaveBaseline: true,
actualBase64Image: base64Actual,
});
expect(vi.mocked(readFileSync)).not.toHaveBeenCalled();
expect(vi.mocked(writeFileSync)).toHaveBeenCalledWith('/path/to/baseline.png', Buffer.from(base64Actual, 'base64'));
expect(logInfoSpy.mock.calls).toMatchSnapshot();
});
it('should throw error when file does not exist and autoSaveBaseline is false', async () => {
accessSpy.mockRejectedValue(new Error('File not found'));
await expect(checkBaselineImageExists({
actualFilePath: '/path/to/actual.png',
baselineFilePath: '/path/to/baseline.png',
autoSaveBaseline: false
})).rejects.toThrow();
expect(accessSpy).toHaveBeenCalledWith('/path/to/baseline.png', 4);
expect(vi.mocked(readFileSync)).not.toHaveBeenCalled();
expect(vi.mocked(writeFileSync)).not.toHaveBeenCalled();
expect(logInfoSpy).not.toHaveBeenCalled();
});
it('should throw error when copying fails', async () => {
accessSpy.mockRejectedValue(new Error('File not found'));
const copyError = new Error('Permission denied');
vi.mocked(readFileSync).mockImplementation(() => { throw copyError; });
await expect(checkBaselineImageExists({
actualFilePath: '/path/to/actual.png',
baselineFilePath: '/path/to/baseline.png',
autoSaveBaseline: true
})).rejects.toThrow();
expect(accessSpy).toHaveBeenCalledWith('/path/to/baseline.png', 4);
expect(vi.mocked(readFileSync)).toHaveBeenCalledWith('/path/to/actual.png');
expect(vi.mocked(writeFileSync)).not.toHaveBeenCalled();
expect(logInfoSpy).not.toHaveBeenCalled();
});
it('should mention missing actual file in error when not saved to disk', async () => {
accessSpy.mockRejectedValue(new Error('File not found'));
vi.mocked(fsPromises.access).mockRejectedValue(new Error('File not found'));
await expect(checkBaselineImageExists({
actualFilePath: '/path/to/actual.png',
baselineFilePath: '/path/to/baseline.png',
autoSaveBaseline: false
})).rejects.toThrow(/actual image was not saved to disk/);
expect(vi.mocked(readFileSync)).not.toHaveBeenCalled();
expect(vi.mocked(writeFileSync)).not.toHaveBeenCalled();
});
});
describe('rotateBase64Image', () => {
let jimpReadMock;
beforeEach(async () => {
const jimp = await import('jimp');
jimpReadMock = vi.mocked(jimp.Jimp.read);
});
afterEach(() => {
vi.clearAllMocks();
});
it('should rotate image by specified degrees', async () => {
const mockImage = {
rotate: vi.fn().mockReturnThis(),
getBase64: vi.fn().mockResolvedValue('')
};
jimpReadMock.mockResolvedValue(mockImage);
const result = await rotateBase64Image({
base64Image: 'originalImageData',
degrees: 90
});
expect(result).toMatchSnapshot();
expect(jimpReadMock.mock.calls).toMatchSnapshot();
expect(mockImage.rotate.mock.calls).toMatchSnapshot();
expect(mockImage.getBase64.mock.calls).toMatchSnapshot();
});
it('should rotate image by 180 degrees', async () => {
const mockImage = {
rotate: vi.fn().mockReturnThis(),
getBase64: vi.fn().mockResolvedValue('')
};
jimpReadMock.mockResolvedValue(mockImage);
const result = await rotateBase64Image({
base64Image: 'originalImageData',
degrees: 180
});
expect(result).toMatchSnapshot();
expect(mockImage.rotate.mock.calls).toMatchSnapshot();
});
it('should handle different base64 input', async () => {
const mockImage = {
rotate: vi.fn().mockReturnThis(),
getBase64: vi.fn().mockResolvedValue('')
};
jimpReadMock.mockResolvedValue(mockImage);
const result = await rotateBase64Image({
base64Image: 'differentImageData',
degrees: 270
});
expect(result).toMatchSnapshot();
expect(jimpReadMock.mock.calls).toMatchSnapshot();
expect(mockImage.rotate.mock.calls).toMatchSnapshot();
});
});
describe('getRotatedImageIfNeeded', () => {
let getBase64ScreenshotSizeMock;
beforeEach(async () => {
const utils = await import('../helpers/utils.js');
getBase64ScreenshotSizeMock = vi.mocked(utils.getBase64ScreenshotSize);
vi.clearAllMocks();
});
afterEach(() => {
vi.clearAllMocks();
});
it('should return original image when no rotation is needed', async () => {
getBase64ScreenshotSizeMock.mockReturnValue({ width: 1920, height: 1080 });
const result = await getRotatedImageIfNeeded({
isWebDriverElementScreenshot: false,
isLandscape: false,
base64Image: 'originalImageData'
});
expect(result).toMatchSnapshot();
expect(getBase64ScreenshotSizeMock.mock.calls).toMatchSnapshot();
});
it('should call rotateBase64Image when landscape and height > width', async () => {
getBase64ScreenshotSizeMock.mockReturnValue({ width: 1080, height: 1920 });
// We'll test that the function calls rotateBase64Image by checking the result
// Since we can't easily mock the internal function, we'll verify the logic works
const result = await getRotatedImageIfNeeded({
isWebDriverElementScreenshot: false,
isLandscape: true,
base64Image: 'originalImageData'
});
expect(result).toMatchSnapshot();
expect(getBase64ScreenshotSizeMock.mock.calls).toMatchSnapshot();
});
it('should not rotate when isWebDriverElementScreenshot is true', async () => {
getBase64ScreenshotSizeMock.mockReturnValue({ width: 1080, height: 1920 });
const result = await getRotatedImageIfNeeded({
isWebDriverElementScreenshot: true,
isLandscape: true,
base64Image: 'originalImageData'
});
expect(result).toMatchSnapshot();
expect(getBase64ScreenshotSizeMock.mock.calls).toMatchSnapshot();
});
it('should not rotate when width >= height', async () => {
getBase64ScreenshotSizeMock.mockReturnValue({ width: 1920, height: 1080 });
const result = await getRotatedImageIfNeeded({
isWebDriverElementScreenshot: false,
isLandscape: true,
base64Image: 'originalImageData'
});
expect(result).toMatchSnapshot();
expect(getBase64ScreenshotSizeMock.mock.calls).toMatchSnapshot();
});
it('should not rotate when not landscape', async () => {
getBase64ScreenshotSizeMock.mockReturnValue({ width: 1080, height: 1920 });
const result = await getRotatedImageIfNeeded({
isWebDriverElementScreenshot: false,
isLandscape: false,
base64Image: 'originalImageData'
});
expect(result).toMatchSnapshot();
expect(getBase64ScreenshotSizeMock.mock.calls).toMatchSnapshot();
});
});
describe('logDimensionWarning', () => {
let logWarnSpy;
beforeEach(() => {
logWarnSpy = vi.spyOn(log, 'warn').mockImplementation(() => { });
});
afterEach(() => {
vi.clearAllMocks();
logWarnSpy.mockRestore();
});
it('should log warning for LEFT type', () => {
logDimensionWarning({
dimension: 60,
maxDimension: 1000,
position: -10,
type: 'LEFT'
});
expect(logWarnSpy.mock.calls).toMatchSnapshot();
});
it('should log warning for RIGHT type', () => {
logDimensionWarning({
dimension: 50,
maxDimension: 1000,
position: 1100,
type: 'RIGHT'
});
expect(logWarnSpy.mock.calls).toMatchSnapshot();
});
it('should log warning for TOP type', () => {
logDimensionWarning({
dimension: 30,
maxDimension: 800,
position: -5,
type: 'TOP'
});
expect(logWarnSpy.mock.calls).toMatchSnapshot();
});
it('should log warning for BOTTOM type', () => {
logDimensionWarning({
dimension: 40,
maxDimension: 800,
position: 850,
type: 'BOTTOM'
});
expect(logWarnSpy.mock.calls).toMatchSnapshot();
});
});
describe('getAdjustedAxis', () => {
it('should return adjusted coordinates within bounds', () => {
const result = getAdjustedAxis({
length: 100,
maxDimension: 1000,
paddingEnd: 10,
paddingStart: 5,
start: 50,
warningType: 'WIDTH'
});
expect(result).toMatchSnapshot();
});
it('should handle zero padding', () => {
const result = getAdjustedAxis({
length: 100,
maxDimension: 1000,
paddingEnd: 0,
paddingStart: 0,
start: 50,
warningType: 'WIDTH'
});
expect(result).toMatchSnapshot();
});
it('should clamp start position to 0 when it goes below 0', () => {
const result = getAdjustedAxis({
length: 100,
maxDimension: 1000,
paddingEnd: 10,
paddingStart: 60, // This will make adjustedStart = 50 - 60 = -10
start: 50,
warningType: 'WIDTH'
});
expect(result).toMatchSnapshot();
});
it('should clamp end position to maxDimension when it exceeds maxDimension', () => {
const result = getAdjustedAxis({
length: 100,
maxDimension: 1000,
paddingEnd: 50, // This will make adjustedEnd = 950 + 100 + 50 = 1100
paddingStart: 10,
start: 950,
warningType: 'WIDTH'
});
expect(result).toMatchSnapshot();
});
it('should handle both start and end clamping', () => {
const result = getAdjustedAxis({
length: 100,
maxDimension: 1000,
paddingEnd: 50,
paddingStart: 60,
start: 50,
warningType: 'WIDTH'
});
expect(result).toMatchSnapshot();
});
it('should handle HEIGHT warning type correctly', () => {
const result = getAdjustedAxis({
length: 100,
maxDimension: 800,
paddingEnd: 50,
paddingStart: 60,
start: 50,
warningType: 'HEIGHT'
});
expect(result).toMatchSnapshot();
});
it('should handle edge case where start is exactly at maxDimension', () => {
const result = getAdjustedAxis({
length: 100,
maxDimension: 1000,
paddingEnd: 10,
paddingStart: 0,
start: 1000,
warningType: 'WIDTH'
});
expect(result).toMatchSnapshot();
});
it('should handle edge case where start is 0', () => {
const result = getAdjustedAxis({
length: 100,
maxDimension: 1000,
paddingEnd: 10,
paddingStart: 0,
start: 0,
warningType: 'WIDTH'
});
expect(result).toMatchSnapshot();
});
it('should handle large padding values', () => {
const result = getAdjustedAxis({
length: 50,
maxDimension: 100,
paddingEnd: 100,
paddingStart: 100,
start: 50,
warningType: 'WIDTH'
});
expect(result).toMatchSnapshot();
});
it('should handle zero length', () => {
const result = getAdjustedAxis({
length: 0,
maxDimension: 1000,
paddingEnd: 10,
paddingStart: 5,
start: 50,
warningType: 'WIDTH'
});
expect(result).toMatchSnapshot();
});
it('should handle negative start position', () => {
const result = getAdjustedAxis({
length: 100,
maxDimension: 1000,
paddingEnd: 10,
paddingStart: 0,
start: -10,
warningType: 'WIDTH'
});
expect(result).toMatchSnapshot();
});
});
describe('handleIOSBezelCorners', () => {
let logWarnSpy;
let getIosBezelImageNamesMock;
let readFileSyncMock;
let getBase64ScreenshotSizeMock;
let mockImage;
beforeEach(async () => {
logWarnSpy = vi.spyOn(log, 'warn').mockImplementation(() => { });
const utilsModule = vi.mocked(await import('../helpers/utils.js'));
getIosBezelImageNamesMock = vi.spyOn(utilsModule, 'getIosBezelImageNames');
getBase64ScreenshotSizeMock = vi.spyOn(utilsModule, 'getBase64ScreenshotSize');
const fsModule = vi.mocked(await import('node:fs'));
readFileSyncMock = vi.spyOn(fsModule, 'readFileSync');
mockImage = {
composite: vi.fn().mockReturnThis(),
getBase64: vi.fn().mockResolvedValue(''),
opacity: vi.fn().mockReturnThis(),
rotate: vi.fn().mockReturnThis(),
};
});
afterEach(() => {
vi.clearAllMocks();
logWarnSpy.mockRestore();
});
it('should do nothing when addIOSBezelCorners is false', async () => {
await handleIOSBezelCorners({
addIOSBezelCorners: false,
image: mockImage,
deviceName: 'iPhone 14 Pro',
devicePixelRatio: 3,
height: 844,
isLandscape: false,
width: 390,
});
expect(getIosBezelImageNamesMock).not.toHaveBeenCalled();
expect(readFileSyncMock).not.toHaveBeenCalled();
expect(logWarnSpy).not.toHaveBeenCalled();
});
it('should handle supported iPhone device', async () => {
getIosBezelImageNamesMock.mockReturnValue({
topImageName: 'iphone14pro-top',
bottomImageName: 'iphone14pro-bottom'
});
readFileSyncMock.mockReturnValue('mockImageData');
getBase64ScreenshotSizeMock.mockReturnValue({ width: 100, height: 50 });
await handleIOSBezelCorners({
addIOSBezelCorners: true,
image: mockImage,
deviceName: 'iPhone 14 Pro',
devicePixelRatio: 3,
height: 844,
isLandscape: false,
width: 390,
});
expect(getIosBezelImageNamesMock.mock.calls).toMatchSnapshot();
expect(readFileSyncMock).toHaveBeenCalledTimes(2);
expect(mockImage.composite).toHaveBeenCalledTimes(2);
expect(logWarnSpy).not.toHaveBeenCalled();
});
it('should handle supported iPhone device in landscape mode', async () => {
getIosBezelImageNamesMock.mockReturnValue({
topImageName: 'iphone14pro-top',
bottomImageName: 'iphone14pro-bottom'
});
readFileSyncMock.mockReturnValue('mockImageData');
getBase64ScreenshotSizeMock.mockReturnValue({ width: 100, height: 50 });
await handleIOSBezelCorners({
addIOSBezelCorners: true,
image: mockImage,
deviceName: 'iPhone 14 Pro',
devicePixelRatio: 3,
height: 390,
isLandscape: true,
width: 844,
});
expect(getIosBezelImageNamesMock.mock.calls).toMatchSnapshot();
expect(readFileSyncMock).toHaveBeenCalledTimes(2);
expect(mockImage.composite).toHaveBeenCalledTimes(2);
expect(logWarnSpy).not.toHaveBeenCalled();
});
it('should handle supported iPad device with sufficient dimensions', async () => {
getIosBezelImageNamesMock.mockReturnValue({
topImageName: 'ipadair-top',
bottomImageName: 'ipadair-bottom'
});
readFileSyncMock.mockReturnValue('mockImageData');
getBase64ScreenshotSizeMock.mockReturnValue({ width: 100, height: 50 });
await handleIOSBezelCorners({
addIOSBezelCorners: true,
image: mockImage,
deviceName: 'iPad Air',
devicePixelRatio: 2,
height: 2400, // 2400 / 2 = 1200 >= 1133
isLandscape: false,
width: 1600, // 1600 / 2 = 800 < 1133, but height meets requirement
});
expect(getIosBezelImageNamesMock.mock.calls).toMatchSnapshot();
expect(readFileSyncMock).toHaveBeenCalledTimes(2);
expect(mockImage.composite).toHaveBeenCalledTimes(2);
expect(logWarnSpy).not.toHaveBeenCalled();
});
it('should not handle iPad device with insufficient dimensions', async () => {
await handleIOSBezelCorners({
addIOSBezelCorners: true,
image: mockImage,
deviceName: 'iPad Air',
devicePixelRatio: 2,
height: 800, // Below 1133 threshold
isLandscape: false,
width: 600,
});
expect(getIosBezelImageNamesMock).not.toHaveBeenCalled();
expect(readFileSyncMock).not.toHaveBeenCalled();
expect(logWarnSpy.mock.calls).toMatchSnapshot();
});
it('should handle device name normalization', async () => {
getIosBezelImageNamesMock.mockReturnValue({
topImageName: 'iphone14pro-top',
bottomImageName: 'iphone14pro-bottom'
});
readFileSyncMock.mockReturnValue('mockImageData');
getBase64ScreenshotSizeMock.mockReturnValue({ width: 100, height: 50 });
await handleIOSBezelCorners({
addIOSBezelCorners: true,
image: mockImage,
deviceName: 'iPhone 14 Pro Simulator (5th generation)',
devicePixelRatio: 3,
height: 844,
isLandscape: false,
width: 390,
});
expect(getIosBezelImageNamesMock.mock.calls).toMatchSnapshot();
expect(readFileSyncMock).toHaveBeenCalledTimes(2);
expect(mockImage.composite).toHaveBeenCalledTimes(2);
expect(logWarnSpy).not.toHaveBeenCalled();
});
it('should handle unsupported device', async () => {
await handleIOSBezelCorners({
addIOSBezelCorners: true,
image: mockImage,
deviceName: 'iPhone 6',
devicePixelRatio: 2,
height: 667,
isLandscape: false,
width: 375,
});
expect(getIosBezelImageNamesMock).not.toHaveBeenCalled();
expect(readFileSyncMock).not.toHaveBeenCalled();
expect(logWarnSpy.mock.calls).toMatchSnapshot();
});
it('should handle missing bezel image names', async () => {
getIosBezelImageNamesMock.mockReturnValue({
topImageName: null,
bottomImageName: null
});
await handleIOSBezelCorners({
addIOSBezelCorners: true,
image: mockImage,
deviceName: 'iPhone 14 Pro',
devicePixelRatio: 3,
height: 844,
isLandscape: false,
width: 390,
});
expect(getIosBezelImageNamesMock.mock.calls).toMatchSnapshot();
expect(readFileSyncMock).not.toHaveBeenCalled();
expect(mockImage.composite).not.toHaveBeenCalled();
expect(logWarnSpy.mock.calls).toMatchSnapshot();
});
it('should handle partial bezel image names', async () => {
getIosBezelImageNamesMock.mockReturnValue({
topImageName: 'iphone14pro-top',
bottomImageName: null
});
await handleIOSBezelCorners({
addIOSBezelCorners: true,
image: mockImage,
deviceName: 'iPhone 14 Pro',
devicePixelRatio: 3,
height: 844,
isLandscape: false,
width: 390,
});
expect(getIosBezelImageNamesMock.mock.calls).toMatchSnapshot();
expect(readFileSyncMock).not.toHaveBeenCalled();
expect(mockImage.composite).not.toHaveBeenCalled();
expect(logWarnSpy.mock.calls).toMatchSnapshot();
});
it('should handle Android device (not iOS)', async () => {
await handleIOSBezelCorners({
addIOSBezelCorners: true,
image: mockImage,
deviceName: 'Samsung Galaxy S21',
devicePixelRatio: 3,
height: 2400,
isLandscape: false,
width: 1080,
});
expect(getIosBezelImageNamesMock).not.toHaveBeenCalled();
expect(readFileSyncMock).not.toHaveBeenCalled();
expect(logWarnSpy.mock.calls).toMatchSnapshot();
});
});
describe('cropAndConvertToDataURL', () => {
let mockImage;
let mockCroppedImage;
const defaultCropOptions = {
addIOSBezelCorners: false,
base64Image: 'originalImageData',
deviceName: 'iPhone 14 Pro',
devicePixelRatio: 3,
height: 100,
isIOS: false,
isLandscape: false,
sourceX: 50,
sourceY: 25,
width: 200,
};
beforeEach(async () => {
mockCroppedImage = {
getBase64: vi.fn().mockResolvedValue(''),
};
mockImage = {
crop: vi.fn().mockReturnValue(mockCroppedImage),
};
const jimpModule = vi.mocked(await import('jimp'));
vi.spyOn(jimpModule.Jimp, 'read').mockResolvedValue(mockImage);
});
afterEach(() => {
vi.clearAllMocks();
});
it('should crop image and return base64 data without iOS bezel corners', async () => {
const result = await cropAndConvertToDataURL(defaultCropOptions);
expect(mockImage.crop.mock.calls).toMatchSnapshot();
expect(mockCroppedImage.getBase64.mock.calls).toMatchSnapshot();
expect(result).toMatchSnapshot();
});
it('should crop image and add iOS bezel corners when isIOS is true', async () => {
const result = await cropAndConvertToDataURL({
...defaultCropOptions,
addIOSBezelCorners: true,
isIOS: true,
});
expect(mockImage.crop.mock.calls).toMatchSnapshot();
expect(mockCroppedImage.getBase64.mock.calls).toMatchSnapshot();
expect(result).toMatchSnapshot();
});
it('should handle landscape orientation with iOS bezel corners', async () => {
const result = await cropAndConvertToDataURL({
...defaultCropOptions,
addIOSBezelCorners: true,
isIOS: true,
isLandscape: true,
});
expect(mockImage.crop.mock.calls).toMatchSnapshot();
expect(mockCroppedImage.getBase64.mock.calls).toMatchSnapshot();
expect(result).toMatchSnapshot();
});
it('should handle Android device (isIOS false) without bezel corners', async () => {
const result = await cropAndConvertToDataURL({
...defaultCropOptions,
addIOSBezelCorners: true,
deviceName: 'Samsung Galaxy S21',
isIOS: false,
});
expect(mockImage.crop.mock.calls).toMatchSnapshot();
expect(mockCroppedImage.getBase64.mock.calls).toMatchSnapshot();
expect(result).toMatchSnapshot();
});
it('should handle zero dimensions', async () => {
const result = await cropAndConvertToDataURL({
...defaultCropOptions,
height: 0,
sourceX: 0,
sourceY: 0,
width: 0,
});
expect(mockImage.crop.mock.calls).toMatchSnapshot();
expect(mockCroppedImage.getBase64.mock.calls).toMatchSnapshot();
expect(result).toMatchSnapshot();
});
it('should handle large crop dimensions', async () => {
const result = await cropAndConvertToDataURL({
...defaultCropOptions,
height: 2000,
sourceX: 1000,
sourceY: 500,
width: 3000,
});
expect(mockImage.crop.mock.calls).toMatchSnapshot();
expect(mockCroppedImage.getBase64.mock.calls).toMatchSnapshot();
expect(result).toMatchSnapshot();
});
it('should handle different base64 input data', async () => {
const result = await cropAndConvertToDataURL({
...defaultCropOptions,
base64Image: 'differentImageData123',
});
expect(mockImage.crop.mock.calls).toMatchSnapshot();
expect(mockCroppedImage.getBase64.mock.calls).toMatchSnapshot();
expect(result).toMatchSnapshot();
});
it('should handle different device pixel ratios', async () => {
const result = await cropAndConvertToDataURL({
...defaultCropOptions,
addIOSBezelCorners: true,
devicePixelRatio: 2,
isIOS: true,
});
expect(mockImage.crop.mock.calls).toMatchSnapshot();
expect(mockCroppedImage.getBase64.mock.calls).toMatchSnapshot();
expect(result).toMatchSnapshot();
});
});
describe('makeCroppedBase64Image', () => {
let getBase64ScreenshotSizeMock;
let mockImage;
let mockCroppedImage;
const defaultCropOptions = {
addIOSBezelCorners: false,
base64Image: 'originalImageData',
deviceName: 'iPhone 14 Pro',
devicePixelRatio: 3,
isWebDriverElementScreenshot: false,
isIOS: false,
isLandscape: false,
rectangles: {
height: 100,
width: 200,
x: 50,
y: 25,
},
resizeDimensions: { top: 0, right: 0, bottom: 0, left: 0 },
};
beforeEach(async () => {
const utilsModule = vi.mocked(await import('../helpers/utils.js'));
getBase64ScreenshotSizeMock = vi.spyOn(utilsModule, 'getBase64ScreenshotSize');
mockCroppedImage = {
getBase64: vi.fn().mockResolvedValue(''),
composite: vi.fn().mockReturnThis(),
opacity: vi.fn().mockReturnThis(),
};
mockImage = {
crop: vi.fn().mockReturnValue(mockCroppedImage),
composite: vi.fn().mockReturnThis(),
getBase64: vi.fn().mockResolvedValue(''),
opacity: vi.fn().mockReturnThis(),
rotate: vi.fn().mockReturnThis(),
};
const jimpModule = vi.mocked(await import('jimp'));
vi.spyOn(jimpModule.Jimp, 'read').mockResolvedValue(mockImage);
getBase64ScreenshotSizeMock.mockReturnValue({ width: 1000, height: 2000 });
});
afterEach(() => {
vi.clearAllMocks();
});
it('should create cropped base64 image with default settings', async () => {
const result = await makeCroppedBase64Image(defaultCropOptions);
expect(getBase64ScreenshotSizeMock.mock.calls).toMatchSnapshot();
expect(mockImage.crop.mock.calls).toMatchSnapshot();
expect(mockCroppedImage.getBase64.mock.calls).toMatchSnapshot();
expect(result).toMatchSnapshot();
});
it('should handle landscape orientation with rotation', async () => {
const result = await makeCroppedBase64Image({
...defaultCropOptions,
isLandscape: true,
});
expect(getBase64ScreenshotSizeMock.mock.calls).toMatchSnapshot();
expect(mockImage.crop.mock.calls).toMatchSnapshot();
expect(result).toMatchSnapshot();
});
it('should handle web driver element screenshots', async () => {
const result = await makeCroppedBase64Image({
...defaultCropOptions,
isWebDriverElementScreenshot: true,
});
expect(getBase64ScreenshotSizeMock.mock.calls).toMatchSnapshot();
expect(mockImage.crop.mock.calls).toMatchSnapshot();
expect(result).toMatchSnapshot();
});
it('should handle iOS devices with bezel corners', async () => {
const result = await makeCroppedBase64Image({
...defaultCropOptions,
addIOSBezelCorners: true,
isIOS: true,
});
expect(getBase64ScreenshotSizeMock.mock.calls).toMatchSnapshot();
expect(mockImage.crop.mock.calls).toMatchSnapshot();
expect(result).toMatchSnapshot();
});
it('should handle custom resize dimensions', async () => {
const result = await makeCroppedBase64Image({
...defaultCropOptions,
resizeDimensions: { top: 10, right: 20, bottom: 15, left: 5 },
});
expect(getBase64ScreenshotSizeMock.mock.calls).toMatchSnapshot();
expect(mockImage.crop.mock.calls).toMatchSnapshot();
expect(result).toMatchSnapshot();
});
it('should handle different rectangle dimensions', async () => {
const result = await makeCroppedBase64Image({
...defaultCropOptions,
rectangles: {
height: 300,
width: 400,
x: 100,
y: 75,
},
});
expect(getBase64ScreenshotSizeMock.mock.calls).toMatchSnapshot();
expect(mockImage.crop.mock.calls).toMatchSnapshot();
expect(result).toMatchSnapshot();
});
it('should handle different screenshot sizes', async () => {
getBase64ScreenshotSizeMock.mockReturnValue({ width: 800, height: 600 });
const result = await makeCroppedBase64Image(defaultCropOptions);
expect(getBase64ScreenshotSizeMock.mock.calls).toMatchSnapshot();
expect(mockImage.crop.mock.calls).toMatchSnapshot();
expect(result).toMatchSnapshot();
});
it('should handle different device pixel ratios', async () => {
const result = await makeCroppedBase64Image({
...defaultCropOptions,
devicePixelRatio: 2,
});
expect(getBase64ScreenshotSizeMock.mock.calls).toMatchSnapshot();
expect(mockImage.crop.mock.calls).toMatchSnapshot();
expect(result).toMatchSnapshot();
});
it('should handle zero rectangle dimensions', async () => {
const result = await makeCroppedBase64Image({
...defaultCropOptions,
rectangles: {
height: 0,
width: 0,
x: 0,
y: 0,
},
});
expect(getBase64ScreenshotSizeMock.mock.calls).toMatchSnapshot();
expect(mockImage.crop.mock.calls).toMatchSnapshot();
expect(result).toMatchSnapshot();
});
it('should handle edge case with padding that exceeds image bounds', async () => {
const result = await makeCroppedBase64Image({
...defaultCropOptions,
rectangles: {
height: 100,
width: 200,
x: 950, // Very close to image width (1000)
y: 1900, // Very close to image height (2000)
},
resizeDimensions: { top: 50, right: 100, bottom: 50, left: 50 },
});
expect(getBase64ScreenshotSizeMock.mock.calls).toMatchSnapshot();
expect(mockImage.crop.mock.calls).toMatchSnapshot();
expect(result).toMatchSnapshot();
});
});
describe('makeFullPageBase64Image', () => {
let getBase64ScreenshotSizeMock;
let mockCanvas;
let mockImage;
const defaultScreenshotsData = {
fullPageHeight: 2000,
fullPageWidth: 1000,
data: [
{
canvasWidth: 1000,
canvasYPosition: 0,
imageHeight: 800,
imageWidth: 1000,
imageXPosition: 0,
imageYPosition: 0,
screenshot: 'screenshot1-data'
},
{
canvasWidth: 1000,
canvasYPosition: 800,
imageHeight: 800,
imageWidth: 1000,
imageXPosition: 0,
imageYPosition: 0,
screenshot: 'screenshot2-data'
},
{
canvasWidth: 1000,
canvasYPosition: 1600,
imageHeight: 400,
imageWidth: 1000,
imageXPosition: 0,
imageYPosition: 0,
screenshot: 'screenshot3-data'
}
]
};
const defaultOptions = {
devicePixelRatio: 2,
isLandscape: false
};
beforeEach(async () => {
const utilsModule = vi.mocked(await import('../helpers/utils.js'));
getBase64ScreenshotSizeMock = vi.spyOn(utilsModule, 'getBase64ScreenshotSize');
mockCanvas = {
composite: vi.fn().mockReturnThis(),
getBase64: vi.fn().mockResolvedValue(''),
};
mockImage = {
bitmap: {
width: 1000,
height: 800,
},
crop: vi.fn().mockReturnThis(),
composite: vi.fn().mockReturnThis(),
getBase64: vi.fn().mockResolvedValue(''),
opacity: vi.fn().mockReturnThis(),
rotate: vi.fn().mockReturnThis(),
};
const jimpModule = vi.mocked(await import('jimp'));
vi.spyOn(jimpModule.Jimp, 'read').mockResolvedValue(mockImage);
vi.mocked(jimpModule.Jimp).mockImplementation((options) => {
if (options && (options.width || options.height)) {
return mockCanvas;
}
return mockImage;
});
getBase64ScreenshotSizeMock.mockReturnValue({ width: 1000, height: 800 });
});
afterEach(() => {
vi.clearAllMocks();
});
it('should create full page base64 image with multiple screenshots', async () => {
const result = await makeFullPageBase64Image(defaultScreenshotsData, defaultOptions);
expect(getBase64ScreenshotSizeMock.mock.calls).toMatchSnapshot();
expect(mockImage.crop.mock.calls).toMatchSnapshot();
expect(mockCanvas.composite.mock.calls).toMatchSnapshot();
expect(mockCanvas.getBase64.mock.calls).toMatchSnapshot();
expect(result).toMatchSnapshot();
});
it('should handle landscape mode with rotation', async () => {
getBase64ScreenshotSizeMock.mockReturnValue({ width: 800, height: 1000 });
const result = await makeFullPageBase64Image(defaultScreenshotsData, {
...defaultOptions,
isLandscape: true
});
expect(getBase64ScreenshotSizeMock.mock.calls).toMatchSnapshot();
expect(mockImage.crop.mock.calls).toMatchSnapshot();
expect(mockCanvas.composite.mock.calls).toMatchSnapshot();
expect(result).toMatchSnapshot();
});
it('should handle single screenshot', async () => {
const singleScreenshotData = {
fullPageHeight: 800,
fullPageWidth: 1000,
data: [
{
canvasWidth: 1000,
canvasYPosition: 0,
imageHeight: 800,
imageWidth: 1000,
imageXPosition: 0,
imageYPosition: 0,
screenshot: 'single-screenshot-data'
}
]
};
const result = await makeFullPageBase64Image(singleScreenshotData, defaultOptions);
expect(getBase64ScreenshotSizeMock.mock.calls).toMatchSnapshot();
expect(mockImage.crop.mock.calls).toMatchSnapshot();
expect(mockCanvas.composite.mock.calls).toMatchSnapshot();
expect(result).toMatchSnapshot();
});
it('should handle different device pixel ratios', async () => {
const result = await makeFullPageBase64Image(defaultScreenshotsData, {
...defaultOptions,
devicePixelRatio: 3
});
expect(getBase64ScreenshotSizeMock.mock.calls).toMatchSnapshot();
expect(mockImage.crop.mock.calls).toMatchSnapshot();
expect(mockCanvas.composite.mock.calls).toMatchSnapshot();
expect(result).toMatchSnapshot();
});
it('should handle screenshots with different dimensions', async () => {
const mixedScreenshotsData = {
fullPageHeight: 1500,
fullPageWidth: 1200,
data: [
{
canvasWidth: 1200,
canvasYPosition: 0,
imageHeight: 600,
imageWidth: 1200,
imageXPosition: 0,
imageYPosition: 0,
screenshot: 'wide-screenshot-data'
},
{
canvasWidth: 1200,
canvasYPosition: 600,
imageHeight: 900,
imageWidth: 1200,
imageXPosition: 0,
imageYPosition: 0,
screenshot: 'tall-screenshot-data'
}
]
};
getBase64ScreenshotSizeMock
.mockReturnValueOnce({ width: 1200, height: 600 })
.mockReturnValueOnce({ width: 1200, height: 900 });
const result = await makeFullPageBase64Image(mixedScreenshotsData, defaultOptions);
expect(getBase64ScreenshotSizeMock.mock.calls).toMatchSnapshot();
expect(mockImage.crop.mock.calls).toMatchSnapshot();
expect(mockCanvas.composite.mock.calls).toMatchSnapshot();
expect(result).toMatchSnapshot();
});
it('should handle screenshots with cropping positions', async () => {
const croppedScreenshotsData = {
fullPageHeight: 1000,
fullPageWidth: 1000,
data: [
{
canvasWidth: 1000,
canvasYPosition: 0,
imageHeight: 500,
imageWidth: 500,
imageXPosition: 100,
imageYPosition: 50,
screenshot: 'cropped-screenshot-data'
}
]
};
const result = await makeFullPageBase64Image(croppedScreenshotsData, defaultOptions);
expect(getBase64ScreenshotSizeMock.mock.calls).toMatchSnapshot();
expect(mockImage.crop.mock.calls).toMatchSnapshot();
expect(mockCanvas.composite.mock.calls).toMatchSnapshot();
expect(result).toMatchSnapshot();
});
it('should handle landscape mode without rotation when width >= height', async () => {
getBase64ScreenshotSizeMock.mockReturnValue({ width: 1000, height: 800 });
const result = await makeFullPageBase64Image(defaultScreenshotsData, {
...defaultOptions,
isLandscape: true
});
expect(getBase64ScreenshotSizeMock.mock.calls).toMatchSnapshot();
expect(mockImage.crop.mock.calls).toMatchSnapshot();
expect(mockCanvas.composite.mock.calls).toMatchSnapshot();
expect(result).toMatchSnapshot();
});
it('should handle empty screenshots array', async () => {
const emptyScreenshotsData = {
fullPageHeight: 0,
fullPageWidth: 1000,
data: []
};
const result = await makeFullPageBase64Image(emptyScreenshotsData, defaultOptions);
expect(getBase64ScreenshotSizeMock.mock.calls).toMatchSnapshot();
expect(mockImage.crop.mock.calls).toMatchSnapshot();
expect(mockCanvas.composite.mock.calls).toMatchSnapshot();
expect(mockCanvas.getBase64.mock.calls).toMatchSnapshot();
expect(result).toMatchSnapshot();
});
it('should handle large canvas dimensions', async () => {
const largeScreenshotsData = {
fullPageHeight: 5000,
fullPageWidth: 3000,
data: [
{
canvasWidth: 3000,
canvasYPosition: 0,
imageHeight: 2000,
imageWidth: 3000,
imageXPosition: 0,
imageYPosition: 0,
screenshot: 'large-screenshot-data'
}
]
};
getBase64ScreenshotSizeMock.mockReturnValue({ width: 3000, height: 2000 });
const result = await makeFullPageBase64Image(largeScreenshotsData, defaultOptions);
expect(getBase64ScreenshotSizeMock.mock.calls).toMatchSnapshot();
expect(mockImage.crop.mock.calls).toMatchSnapshot();
expect(mockCanvas.composite.mock.calls).toMatchSnapshot();
expect(result).toMatchSnapshot();
});
it('should handle different screenshot data for each iteration', async () => {
const result = await makeFullPageBase64Image(defaultScreenshotsData, defaultOptions);
expect(getBase64ScreenshotSizeMock.mock.calls).toMatchSnapshot();
expect(result).toMatchSnapshot();
});
it('should handle canvas Y positions correctly', async () => {
const result = await makeFullPageBase64Image(defaultScreenshotsData, defaultOptions);
expect(mockCanvas.composite.mock.calls).toMatchSnapshot();
expect(result).toMatchSnapshot();
});
});
describe('takeResizedBase64Screenshot', () => {
let mockBrowserInstance;
let mockElement;
let mockElementRegion;
let takeBase64ScreenshotMock;
let calculateDprDataMock;
let isWdioElementMock;
const defaultOptions = {
browserInstance: {},
element: {},
devicePixelRatio: 2,
isIOS: false,
resizeDimensions: { top: 0, right: 0, bottom: 0, left: 0 }
};
beforeEach(async () => {
mockElementRegion = {
height: 100,
width: 200,
x: 50,
y: 25
};
mockElement = {
elementId: 'test-element-id',