Verify ID3 Frame contents prior to writing

This commit is contained in:
Serial 2022-01-13 14:42:34 -05:00
parent e6b714eac4
commit c27264f50b
No known key found for this signature in database
GPG key ID: DA95198DC17C4568
4 changed files with 60 additions and 2 deletions

View file

@ -49,6 +49,9 @@ pub enum LoftyError {
#[cfg(feature = "id3v2")] #[cfg(feature = "id3v2")]
/// Arises when invalid data is encountered while reading an ID3v2 synchronized text frame /// Arises when invalid data is encountered while reading an ID3v2 synchronized text frame
BadSyncText, BadSyncText,
#[cfg(feature = "id3v2")]
/// Arises when attempting to write an invalid Frame (Bad `FrameID`/`FrameValue` pairing)
BadFrame(String, &'static str),
/// Arises when an atom contains invalid data /// Arises when an atom contains invalid data
BadAtom(&'static str), BadAtom(&'static str),
@ -131,6 +134,12 @@ impl Display for LoftyError {
), ),
#[cfg(feature = "id3v2")] #[cfg(feature = "id3v2")]
LoftyError::BadSyncText => write!(f, "ID3v2: Encountered invalid data in SYLT frame"), LoftyError::BadSyncText => write!(f, "ID3v2: Encountered invalid data in SYLT frame"),
#[cfg(feature = "id3v2")]
LoftyError::BadFrame(frame_id, frame_value) => write!(
f,
"ID3v2: Attempted to write an invalid frame. ID: \"{}\", Value: \"{}\"",
frame_id, frame_value
),
LoftyError::BadAtom(message) => write!(f, "MP4 Atom: {}", message), LoftyError::BadAtom(message) => write!(f, "MP4 Atom: {}", message),
// Files // Files

View file

@ -117,7 +117,7 @@ pub(super) fn parse_content(
"TXXX" => parse_user_defined(content, false)?, "TXXX" => parse_user_defined(content, false)?,
"WXXX" => parse_user_defined(content, true)?, "WXXX" => parse_user_defined(content, true)?,
"COMM" | "USLT" => parse_text_language(content, id)?, "COMM" | "USLT" => parse_text_language(content, id)?,
_ if id.starts_with('T') || id == "WFED" => parse_text(content)?, _ if id.starts_with('T') => parse_text(content)?,
// Apple proprietary frames // Apple proprietary frames
"WFED" | "GRP1" => parse_text(content)?, "WFED" | "GRP1" => parse_text(content)?,
_ if id.starts_with('W') => parse_link(content)?, _ if id.starts_with('W') => parse_link(content)?,

View file

@ -282,6 +282,7 @@ impl Id3v2Tag {
/// ///
/// * Attempting to write the tag to a format that does not support it /// * Attempting to write the tag to a format that does not support it
/// * Attempting to write an encrypted frame without a valid method symbol or data length indicator /// * Attempting to write an encrypted frame without a valid method symbol or data length indicator
/// * Attempting to write an invalid [`FrameID`]/[`FrameValue`] pairing
pub fn write_to(&self, file: &mut File) -> Result<()> { pub fn write_to(&self, file: &mut File) -> Result<()> {
Id3v2TagRef::new(self.flags, self.frames.iter().filter_map(Frame::as_opt_ref)) Id3v2TagRef::new(self.flags, self.frames.iter().filter_map(Frame::as_opt_ref))
.write_to(file) .write_to(file)
@ -576,6 +577,26 @@ mod tests {
crate::tag_utils::test_utils::verify_tag(&tag, true, true); crate::tag_utils::test_utils::verify_tag(&tag, true, true);
} }
#[test]
fn fail_write_bad_frame() {
let mut tag = Id3v2Tag::default();
tag.insert(Frame {
id: FrameID::Valid(String::from("ABCD")),
value: FrameValue::URL(String::from("FOO URL")),
flags: Default::default(),
});
let res = tag.dump_to(&mut Vec::<u8>::new());
assert!(res.is_err());
assert_eq!(
res.unwrap_err().to_string(),
String::from(
"ID3v2: Attempted to write an invalid frame. ID: \"ABCD\", Value: \"URL\""
)
);
}
#[test] #[test]
fn tag_to_id3v2() { fn tag_to_id3v2() {
fn verify_frame(tag: &Id3v2Tag, id: &str, value: &str) { fn verify_frame(tag: &Id3v2Tag, id: &str, value: &str) {

View file

@ -1,5 +1,5 @@
use crate::error::{LoftyError, Result}; use crate::error::{LoftyError, Result};
use crate::id3::v2::frame::{FrameFlags, FrameRef}; use crate::id3::v2::frame::{FrameFlags, FrameRef, FrameValue};
use crate::id3::v2::synch_u32; use crate::id3::v2::synch_u32;
use std::io::Write; use std::io::Write;
@ -14,6 +14,7 @@ where
W: Write, W: Write,
{ {
for frame in frames { for frame in frames {
verify_frame(&frame)?;
let value = frame.value.as_bytes()?; let value = frame.value.as_bytes()?;
write_frame(writer, frame.id, frame.flags, &value)?; write_frame(writer, frame.id, frame.flags, &value)?;
@ -22,6 +23,33 @@ where
Ok(()) Ok(())
} }
fn verify_frame(frame: &FrameRef) -> Result<()> {
match (frame.id, frame.value.as_ref()) {
("APIC", FrameValue::Picture { .. })
| ("USLT", FrameValue::UnSyncText(_))
| ("COMM", FrameValue::Comment(_))
| ("TXXX", FrameValue::UserText(_))
| ("WXXX", FrameValue::UserURL(_))
| (_, FrameValue::Binary(_))
| ("WFED" | "GRP1", FrameValue::Text { .. }) => Ok(()),
(id, FrameValue::Text { .. }) if id.starts_with('T') => Ok(()),
(id, FrameValue::URL(_)) if id.starts_with('W') => Ok(()),
(id, frame_value) => Err(LoftyError::BadFrame(
id.to_string(),
match frame_value {
FrameValue::Comment(_) => "Comment",
FrameValue::UnSyncText(_) => "UnSyncText",
FrameValue::Text { .. } => "Text",
FrameValue::UserText(_) => "UserText",
FrameValue::URL(_) => "URL",
FrameValue::UserURL(_) => "UserURL",
FrameValue::Picture { .. } => "Picture",
FrameValue::Binary(_) => "Binary",
},
)),
}
}
fn write_frame<W>(writer: &mut W, name: &str, flags: FrameFlags, value: &[u8]) -> Result<()> fn write_frame<W>(writer: &mut W, name: &str, flags: FrameFlags, value: &[u8]) -> Result<()>
where where
W: Write, W: Write,