24 releases
0.12.1 | Feb 25, 2025 |
---|---|
0.11.0 | Jun 15, 2021 |
0.9.1 | Feb 14, 2021 |
0.7.3 | Dec 13, 2020 |
0.1.0 | Mar 26, 2020 |
#100 in Parser implementations
37,767 downloads per month
Used in 46 crates
(10 directly)
325KB
7K
SLoC
mp4ameta
A library for reading and writing iTunes style MPEG-4 audio metadata.
Most commonly this kind of metadata is found inside m4a
or m4b
files but basically any mp4
container supports it.
Examples
The Easy Way
let mut tag = mp4ameta::Tag::read_from_path("music.m4a").unwrap();
println!("{}", tag.artist().unwrap());
tag.set_artist("artist");
tag.write_to_path("music.m4a").unwrap();
The Hard Way
use mp4ameta::{Data, Fourcc, Tag};
let mut tag = Tag::read_from_path("music.m4a").unwrap();
let artist_ident = Fourcc(*b"\xa9ART");
let artist = tag.strings_of(&artist_ident).next().unwrap();
println!("{}", artist);
tag.set_data(artist_ident, Data::Utf8("artist".to_owned()));
tag.write_to_path("music.m4a").unwrap();
Using Freeform Identifiers
use mp4ameta::{Data, FreeformIdent, Tag};
let mut tag = Tag::read_from_path("music.m4a").unwrap();
let isrc_ident = FreeformIdent::new_static("com.apple.iTunes", "ISRC");
let isrc = tag.strings_of(&isrc_ident).next().unwrap();
println!("{}", isrc);
tag.set_data(isrc_ident, Data::Utf8("isrc".to_owned()));
tag.write_to_path("music.m4a").unwrap();
Chapters
There are two ways of storing chapters in mp4 files. They can either be stored inside a chapter list, or a chapter track.
use mp4ameta::{Chapter, Tag};
use std::time::Duration;
let mut tag = Tag::read_from_path("audiobook.m4b").unwrap();
for chapter in tag.chapter_track() {
let mins = chapter.start.as_secs() / 60;
let secs = chapter.start.as_secs() % 60;
println!("{mins:02}:{secs:02} {}", chapter.title);
}
tag.chapter_track_mut().clear();
tag.chapter_list_mut().extend([
Chapter::new(Duration::ZERO, "first chapter"),
Chapter::new(Duration::from_secs(3 * 60 + 42), "second chapter"),
Chapter::new(Duration::from_secs(7 * 60 + 13), "third chapter"),
]);
tag.write_to_path("audiobook.m4b").unwrap();
Read and Write Configurations
Read only the data that is relevant for your usecase. And (over)write only the data that you want to edit.
By default all data is read and written.
use mp4ameta::{ChplTimescale, ReadConfig, Tag, WriteConfig};
// Only read the metadata item list, not chapters or audio information
let read_cfg = ReadConfig {
read_meta_items: true,
read_image_data: false,
read_chapter_list: false,
read_chapter_track: false,
read_audio_info: false,
chpl_timescale: ChplTimescale::DEFAULT,
};
let mut tag = Tag::read_with_path("music.m4a", &read_cfg).unwrap();
println!("{tag}");
tag.clear_meta_items();
// Only overwrite the metadata item list, leave chapters intact
let write_cfg = WriteConfig {
write_meta_items: true,
write_chapter_list: false,
write_chapter_track: false,
chpl_timescale: ChplTimescale::DEFAULT,
};
tag.write_with_path("music.m4a", &write_cfg).unwrap();
Useful Links
- QuickTime spec
- MultimediaWiki QuickTime container
- AtomicParsley docs
- Mutagen docs
- Hydrogen audio tag mapping
- MusicBrainz Picard tag mapping
- Filetype list
Testing
Run all tests:
cargo test
Test this library on your collection:
cargo test -- --nocapture collection <path>