audiobook_forge/cli/
commands.rs1use clap::{Parser, Subcommand, Args};
4use std::path::PathBuf;
5
6use crate::VERSION;
7
8#[derive(Parser)]
10#[command(name = "audiobook-forge")]
11#[command(version = VERSION)]
12#[command(about = "Convert audiobook directories to M4B format with chapters and metadata")]
13#[command(long_about = "
14Audiobook Forge is a CLI tool that converts audiobook directories containing
15MP3 files into high-quality M4B audiobook files with proper chapters and metadata.
16
17Features:
18• Automatic quality detection and preservation
19• Smart chapter generation from multiple sources
20• Parallel batch processing
21• Metadata extraction and enhancement
22• Cover art embedding
23")]
24pub struct Cli {
25 #[command(subcommand)]
26 pub command: Commands,
27
28 #[arg(global = true, short, long)]
30 pub verbose: bool,
31}
32
33#[derive(Subcommand)]
34pub enum Commands {
35 Build(BuildArgs),
37
38 Organize(OrganizeArgs),
40
41 #[command(subcommand)]
43 Config(ConfigCommands),
44
45 #[command(subcommand)]
47 Metadata(MetadataCommands),
48
49 Match(MatchArgs),
51
52 Check,
54
55 Version,
57}
58
59#[derive(Args)]
60pub struct BuildArgs {
61 #[arg(short, long)]
63 pub root: Option<PathBuf>,
64
65 #[arg(short, long)]
67 pub out: Option<PathBuf>,
68
69 #[arg(short = 'j', long, value_parser = clap::value_parser!(u8).range(1..=8))]
71 pub parallel: Option<u8>,
72
73 #[arg(long)]
75 pub skip_existing: Option<bool>,
76
77 #[arg(long)]
79 pub force: bool,
80
81 #[arg(long)]
83 pub normalize: bool,
84
85 #[arg(long)]
87 pub dry_run: bool,
88
89 #[arg(long)]
91 pub prefer_stereo: Option<bool>,
92
93 #[arg(long, value_parser = ["auto", "files", "cue", "id3", "none"])]
95 pub chapter_source: Option<String>,
96
97 #[arg(long)]
99 pub cover_names: Option<String>,
100
101 #[arg(long)]
103 pub language: Option<String>,
104
105 #[arg(long)]
107 pub keep_temp: bool,
108
109 #[arg(long)]
111 pub delete_originals: bool,
112
113 #[arg(long)]
115 pub use_apple_silicon_encoder: Option<bool>,
116
117 #[arg(long)]
119 pub fetch_audible: bool,
120
121 #[arg(long)]
123 pub audible_region: Option<String>,
124
125 #[arg(long)]
127 pub audible_auto_match: bool,
128
129 #[arg(long)]
131 pub config: Option<PathBuf>,
132}
133
134#[derive(Args)]
135pub struct OrganizeArgs {
136 #[arg(short, long)]
138 pub root: Option<PathBuf>,
139
140 #[arg(long)]
142 pub dry_run: bool,
143
144 #[arg(long)]
146 pub config: Option<PathBuf>,
147}
148
149#[derive(Subcommand)]
150pub enum ConfigCommands {
151 Init {
153 #[arg(long)]
155 force: bool,
156 },
157
158 Show {
160 #[arg(long)]
162 config: Option<PathBuf>,
163 },
164
165 Validate {
167 #[arg(long)]
169 config: Option<PathBuf>,
170 },
171
172 Path,
174
175 Edit,
177}
178
179#[derive(Subcommand)]
180pub enum MetadataCommands {
181 Fetch {
183 #[arg(long)]
185 asin: Option<String>,
186
187 #[arg(long)]
189 title: Option<String>,
190
191 #[arg(long)]
193 author: Option<String>,
194
195 #[arg(long, default_value = "us")]
197 region: String,
198
199 #[arg(long)]
201 output: Option<PathBuf>,
202 },
203
204 Enrich {
206 #[arg(long)]
208 file: PathBuf,
209
210 #[arg(long)]
212 asin: Option<String>,
213
214 #[arg(long)]
216 auto_detect: bool,
217
218 #[arg(long, default_value = "us")]
220 region: String,
221 },
222}
223
224#[derive(Args)]
226pub struct MatchArgs {
227 #[arg(long, short = 'f', conflicts_with = "dir")]
229 pub file: Option<PathBuf>,
230
231 #[arg(long, short = 'd', conflicts_with = "file")]
233 pub dir: Option<PathBuf>,
234
235 #[arg(long)]
237 pub title: Option<String>,
238
239 #[arg(long)]
241 pub author: Option<String>,
242
243 #[arg(long)]
245 pub auto: bool,
246
247 #[arg(long, default_value = "us")]
249 pub region: String,
250
251 #[arg(long)]
253 pub keep_cover: bool,
254
255 #[arg(long)]
257 pub dry_run: bool,
258}