25 unstable releases
Uses new Rust 2024
| 0.13.0 | May 29, 2025 |
|---|---|
| 0.12.1 | Feb 25, 2025 |
| 0.11.0 | Jun 15, 2021 |
| 0.9.1 | Feb 14, 2021 |
| 0.1.0 | Mar 26, 2020 |
#21 in Audio
22,928 downloads per month
Used in 51 crates
(13 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>