1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
|
use std::io::{self, Write};
use std::fs::{File, OpenOptions};
use argparse::{ArgumentParser, Store, StoreTrue, StoreOption};
use crate::error::IntoIOError;
use crate::id3v2::{ID3v2, Frame};
use crate::options::Options;
mod encoding;
mod error;
mod id3v2;
mod options;
fn parse_options_into(opt: &mut Options) {
let mut ap = ArgumentParser::new();
ap.set_description("ID3v2 tag editor/fixer. Incomplete/work-in-progress. Support for ID3v2.3, with partial support of ID3v2.4 (no footer tags supported).");
ap.refer(&mut opt.write)
.add_option(&["-w", "--write"], StoreTrue,
"Write updated information instead of just displaying. Unmodified or unknown tags are preserved as-is.");
ap.refer(&mut opt.latin1_as_utf8)
.add_option(&["--assume-utf8"], StoreTrue,
"Assume that all strings specified in the MP3 as Latin-1 are really UTF-8.");
ap.refer(&mut opt.set_tags.album) .add_option(&["-A", "--album"], StoreOption, "Set/replace album");
ap.refer(&mut opt.set_tags.artist).add_option(&["-a", "--artist"], StoreOption, "Set/replace artist");
ap.refer(&mut opt.set_tags.title) .add_option(&["-t", "--title"], StoreOption, "Set/replace title");
ap.refer(&mut opt.set_tags.track) .add_option(&["-T", "--track"], StoreOption, "Set/replace track number (num or num/num)");
ap.refer(&mut opt.set_tags.year) .add_option(&["-y", "--year"], StoreOption, "Set/replace year");
ap.refer(&mut opt.file)
.required()
.add_argument("file", Store,
"File to operate on (probably a .mp3)");
ap.parse_args_or_exit();
}
fn parse_options() -> Options {
let mut options = Default::default();
parse_options_into(&mut options);
options
}
fn print_tag(tag: &ID3v2) -> io::Result<()> {
for frame in &tag.frames {
match frame.interpret()? {
Some(Frame::TALB(album)) => println!("Album : '{}'", album),
Some(Frame::TIT2(title)) => println!("Title : '{}'", title),
Some(Frame::TPE1(artist)) => println!("Artist: '{}'", artist),
Some(Frame::TYER(year)) => println!("Year : '{}'", year),
Some(Frame::TRCK(track)) => println!("Track : '{}'", track),
None => {
println!("Unknown frame: {:?}", frame);
}
}
}
Ok(())
}
fn modify_tag(tag: &mut ID3v2, new_frame: Frame) -> io::Result<()> {
let mut indices = Vec::new();
for (i, frame) in tag.frames.iter().enumerate() {
if frame.get_id() == new_frame.id() {
indices.push(i);
}
}
match indices.len() {
0 => {
tag.frames.push(tag.to_raw(new_frame)?);
}
1 => {
tag.frames[indices[0]] = tag.to_raw(new_frame)?;
}
_ => {
return Err(format!("Multiple frames of type {} found", new_frame.id()).ioerr());
}
}
Ok(())
}
fn main() -> io::Result<()> {
let options = parse_options();
let mut tag = ID3v2::from_stream(&mut File::open(&options.file)?)?;
// println!("{:?}", tag);
if options.latin1_as_utf8 {
for frame in &mut tag.frames {
if let Some(new_frame) =
frame.map_string(|s| encoding::from_utf8_mistaken_as_latin1(&s)
.expect("String cannot be encoded in ID3v2 (outside of Unicode BMP)"))? {
*frame = new_frame;
}
}
}
if let Some(s) = options.set_tags.album { modify_tag(&mut tag, Frame::TALB(s))?; }
if let Some(s) = options.set_tags.artist { modify_tag(&mut tag, Frame::TPE1(s))?; }
if let Some(s) = options.set_tags.title { modify_tag(&mut tag, Frame::TIT2(s))?; }
if let Some(s) = options.set_tags.track { modify_tag(&mut tag, Frame::TRCK(s))?; }
if let Some(s) = options.set_tags.year { modify_tag(&mut tag, Frame::TYER(s))?; }
print_tag(&tag)?;
// TODO: if -w, then write tags to file (if it fits)
if options.write {
let encoded = tag.encode()?;
let mut f = OpenOptions::new().write(true).open(&options.file)?;
f.write_all(&encoded)?;
println!("Tag written!");
}
Ok(())
}
|