#audio-metadata #metadata #mp4 #parser #m4a

mp4ameta

A library for reading and writing iTunes style MPEG-4 audio metadata

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

Download history 8604/week @ 2024-11-06 7492/week @ 2024-11-13 10537/week @ 2024-11-20 11605/week @ 2024-11-27 8683/week @ 2024-12-04 9459/week @ 2024-12-11 6930/week @ 2024-12-18 4925/week @ 2024-12-25 5389/week @ 2025-01-01 7148/week @ 2025-01-08 7487/week @ 2025-01-15 11012/week @ 2025-01-22 11706/week @ 2025-01-29 11400/week @ 2025-02-05 11638/week @ 2025-02-12

37,767 downloads per month
Used in 46 crates (10 directly)

MIT/Apache

325KB
7K SLoC

mp4ameta

Build Crate Documentation License LOC

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();

Testing

Run all tests:
cargo test

Test this library on your collection:
cargo test -- --nocapture collection <path>

No runtime deps