Files
tranquil-pds/tests/image_processing.rs
2025-12-20 13:05:43 +02:00

267 lines
8.5 KiB
Rust

use image::{DynamicImage, ImageFormat};
use std::io::Cursor;
use tranquil_pds::image::{
DEFAULT_MAX_FILE_SIZE, ImageError, ImageProcessor, OutputFormat, THUMB_SIZE_FEED,
THUMB_SIZE_FULL,
};
fn create_test_png(width: u32, height: u32) -> Vec<u8> {
let img = DynamicImage::new_rgb8(width, height);
let mut buf = Vec::new();
img.write_to(&mut Cursor::new(&mut buf), ImageFormat::Png)
.unwrap();
buf
}
fn create_test_jpeg(width: u32, height: u32) -> Vec<u8> {
let img = DynamicImage::new_rgb8(width, height);
let mut buf = Vec::new();
img.write_to(&mut Cursor::new(&mut buf), ImageFormat::Jpeg)
.unwrap();
buf
}
fn create_test_gif(width: u32, height: u32) -> Vec<u8> {
let img = DynamicImage::new_rgb8(width, height);
let mut buf = Vec::new();
img.write_to(&mut Cursor::new(&mut buf), ImageFormat::Gif)
.unwrap();
buf
}
fn create_test_webp(width: u32, height: u32) -> Vec<u8> {
let img = DynamicImage::new_rgb8(width, height);
let mut buf = Vec::new();
img.write_to(&mut Cursor::new(&mut buf), ImageFormat::WebP)
.unwrap();
buf
}
#[test]
fn test_format_support() {
let processor = ImageProcessor::new();
let png = create_test_png(500, 500);
let result = processor.process(&png, "image/png").unwrap();
assert_eq!(result.original.width, 500);
assert_eq!(result.original.height, 500);
let jpeg = create_test_jpeg(400, 300);
let result = processor.process(&jpeg, "image/jpeg").unwrap();
assert_eq!(result.original.width, 400);
assert_eq!(result.original.height, 300);
let gif = create_test_gif(200, 200);
let result = processor.process(&gif, "image/gif").unwrap();
assert_eq!(result.original.width, 200);
let webp = create_test_webp(300, 200);
let result = processor.process(&webp, "image/webp").unwrap();
assert_eq!(result.original.width, 300);
}
#[test]
fn test_thumbnail_generation() {
let processor = ImageProcessor::new();
let small = create_test_png(100, 100);
let result = processor.process(&small, "image/png").unwrap();
assert!(
result.thumbnail_feed.is_none(),
"Small image should not get feed thumbnail"
);
assert!(
result.thumbnail_full.is_none(),
"Small image should not get full thumbnail"
);
let medium = create_test_png(500, 500);
let result = processor.process(&medium, "image/png").unwrap();
assert!(
result.thumbnail_feed.is_some(),
"Medium image should have feed thumbnail"
);
assert!(
result.thumbnail_full.is_none(),
"Medium image should NOT have full thumbnail"
);
let large = create_test_png(2000, 2000);
let result = processor.process(&large, "image/png").unwrap();
assert!(
result.thumbnail_feed.is_some(),
"Large image should have feed thumbnail"
);
assert!(
result.thumbnail_full.is_some(),
"Large image should have full thumbnail"
);
let thumb = result.thumbnail_feed.unwrap();
assert!(thumb.width <= THUMB_SIZE_FEED && thumb.height <= THUMB_SIZE_FEED);
let full = result.thumbnail_full.unwrap();
assert!(full.width <= THUMB_SIZE_FULL && full.height <= THUMB_SIZE_FULL);
let at_feed = create_test_png(THUMB_SIZE_FEED, THUMB_SIZE_FEED);
let above_feed = create_test_png(THUMB_SIZE_FEED + 1, THUMB_SIZE_FEED + 1);
assert!(
processor
.process(&at_feed, "image/png")
.unwrap()
.thumbnail_feed
.is_none()
);
assert!(
processor
.process(&above_feed, "image/png")
.unwrap()
.thumbnail_feed
.is_some()
);
let at_full = create_test_png(THUMB_SIZE_FULL, THUMB_SIZE_FULL);
let above_full = create_test_png(THUMB_SIZE_FULL + 1, THUMB_SIZE_FULL + 1);
assert!(
processor
.process(&at_full, "image/png")
.unwrap()
.thumbnail_full
.is_none()
);
assert!(
processor
.process(&above_full, "image/png")
.unwrap()
.thumbnail_full
.is_some()
);
let disabled = ImageProcessor::new().with_thumbnails(false);
let result = disabled.process(&large, "image/png").unwrap();
assert!(result.thumbnail_feed.is_none() && result.thumbnail_full.is_none());
}
#[test]
fn test_output_format_conversion() {
let png = create_test_png(300, 300);
let jpeg = create_test_jpeg(300, 300);
let webp_proc = ImageProcessor::new().with_output_format(OutputFormat::WebP);
assert_eq!(
webp_proc
.process(&png, "image/png")
.unwrap()
.original
.mime_type,
"image/webp"
);
let jpeg_proc = ImageProcessor::new().with_output_format(OutputFormat::Jpeg);
assert_eq!(
jpeg_proc
.process(&png, "image/png")
.unwrap()
.original
.mime_type,
"image/jpeg"
);
let png_proc = ImageProcessor::new().with_output_format(OutputFormat::Png);
assert_eq!(
png_proc
.process(&jpeg, "image/jpeg")
.unwrap()
.original
.mime_type,
"image/png"
);
}
#[test]
fn test_size_and_dimension_limits() {
assert_eq!(DEFAULT_MAX_FILE_SIZE, 10 * 1024 * 1024);
let max_dim = ImageProcessor::new().with_max_dimension(1000);
let large = create_test_png(2000, 2000);
let result = max_dim.process(&large, "image/png");
assert!(matches!(
result,
Err(ImageError::TooLarge {
width: 2000,
height: 2000,
max_dimension: 1000
})
));
let max_file = ImageProcessor::new().with_max_file_size(100);
let data = create_test_png(500, 500);
let result = max_file.process(&data, "image/png");
assert!(matches!(
result,
Err(ImageError::FileTooLarge { max_size: 100, .. })
));
}
#[test]
fn test_error_handling() {
let processor = ImageProcessor::new();
let result = processor.process(b"this is not an image", "application/octet-stream");
assert!(matches!(result, Err(ImageError::UnsupportedFormat(_))));
let result = processor.process(b"\x89PNG\r\n\x1a\ncorrupted data here", "image/png");
assert!(matches!(result, Err(ImageError::DecodeError(_))));
}
#[test]
fn test_aspect_ratio_preservation() {
let processor = ImageProcessor::new();
let landscape = create_test_png(1600, 800);
let result = processor.process(&landscape, "image/png").unwrap();
let thumb = result.thumbnail_full.unwrap();
let original_ratio = 1600.0 / 800.0;
let thumb_ratio = thumb.width as f64 / thumb.height as f64;
assert!((original_ratio - thumb_ratio).abs() < 0.1);
let portrait = create_test_png(800, 1600);
let result = processor.process(&portrait, "image/png").unwrap();
let thumb = result.thumbnail_full.unwrap();
let original_ratio = 800.0 / 1600.0;
let thumb_ratio = thumb.width as f64 / thumb.height as f64;
assert!((original_ratio - thumb_ratio).abs() < 0.1);
}
#[test]
fn test_utilities_and_builder() {
assert!(ImageProcessor::is_supported_mime_type("image/jpeg"));
assert!(ImageProcessor::is_supported_mime_type("image/jpg"));
assert!(ImageProcessor::is_supported_mime_type("image/png"));
assert!(ImageProcessor::is_supported_mime_type("image/gif"));
assert!(ImageProcessor::is_supported_mime_type("image/webp"));
assert!(ImageProcessor::is_supported_mime_type("IMAGE/PNG"));
assert!(ImageProcessor::is_supported_mime_type("Image/Jpeg"));
assert!(!ImageProcessor::is_supported_mime_type("image/bmp"));
assert!(!ImageProcessor::is_supported_mime_type("image/tiff"));
assert!(!ImageProcessor::is_supported_mime_type("text/plain"));
let data = create_test_png(100, 100);
let processor = ImageProcessor::new();
let result = processor.process(&data, "application/octet-stream");
assert!(result.is_ok(), "Should detect PNG format from data");
let jpeg = create_test_jpeg(100, 100);
let stripped = ImageProcessor::strip_exif(&jpeg).unwrap();
assert!(!stripped.is_empty());
let processor = ImageProcessor::new()
.with_max_dimension(2048)
.with_max_file_size(5 * 1024 * 1024)
.with_output_format(OutputFormat::Jpeg)
.with_thumbnails(true);
let data = create_test_png(500, 500);
let result = processor.process(&data, "image/png").unwrap();
assert_eq!(result.original.mime_type, "image/jpeg");
assert!(!result.original.data.is_empty());
assert!(result.original.width > 0 && result.original.height > 0);
}