192 lines
6.0 KiB
Rust
192 lines
6.0 KiB
Rust
use predicates::prelude::*;
|
|
use predicates::reflection::{Case, Child, PredicateReflection, Product};
|
|
use std::fmt;
|
|
use std::io::{BufReader, Cursor};
|
|
use std::path::{Path, PathBuf};
|
|
|
|
use rsvg::rsvg_convert_only::{SharedImageSurface, SurfaceType};
|
|
|
|
use rsvg::test_utils::compare_surfaces::BufferDiff;
|
|
use rsvg::test_utils::reference_utils::{Compare, Deviation, Reference, surface_from_png};
|
|
|
|
/// Checks that the variable of type [u8] can be parsed as a PNG file.
|
|
#[derive(Debug)]
|
|
pub struct PngPredicate {}
|
|
|
|
impl PngPredicate {
|
|
pub fn with_size(self, w: u32, h: u32) -> SizePredicate<Self> {
|
|
SizePredicate::<Self> { p: self, w, h }
|
|
}
|
|
|
|
pub fn with_contents<P: AsRef<Path>>(self, reference: P) -> ReferencePredicate<Self> {
|
|
let mut path = PathBuf::new();
|
|
path.push(reference);
|
|
ReferencePredicate::<Self> { p: self, path }
|
|
}
|
|
}
|
|
|
|
impl Predicate<[u8]> for PngPredicate {
|
|
fn eval(&self, data: &[u8]) -> bool {
|
|
let decoder = png::Decoder::new(Cursor::new(data));
|
|
decoder.read_info().is_ok()
|
|
}
|
|
|
|
fn find_case<'a>(&'a self, _expected: bool, data: &[u8]) -> Option<Case<'a>> {
|
|
let decoder = png::Decoder::new(Cursor::new(data));
|
|
match decoder.read_info() {
|
|
Ok(_) => None,
|
|
Err(e) => Some(Case::new(Some(self), false).add_product(Product::new("Error", e))),
|
|
}
|
|
}
|
|
}
|
|
|
|
impl PredicateReflection for PngPredicate {}
|
|
|
|
impl fmt::Display for PngPredicate {
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
|
write!(f, "is a PNG")
|
|
}
|
|
}
|
|
|
|
/// Extends a PngPredicate by a check for a given size of the PNG file.
|
|
#[derive(Debug)]
|
|
pub struct SizePredicate<PngPredicate> {
|
|
p: PngPredicate,
|
|
w: u32,
|
|
h: u32,
|
|
}
|
|
|
|
impl SizePredicate<PngPredicate> {
|
|
fn eval_info(&self, info: &png::Info) -> bool {
|
|
info.width == self.w && info.height == self.h
|
|
}
|
|
|
|
fn find_case_for_info<'a>(&'a self, expected: bool, info: &png::Info) -> Option<Case<'a>> {
|
|
if self.eval_info(info) == expected {
|
|
let product = self.product_for_info(info);
|
|
Some(Case::new(Some(self), false).add_product(product))
|
|
} else {
|
|
None
|
|
}
|
|
}
|
|
|
|
fn product_for_info(&self, info: &png::Info) -> Product {
|
|
let actual_size = format!("{} x {}", info.width, info.height);
|
|
Product::new("actual size", actual_size)
|
|
}
|
|
}
|
|
|
|
impl Predicate<[u8]> for SizePredicate<PngPredicate> {
|
|
fn eval(&self, data: &[u8]) -> bool {
|
|
let decoder = png::Decoder::new(Cursor::new(data));
|
|
match decoder.read_info() {
|
|
Ok(reader) => self.eval_info(reader.info()),
|
|
_ => false,
|
|
}
|
|
}
|
|
|
|
fn find_case<'a>(&'a self, expected: bool, data: &[u8]) -> Option<Case<'a>> {
|
|
let decoder = png::Decoder::new(Cursor::new(data));
|
|
match decoder.read_info() {
|
|
Ok(reader) => self.find_case_for_info(expected, reader.info()),
|
|
Err(e) => Some(Case::new(Some(self), false).add_product(Product::new("Error", e))),
|
|
}
|
|
}
|
|
}
|
|
|
|
impl PredicateReflection for SizePredicate<PngPredicate> {
|
|
fn children<'a>(&'a self) -> Box<dyn Iterator<Item = Child<'a>> + 'a> {
|
|
let params = vec![Child::new("predicate", &self.p)];
|
|
Box::new(params.into_iter())
|
|
}
|
|
}
|
|
|
|
impl fmt::Display for SizePredicate<PngPredicate> {
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
|
write!(f, "is a PNG with size {} x {}", self.w, self.h)
|
|
}
|
|
}
|
|
|
|
/// Extends a PngPredicate by a comparison to the contents of a reference file
|
|
#[derive(Debug)]
|
|
pub struct ReferencePredicate<PngPredicate> {
|
|
p: PngPredicate,
|
|
path: PathBuf,
|
|
}
|
|
|
|
impl ReferencePredicate<PngPredicate> {
|
|
fn diff_acceptable(diff: &BufferDiff) -> bool {
|
|
match diff {
|
|
BufferDiff::DifferentSizes => false,
|
|
BufferDiff::Diff(diff) => !diff.inacceptable(),
|
|
}
|
|
}
|
|
|
|
fn diff_surface(&self, surface: &SharedImageSurface) -> Option<BufferDiff> {
|
|
let reference = Reference::from_png(&self.path);
|
|
if let Ok(diff) = reference.compare(surface) {
|
|
if !Self::diff_acceptable(&diff) {
|
|
return Some(diff);
|
|
}
|
|
}
|
|
None
|
|
}
|
|
|
|
fn find_case_for_surface<'a>(
|
|
&'a self,
|
|
expected: bool,
|
|
surface: &SharedImageSurface,
|
|
) -> Option<Case<'a>> {
|
|
let diff = self.diff_surface(surface);
|
|
if diff.is_some() != expected {
|
|
let product = self.product_for_diff(&diff.unwrap());
|
|
Some(Case::new(Some(self), false).add_product(product))
|
|
} else {
|
|
None
|
|
}
|
|
}
|
|
|
|
fn product_for_diff(&self, diff: &BufferDiff) -> Product {
|
|
let difference = format!("{}", diff);
|
|
Product::new("images differ", difference)
|
|
}
|
|
}
|
|
|
|
impl Predicate<[u8]> for ReferencePredicate<PngPredicate> {
|
|
fn eval(&self, data: &[u8]) -> bool {
|
|
if let Ok(surface) = surface_from_png(&mut BufReader::new(data)) {
|
|
let surface = SharedImageSurface::wrap(surface, SurfaceType::SRgb).unwrap();
|
|
self.diff_surface(&surface).is_some()
|
|
} else {
|
|
false
|
|
}
|
|
}
|
|
|
|
fn find_case<'a>(&'a self, expected: bool, data: &[u8]) -> Option<Case<'a>> {
|
|
match surface_from_png(&mut BufReader::new(data)) {
|
|
Ok(surface) => {
|
|
let surface = SharedImageSurface::wrap(surface, SurfaceType::SRgb).unwrap();
|
|
self.find_case_for_surface(expected, &surface)
|
|
}
|
|
Err(e) => Some(Case::new(Some(self), false).add_product(Product::new("Error", e))),
|
|
}
|
|
}
|
|
}
|
|
|
|
impl PredicateReflection for ReferencePredicate<PngPredicate> {
|
|
fn children<'a>(&'a self) -> Box<dyn Iterator<Item = Child<'a>> + 'a> {
|
|
let params = vec![Child::new("predicate", &self.p)];
|
|
Box::new(params.into_iter())
|
|
}
|
|
}
|
|
|
|
impl fmt::Display for ReferencePredicate<PngPredicate> {
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
|
write!(
|
|
f,
|
|
"is a PNG that matches the reference {}",
|
|
self.path.display()
|
|
)
|
|
}
|
|
}
|