@@ -6,12 +6,16 @@ use clap::builder::PossibleValuesParser;
6
6
use clap:: builder:: TypedValueParser as _;
7
7
use clap:: { value_parser, Args , Command , Parser , ValueHint } ;
8
8
use clap_complete:: { generate, Generator , Shell } ;
9
+ use merge:: Merge ;
9
10
use num_format:: CustomFormat ;
10
11
use onefetch_image:: ImageProtocol ;
11
12
use onefetch_manifest:: ManifestType ;
12
13
use regex:: Regex ;
13
- use serde:: Serialize ;
14
+ use serde:: de:: Visitor ;
15
+ use serde:: { Deserialize , Serialize , Serializer } ;
16
+ use serde_json:: error;
14
17
use std:: env;
18
+ use std:: fmt:: Write ;
15
19
use std:: io;
16
20
use std:: path:: PathBuf ;
17
21
use std:: str:: FromStr ;
@@ -20,12 +24,17 @@ use strum::IntoEnumIterator;
20
24
const COLOR_RESOLUTIONS : [ & str ; 5 ] = [ "16" , "32" , "64" , "128" , "256" ] ;
21
25
pub const NO_BOTS_DEFAULT_REGEX_PATTERN : & str = r"(?:-|\s)[Bb]ot$|\[[Bb]ot\]" ;
22
26
23
- #[ derive( Clone , Debug , Parser , PartialEq , Eq ) ]
27
+ #[ derive( Clone , Debug , Parser , PartialEq , Eq , Deserialize , Serialize , Merge ) ]
24
28
#[ command( version, about) ]
25
29
pub struct CliOptions {
26
30
/// Run as if onefetch was started in <input> instead of the current working directory
27
31
#[ arg( default_value = "." , hide_default_value = true , value_hint = ValueHint :: DirPath ) ]
32
+ #[ merge( skip) ]
28
33
pub input : PathBuf ,
34
+ /// Specify custom path for config file. Default is $XDG_CONFIG_HOME/onefetch/config.toml
35
+ #[ arg( long, value_hint = ValueHint :: FilePath ) ]
36
+ #[ merge( skip) ]
37
+ pub config_path : Option < PathBuf > ,
29
38
#[ command( flatten) ]
30
39
pub info : InfoCliOptions ,
31
40
#[ command( flatten) ]
@@ -42,7 +51,7 @@ pub struct CliOptions {
42
51
pub other : OtherCliOptions ,
43
52
}
44
53
45
- #[ derive( Clone , Debug , Args , PartialEq , Eq ) ]
54
+ #[ derive( Clone , Debug , Args , PartialEq , Eq , Merge , Deserialize , Serialize ) ]
46
55
#[ command( next_help_heading = "INFO" ) ]
47
56
pub struct InfoCliOptions {
48
57
/// Allows you to disable FIELD(s) from appearing in the output
@@ -54,18 +63,23 @@ pub struct InfoCliOptions {
54
63
value_enum,
55
64
value_name = "FIELD"
56
65
) ]
66
+ #[ merge( strategy = overwrite_vector) ]
57
67
pub disabled_fields : Vec < InfoType > ,
58
68
/// Hides the title
59
69
#[ arg( long) ]
70
+ #[ merge( strategy = merge:: bool :: overwrite_false) ]
60
71
pub no_title : bool ,
61
72
/// Maximum NUM of authors to be shown
62
73
#[ arg( long, default_value_t = 3usize , value_name = "NUM" ) ]
74
+ #[ merge( strategy = overwrite) ]
63
75
pub number_of_authors : usize ,
64
76
/// Maximum NUM of languages to be shown
65
77
#[ arg( long, default_value_t = 6usize , value_name = "NUM" ) ]
78
+ #[ merge( strategy = overwrite) ]
66
79
pub number_of_languages : usize ,
67
80
/// Maximum NUM of file churns to be shown
68
81
#[ arg( long, default_value_t = 3usize , value_name = "NUM" ) ]
82
+ #[ merge( strategy = overwrite) ]
69
83
pub number_of_file_churns : usize ,
70
84
/// Minimum NUM of commits from HEAD used to compute the churn summary
71
85
///
@@ -75,6 +89,7 @@ pub struct InfoCliOptions {
75
89
pub churn_pool_size : Option < usize > ,
76
90
/// Ignore all files & directories matching EXCLUDE
77
91
#[ arg( long, short, num_args = 1 ..) ]
92
+ #[ merge( strategy = overwrite_vector) ]
78
93
pub exclude : Vec < String > ,
79
94
/// Exclude [bot] commits. Use <REGEX> to override the default pattern
80
95
#[ arg(
@@ -84,21 +99,27 @@ pub struct InfoCliOptions {
84
99
default_missing_value = NO_BOTS_DEFAULT_REGEX_PATTERN ,
85
100
value_name = "REGEX"
86
101
) ]
102
+ #[ merge( strategy = overwrite) ]
87
103
pub no_bots : Option < MyRegex > ,
88
104
/// Ignores merge commits
89
105
#[ arg( long) ]
106
+ #[ merge( strategy = merge:: bool :: overwrite_false) ]
90
107
pub no_merges : bool ,
91
108
/// Show the email address of each author
92
109
#[ arg( long, short = 'E' ) ]
110
+ #[ merge( strategy = merge:: bool :: overwrite_false) ]
93
111
pub email : bool ,
94
112
/// Display repository URL as HTTP
95
113
#[ arg( long) ]
114
+ #[ merge( strategy = merge:: bool :: overwrite_false) ]
96
115
pub http_url : bool ,
97
116
/// Hide token in repository URL
98
117
#[ arg( long) ]
118
+ #[ merge( strategy = merge:: bool :: overwrite_false) ]
99
119
pub hide_token : bool ,
100
120
/// Count hidden files and directories
101
121
#[ arg( long) ]
122
+ #[ merge( strategy = merge:: bool :: overwrite_false) ]
102
123
pub include_hidden : bool ,
103
124
/// Filters output by language type
104
125
#[ arg(
@@ -108,10 +129,11 @@ pub struct InfoCliOptions {
108
129
short = 'T' ,
109
130
value_enum,
110
131
) ]
132
+ #[ merge( strategy = overwrite_vector) ]
111
133
pub r#type : Vec < LanguageType > ,
112
134
}
113
135
114
- #[ derive( Clone , Debug , Args , PartialEq , Eq ) ]
136
+ #[ derive( Clone , Debug , Args , PartialEq , Eq , Merge , Deserialize , Serialize ) ]
115
137
#[ command( next_help_heading = "ASCII" ) ]
116
138
pub struct AsciiCliOptions {
117
139
/// Takes a non-empty STRING as input to replace the ASCII logo
@@ -131,6 +153,7 @@ pub struct AsciiCliOptions {
131
153
short = 'c' ,
132
154
value_parser = value_parser!( u8 ) . range( ..16 ) ,
133
155
) ]
156
+ #[ merge( strategy = overwrite_vector) ]
134
157
pub ascii_colors : Vec < u8 > ,
135
158
/// Which LANGUAGE's ascii art to print
136
159
#[ arg(
@@ -140,22 +163,25 @@ pub struct AsciiCliOptions {
140
163
value_enum,
141
164
hide_possible_values = true
142
165
) ]
166
+ #[ serde( skip) ]
143
167
pub ascii_language : Option < Language > ,
144
168
/// Specify when to use true color
145
169
///
146
170
/// If set to auto: true color will be enabled if supported by the terminal
147
171
#[ arg( long, default_value = "auto" , value_name = "WHEN" , value_enum) ]
172
+ #[ merge( strategy = overwrite) ]
148
173
pub true_color : When ,
149
174
}
150
175
151
- #[ derive( Clone , Debug , Args , PartialEq , Eq ) ]
176
+ #[ derive( Clone , Debug , Args , PartialEq , Eq , Merge , Deserialize , Serialize ) ]
152
177
#[ command( next_help_heading = "IMAGE" ) ]
153
178
pub struct ImageCliOptions {
154
179
/// Path to the IMAGE file
155
180
#[ arg( long, short, value_hint = ValueHint :: FilePath ) ]
156
181
pub image : Option < PathBuf > ,
157
182
/// Which image PROTOCOL to use
158
183
#[ arg( long, value_enum, requires = "image" , value_name = "PROTOCOL" ) ]
184
+ #[ serde( skip) ]
159
185
pub image_protocol : Option < ImageProtocol > ,
160
186
/// VALUE of color resolution to use with SIXEL backend
161
187
#[ arg(
@@ -166,10 +192,11 @@ pub struct ImageCliOptions {
166
192
value_parser = PossibleValuesParser :: new( COLOR_RESOLUTIONS )
167
193
. map( |s| s. parse:: <usize >( ) . unwrap( ) )
168
194
) ]
195
+ #[ merge( strategy = overwrite) ]
169
196
pub color_resolution : usize ,
170
197
}
171
198
172
- #[ derive( Clone , Debug , Args , PartialEq , Eq ) ]
199
+ #[ derive( Clone , Debug , Args , PartialEq , Eq , Merge , Deserialize , Serialize ) ]
173
200
#[ command( next_help_heading = "TEXT FORMATTING" ) ]
174
201
pub struct TextForamttingCliOptions {
175
202
/// Changes the text colors (X X X...)
@@ -186,59 +213,71 @@ pub struct TextForamttingCliOptions {
186
213
value_parser = value_parser!( u8 ) . range( ..16 ) ,
187
214
num_args = 1 ..=6
188
215
) ]
216
+ #[ merge( strategy = overwrite_vector) ]
189
217
pub text_colors : Vec < u8 > ,
190
218
/// Use ISO 8601 formatted timestamps
191
219
#[ arg( long, short = 'z' ) ]
220
+ #[ merge( strategy = merge:: bool :: overwrite_false) ]
192
221
pub iso_time : bool ,
193
222
/// Which thousands SEPARATOR to use
194
223
#[ arg( long, value_name = "SEPARATOR" , default_value = "plain" , value_enum) ]
224
+ #[ merge( strategy = overwrite) ]
195
225
pub number_separator : NumberSeparator ,
196
226
/// Turns off bold formatting
197
227
#[ arg( long) ]
228
+ #[ merge( strategy = merge:: bool :: overwrite_false) ]
198
229
pub no_bold : bool ,
199
230
}
200
- #[ derive( Clone , Debug , Args , PartialEq , Eq , Default ) ]
231
+ #[ derive( Clone , Debug , Args , PartialEq , Eq , Default , Merge , Deserialize , Serialize ) ]
201
232
#[ command( next_help_heading = "VISUALS" ) ]
202
233
pub struct VisualsCliOptions {
203
234
/// Hides the color palette
204
235
#[ arg( long) ]
236
+ #[ merge( strategy = merge:: bool :: overwrite_false) ]
205
237
pub no_color_palette : bool ,
206
238
/// Hides the ascii art or image if provided
207
239
#[ arg( long) ]
240
+ #[ merge( strategy = merge:: bool :: overwrite_false) ]
208
241
pub no_art : bool ,
209
242
/// Use Nerd Font icons
210
243
///
211
244
/// Replaces language chips with Nerd Font icons
212
245
#[ arg( long) ]
246
+ #[ merge( strategy = merge:: bool :: overwrite_false) ]
213
247
pub nerd_fonts : bool ,
214
248
}
215
249
216
- #[ derive( Clone , Debug , Args , PartialEq , Eq , Default ) ]
250
+ #[ derive( Clone , Debug , Args , PartialEq , Eq , Default , Merge , Deserialize , Serialize ) ]
217
251
#[ command( next_help_heading = "DEVELOPER" ) ]
218
252
pub struct DeveloperCliOptions {
219
253
/// Outputs Onefetch in a specific format
220
254
#[ arg( long, short, value_name = "FORMAT" , value_enum) ]
255
+ #[ serde( skip) ]
221
256
pub output : Option < SerializationFormat > ,
222
257
/// If provided, outputs the completion file for given SHELL
223
258
#[ arg( long = "generate" , value_name = "SHELL" , value_enum) ]
259
+ #[ serde( skip) ]
224
260
pub completion : Option < Shell > ,
225
261
}
226
262
227
- #[ derive( Clone , Debug , Args , PartialEq , Eq , Default ) ]
263
+ #[ derive( Clone , Debug , Args , PartialEq , Eq , Default , Merge , Deserialize , Serialize ) ]
228
264
#[ command( next_help_heading = "OTHER" ) ]
229
265
pub struct OtherCliOptions {
230
266
/// Prints out supported languages
231
267
#[ arg( long, short) ]
268
+ #[ merge( strategy = merge:: bool :: overwrite_false) ]
232
269
pub languages : bool ,
233
270
/// Prints out supported package managers
234
271
#[ arg( long, short) ]
272
+ #[ merge( strategy = merge:: bool :: overwrite_false) ]
235
273
pub package_managers : bool ,
236
274
}
237
275
238
276
impl Default for CliOptions {
239
277
fn default ( ) -> CliOptions {
240
278
CliOptions {
241
279
input : PathBuf :: from ( "." ) ,
280
+ config_path : Default :: default ( ) ,
242
281
info : InfoCliOptions :: default ( ) ,
243
282
text_formatting : TextForamttingCliOptions :: default ( ) ,
244
283
visuals : VisualsCliOptions :: default ( ) ,
@@ -302,6 +341,15 @@ impl Default for ImageCliOptions {
302
341
}
303
342
}
304
343
344
+ pub fn overwrite < T > ( left : & mut T , right : T ) {
345
+ * left = right;
346
+ }
347
+
348
+ pub fn overwrite_vector < T > ( left : & mut Vec < T > , mut right : Vec < T > ) {
349
+ left. clear ( ) ;
350
+ left. append ( & mut right) ;
351
+ }
352
+
305
353
pub fn print_supported_languages ( ) -> Result < ( ) > {
306
354
for l in Language :: iter ( ) {
307
355
println ! ( "{l}" ) ;
@@ -337,14 +385,14 @@ pub fn print_completions<G: Generator>(gen: G, cmd: &mut Command) {
337
385
generate ( gen, cmd, cmd. get_name ( ) . to_string ( ) , & mut io:: stdout ( ) ) ;
338
386
}
339
387
340
- #[ derive( clap:: ValueEnum , Clone , PartialEq , Eq , Debug ) ]
388
+ #[ derive( clap:: ValueEnum , Clone , PartialEq , Eq , Debug , Deserialize , Serialize ) ]
341
389
pub enum When {
342
390
Auto ,
343
391
Never ,
344
392
Always ,
345
393
}
346
394
347
- #[ derive( clap:: ValueEnum , Clone , PartialEq , Eq , Debug , Serialize , Copy ) ]
395
+ #[ derive( clap:: ValueEnum , Clone , PartialEq , Eq , Debug , Serialize , Deserialize , Copy ) ]
348
396
pub enum NumberSeparator {
349
397
Plain ,
350
398
Comma ,
@@ -463,3 +511,37 @@ impl FromStr for MyRegex {
463
511
Ok ( MyRegex ( Regex :: new ( s) ?) )
464
512
}
465
513
}
514
+
515
+ impl Serialize for MyRegex {
516
+ fn serialize < S > ( & self , serializer : S ) -> Result < S :: Ok , S :: Error >
517
+ where
518
+ S : Serializer ,
519
+ { serializer. serialize_str ( self . 0 . as_str ( ) ) }
520
+ }
521
+
522
+ pub struct RegVisitor ;
523
+
524
+ impl < ' de > Visitor < ' de > for RegVisitor {
525
+ type Value = MyRegex ;
526
+
527
+ fn expecting ( & self , formatter : & mut std:: fmt:: Formatter ) -> std:: fmt:: Result {
528
+ formatter. write_str ( "regex" )
529
+ }
530
+
531
+ fn visit_str < E > ( self , v : & str ) -> Result < Self :: Value , E >
532
+ where
533
+ E : serde:: de:: Error , {
534
+ match MyRegex :: from_str ( v) {
535
+ Ok ( regex) => Ok ( regex) ,
536
+ Err ( error) => Err ( serde:: de:: Error :: custom ( error) )
537
+ }
538
+ }
539
+ }
540
+
541
+ impl < ' de > Deserialize < ' de > for MyRegex {
542
+ fn deserialize < D > ( deserializer : D ) -> Result < Self , D :: Error >
543
+ where
544
+ D : serde:: Deserializer < ' de > , {
545
+ deserializer. deserialize_str ( RegVisitor )
546
+ }
547
+ }
0 commit comments