mirror of
https://github.com/Serial-ATA/lofty-rs
synced 2024-12-13 14:12:31 +00:00
Add PictureInformation tests
This commit is contained in:
parent
bae894ccef
commit
9878e89b33
7 changed files with 92 additions and 16 deletions
3
.gitignore
vendored
3
.gitignore
vendored
|
@ -11,4 +11,5 @@
|
||||||
**/Cargo.lock
|
**/Cargo.lock
|
||||||
|
|
||||||
# Test assets
|
# Test assets
|
||||||
/tests/assets/
|
/tests/assets/
|
||||||
|
/tests/picture/assets/
|
||||||
|
|
|
@ -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)?)
|
||||||
|
|
BIN
tests/picture/assets/jpeg_640x628.jpg
Normal file
BIN
tests/picture/assets/jpeg_640x628.jpg
Normal file
Binary file not shown.
After Width: | Height: | Size: 41 KiB |
BIN
tests/picture/assets/png_640x628.png
Normal file
BIN
tests/picture/assets/png_640x628.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 52 KiB |
BIN
tests/picture/assets/png_640x628_plte.png
Normal file
BIN
tests/picture/assets/png_640x628_plte.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 12 KiB |
58
tests/picture/information.rs
Normal file
58
tests/picture/information.rs
Normal 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
1
tests/picture/main.rs
Normal file
|
@ -0,0 +1 @@
|
||||||
|
mod information;
|
Loading…
Reference in a new issue