#![warn(missing_docs)] //! A command-line tool to embed images into mp3 files. The real work is done by the "id3" crate, //! but this project makes it easier to deal with embedded cover art in particular. use std::path::Path; use anyhow::anyhow; use image::DynamicImage; /// Embed the image from `image_filename` into `music_filename`, in-place. Any errors reading ID3 /// tags from the music file or parsing the image get propagated upwards. /// /// The image is encoded as a JPEG with a 90% quality setting, and embedded as a "Front cover". /// Tags get written as ID3v2.3. /// pub fn embed_image(music_filename: &Path, image_filename: &Path) -> anyhow::Result<()> { let image = image::open(&image_filename). map_err(|e| anyhow!("Error reading image {:?}: {}", image_filename, e))?; embed_image_from_memory(music_filename, &image) } /// Embed the image `image` into `music_filename`, in-place. Any errors reading ID3 /// tags from the music file get propagated upwards. /// /// The image is encoded as a JPEG with a 90% quality setting, and embedded as a "Front cover". /// Tags get written as ID3v2.3. /// pub fn embed_image_from_memory(music_filename: &Path, image: &image::DynamicImage) -> anyhow::Result<()> { let mut tag = read_tag(music_filename)?; let mut encoded_image_bytes = Vec::new(); // Unwrap: Writing to a Vec should always succeed; image.write_to(&mut encoded_image_bytes, image::ImageOutputFormat::Jpeg(90)).unwrap(); tag.add_picture(id3::frame::Picture { mime_type: "image/jpeg".to_string(), picture_type: id3::frame::PictureType::CoverFront, description: String::new(), data: encoded_image_bytes, }); tag.write_to_path(music_filename, id3::Version::Id3v23). map_err(|e| anyhow!("Error writing image to music file {:?}: {}", music_filename, e))?; Ok(()) } /// Extract the first found embedded image from `music_filename` and write it as a file with the /// given `image_filename`. The image file will be silently overwritten if it exists. /// /// Any errors from parsing id3 tags will be propagated. The function will also return an error if /// there's no embedded images in the mp3 file. /// pub fn extract_first_image(music_filename: &Path, image_filename: &Path) -> anyhow::Result<()> { extract_first_image_as_img(music_filename)? .save(&image_filename). map_err(|e| anyhow!("Couldn't write image file {:?}: {}", image_filename, e)) } /// Extract the first found embedded image from `music_filename` and return it as image object /// /// Any errors from parsing id3 tags will be propagated. The function will also return an error if /// there's no embedded images in the mp3 file. /// pub fn extract_first_image_as_img(music_filename: &Path) -> anyhow::Result { let tag = read_tag(music_filename)?; let first_picture = tag.pictures().next(); if let Some(p) = first_picture { return image::load_from_memory(&p.data) .map_err(|e| anyhow!("Couldn't load image: {}", e)); } else { Err(anyhow!("No image found in music file")) } } /// Remove all embedded images from the given `music_filename`. In effect, this removes all tags of /// type "APIC". /// /// If the mp3 file's ID3 tags can't be parsed, the error will be propagated upwards. /// pub fn remove_images(music_filename: &Path) -> anyhow::Result<()> { let mut tag = read_tag(music_filename)?; tag.remove("APIC"); tag.write_to_path(music_filename, id3::Version::Id3v23). map_err(|e| anyhow!("Error updating music file {:?}: {}", music_filename, e))?; Ok(()) } fn read_tag(path: &Path) -> anyhow::Result { id3::Tag::read_from_path(&path).or_else(|e| { eprintln!("Warning: file metadata is corrupted, trying to read partial tag: {}", path.display()); e.partial_tag.clone().ok_or_else(|| anyhow!("Error reading music file {:?}: {}", path, e)) }) }