Add PictureInformation tests

This commit is contained in:
Serial 2021-11-22 17:14:02 -05:00
parent bae894ccef
commit 9878e89b33
7 changed files with 92 additions and 16 deletions

3
.gitignore vendored
View file

@ -11,4 +11,5 @@
**/Cargo.lock **/Cargo.lock
# Test assets # Test assets
/tests/assets/ /tests/assets/
/tests/picture/assets/

View file

@ -239,11 +239,11 @@ impl PictureInformation {
pub fn from_picture(picture: &Picture) -> Result<Self> { pub fn from_picture(picture: &Picture) -> Result<Self> {
let reader = &mut &*picture.data; let reader = &mut &*picture.data;
let mut identifier = [0; 4]; if reader.len() < 4 {
return Err(LoftyError::NotAPicture);
}
reader.read_exact(&mut identifier)?; match reader[..4] {
match identifier {
[0x89, b'P', b'N', b'G'] => Ok(Self::from_png(reader).unwrap_or_default()), [0x89, b'P', b'N', b'G'] => Ok(Self::from_png(reader).unwrap_or_default()),
[0xFF, 0xD8, 0xFF, ..] => Ok(Self::from_jpeg(reader).unwrap_or_default()), [0xFF, 0xD8, 0xFF, ..] => Ok(Self::from_jpeg(reader).unwrap_or_default()),
_ => Err(LoftyError::UnsupportedPicture), _ => Err(LoftyError::UnsupportedPicture),
@ -251,17 +251,18 @@ impl PictureInformation {
} }
pub fn from_png(reader: &mut &[u8]) -> Result<Self> { pub fn from_png(reader: &mut &[u8]) -> Result<Self> {
let mut remaining_sig = [0; 4]; let mut sig = [0; 8];
reader.read_exact(&mut remaining_sig)?; reader.read_exact(&mut sig)?;
if remaining_sig != [0x0D, 0x0A, 0x1A, 0x0A] { if sig != [0x89, b'P', b'N', b'G', 0x0D, 0x0A, 0x1A, 0x0A] {
return Err(LoftyError::NotAPicture); return Err(LoftyError::NotAPicture);
} }
reader.read_exact(&mut remaining_sig)?; let mut ihdr = [0; 8];
reader.read_exact(&mut ihdr)?;
// Verify the signature is immediately followed by the IHDR chunk // Verify the signature is immediately followed by the IHDR chunk
if remaining_sig != [0x49, 0x48, 0x44, 0x52] { if !ihdr.ends_with(&[0x49, 0x48, 0x44, 0x52]) {
return Err(LoftyError::NotAPicture); return Err(LoftyError::NotAPicture);
} }
@ -284,11 +285,12 @@ impl PictureInformation {
let mut reader = Cursor::new(reader); let mut reader = Cursor::new(reader);
// Skip 3 bytes // Skip 7 bytes
// Compression method (1) // Compression method (1)
// Filter method (1) // Filter method (1)
// Interlace method (1) // Interlace method (1)
reader.seek(SeekFrom::Current(3))?; // CRC (4)
reader.seek(SeekFrom::Current(7))?;
let mut num_colors = 0; let mut num_colors = 0;
let mut chunk_type = [0; 4]; let mut chunk_type = [0; 4];
@ -316,16 +318,30 @@ impl PictureInformation {
} }
pub fn from_jpeg(reader: &mut &[u8]) -> Result<Self> { pub fn from_jpeg(reader: &mut &[u8]) -> Result<Self> {
let mut marker = [0; 4];
reader.read_exact(&mut marker)?;
if !matches!(marker, [0xFF, 0xD8, 0xFF, ..]) {
return Err(LoftyError::NotAPicture);
}
let mut section_len = reader.read_u16::<BigEndian>()?; let mut section_len = reader.read_u16::<BigEndian>()?;
let mut reader = Cursor::new(reader); let mut reader = Cursor::new(reader);
reader.seek(SeekFrom::Current(i64::from(section_len + 2)))?; // The length contains itself
reader.seek(SeekFrom::Current(i64::from(section_len - 2)))?;
while let Ok(0xFF) = reader.read_u8() { while let Ok(0xFF) = reader.read_u8() {
let marker = reader.read_u8()?; let marker = reader.read_u8()?;
section_len = reader.read_u16::<BigEndian>()?; section_len = reader.read_u16::<BigEndian>()?;
// This marks the SOS (Start of Scan), which is
// the end of the header
if marker == 0xDA {
break;
}
// We are looking for a frame with a "SOFn" marker, // We are looking for a frame with a "SOFn" marker,
// with `n` either being 0 or 2. Since there isn't a // with `n` either being 0 or 2. Since there isn't a
// header like PNG, we actually need to search for this // header like PNG, we actually need to search for this
@ -344,7 +360,7 @@ impl PictureInformation {
}); });
} }
reader.seek(SeekFrom::Current(i64::from(section_len + 2)))?; reader.seek(SeekFrom::Current(i64::from(section_len - 2)))?;
} }
Err(LoftyError::NotAPicture) Err(LoftyError::NotAPicture)
@ -426,7 +442,7 @@ impl Picture {
text_encoding, text_encoding,
true, true,
))? ))?
}, }
None => data.write_u8(0)?, None => data.write_u8(0)?,
} }
@ -473,7 +489,7 @@ impl Picture {
return Err(LoftyError::BadPictureFormat( return Err(LoftyError::BadPictureFormat(
String::from_utf8_lossy(&format).to_string(), String::from_utf8_lossy(&format).to_string(),
)) ))
}, }
} }
} else { } else {
(crate::logic::id3::v2::util::text_utils::decode_text(&mut cursor, encoding, true)?) (crate::logic::id3::v2::util::text_utils::decode_text(&mut cursor, encoding, true)?)

Binary file not shown.

After

Width:  |  Height:  |  Size: 41 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 52 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 12 KiB

View file

@ -0,0 +1,58 @@
use lofty::picture::PictureInformation;
use std::fs::File;
use std::io::Read;
#[test]
fn read_png() {
// 640x628
let mut f = File::open("tests/picture/assets/png_640x628.png").unwrap();
let mut buf = Vec::new();
f.read_to_end(&mut buf).unwrap();
let information = PictureInformation::from_png(&mut &*buf).unwrap();
assert_eq!(information.width, 640);
assert_eq!(information.height, 628);
assert_eq!(information.color_depth, 8);
// No PLTE chunk
assert_eq!(information.num_colors, 0);
}
#[test]
fn read_png_plte() {
// PNG image with a PLTE chunk (indexed color)
let mut f = File::open("tests/picture/assets/png_640x628_plte.png").unwrap();
let mut buf = Vec::new();
f.read_to_end(&mut buf).unwrap();
let information = PictureInformation::from_png(&mut &*buf).unwrap();
assert_eq!(information.width, 640);
assert_eq!(information.height, 628);
assert_eq!(information.color_depth, 8);
// This field is actually filled since we
// have a PLTE chunk
assert_eq!(information.num_colors, 118);
}
#[test]
fn read_jpeg() {
let mut f = File::open("tests/picture/assets/jpeg_640x628.jpg").unwrap();
let mut buf = Vec::new();
f.read_to_end(&mut buf).unwrap();
let information = PictureInformation::from_jpeg(&mut &*buf).unwrap();
assert_eq!(information.width, 640);
assert_eq!(information.height, 628);
assert_eq!(information.color_depth, 24);
// Always 0, not applicable for JPEG
assert_eq!(information.num_colors, 0);
}

1
tests/picture/main.rs Normal file
View file

@ -0,0 +1 @@
mod information;