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
use std::path::PathBuf;
use std::ffi::OsString;
use lexopt::prelude::*;
#[derive(Debug, PartialEq, Eq)]
pub struct Args {
pub filename: PathBuf,
pub read: bool,
pub write: bool,
pub tag_version: Option<id3::Version>,
}
pub fn parse_args<I>(args: I) -> Result<Args, lexopt::Error>
where
I: IntoIterator + 'static,
I::Item: Into<OsString>,
{
let mut filename_input = None;
let mut tag_version = None;
let mut read = false;
let mut write = false;
let mut parser = lexopt::Parser::from_iter(args);
while let Some(arg) = parser.next()? {
match arg {
Short('r') | Long("read") => read = true,
Short('w') | Long("write") => write = true,
Long("tag-version") => {
let mut input = parser.value()?;
input.make_ascii_lowercase();
if input == "id3v2.2" {
tag_version = Some(id3::Version::Id3v22);
} else if input == "id3v2.3" {
tag_version = Some(id3::Version::Id3v23);
} else if input == "id3v2.4" {
tag_version = Some(id3::Version::Id3v24);
} else {
let error = format!("Unsupported ID3 version: {:?}. Expected ID3v2.{{2,3,4}}", input);
return Err(lexopt::Error::Custom(error.into()));
}
},
Value(val) if filename_input.is_none() => {
filename_input = Some(PathBuf::from(val));
}
Short('V') | Long("version") => {
println!("id3-json {}", env!("CARGO_PKG_VERSION"));
std::process::exit(0);
}
Long("help") => {
print_help();
std::process::exit(0);
}
_ => return Err(arg.unexpected()),
}
}
let Some(filename) = filename_input else {
let error = String::from("Missing <filename.mp3>");
return Err(lexopt::Error::Custom(error.into()));
};
if !read && !write {
read = true;
}
Ok(Args { filename, read, write, tag_version })
}
fn print_help() {
println!("id3-json {}", env!("CARGO_PKG_VERSION"));
println!();
println!("USAGE:");
println!(" id3-json [FLAGS] <music-file.mp3>");
println!();
println!("FLAGS:");
println!(" -r, --read Reads tags from the file and outputs them to STDOUT as JSON.");
println!(" If neither `read` nor `write` are given, will read by default.");
println!();
println!(" -w, --write Write mode, expects a JSON on STDIN with valid tag values.");
println!(" If also given `read`, will print the resulting tags afterwards");
println!();
println!(" --tag-version <ID3v2.{{2,3,4}}>");
println!(" On write, sets the tags' version to 2.2, 2.3, or 2.4.");
println!();
println!(" -V, --version Prints version information");
println!();
println!("ARGS:");
println!(" <music-file.mp3> Music file to read tags from or write tags to");
}