Skip to content

Commit d1a6ce6

Browse files
committed
review me
1 parent dc641e3 commit d1a6ce6

File tree

6 files changed

+144
-14
lines changed

6 files changed

+144
-14
lines changed

src/cli.rs

+93-11
Original file line numberDiff line numberDiff line change
@@ -6,12 +6,16 @@ use clap::builder::PossibleValuesParser;
66
use clap::builder::TypedValueParser as _;
77
use clap::{value_parser, Args, Command, Parser, ValueHint};
88
use clap_complete::{generate, Generator, Shell};
9+
use merge::Merge;
910
use num_format::CustomFormat;
1011
use onefetch_image::ImageProtocol;
1112
use onefetch_manifest::ManifestType;
1213
use regex::Regex;
13-
use serde::Serialize;
14+
use serde::de::Visitor;
15+
use serde::{Deserialize, Serialize, Serializer};
16+
use serde_json::error;
1417
use std::env;
18+
use std::fmt::Write;
1519
use std::io;
1620
use std::path::PathBuf;
1721
use std::str::FromStr;
@@ -20,12 +24,17 @@ use strum::IntoEnumIterator;
2024
const COLOR_RESOLUTIONS: [&str; 5] = ["16", "32", "64", "128", "256"];
2125
pub const NO_BOTS_DEFAULT_REGEX_PATTERN: &str = r"(?:-|\s)[Bb]ot$|\[[Bb]ot\]";
2226

23-
#[derive(Clone, Debug, Parser, PartialEq, Eq)]
27+
#[derive(Clone, Debug, Parser, PartialEq, Eq, Deserialize, Serialize, Merge)]
2428
#[command(version, about)]
2529
pub struct CliOptions {
2630
/// Run as if onefetch was started in <input> instead of the current working directory
2731
#[arg(default_value = ".", hide_default_value = true, value_hint = ValueHint::DirPath)]
32+
#[merge(skip)]
2833
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>,
2938
#[command(flatten)]
3039
pub info: InfoCliOptions,
3140
#[command(flatten)]
@@ -42,7 +51,7 @@ pub struct CliOptions {
4251
pub other: OtherCliOptions,
4352
}
4453

45-
#[derive(Clone, Debug, Args, PartialEq, Eq)]
54+
#[derive(Clone, Debug, Args, PartialEq, Eq, Merge, Deserialize, Serialize)]
4655
#[command(next_help_heading = "INFO")]
4756
pub struct InfoCliOptions {
4857
/// Allows you to disable FIELD(s) from appearing in the output
@@ -54,18 +63,23 @@ pub struct InfoCliOptions {
5463
value_enum,
5564
value_name = "FIELD"
5665
)]
66+
#[merge(strategy = overwrite_vector)]
5767
pub disabled_fields: Vec<InfoType>,
5868
/// Hides the title
5969
#[arg(long)]
70+
#[merge(strategy = merge::bool::overwrite_false)]
6071
pub no_title: bool,
6172
/// Maximum NUM of authors to be shown
6273
#[arg(long, default_value_t = 3usize, value_name = "NUM")]
74+
#[merge(strategy = overwrite)]
6375
pub number_of_authors: usize,
6476
/// Maximum NUM of languages to be shown
6577
#[arg(long, default_value_t = 6usize, value_name = "NUM")]
78+
#[merge(strategy = overwrite)]
6679
pub number_of_languages: usize,
6780
/// Maximum NUM of file churns to be shown
6881
#[arg(long, default_value_t = 3usize, value_name = "NUM")]
82+
#[merge(strategy = overwrite)]
6983
pub number_of_file_churns: usize,
7084
/// Minimum NUM of commits from HEAD used to compute the churn summary
7185
///
@@ -75,6 +89,7 @@ pub struct InfoCliOptions {
7589
pub churn_pool_size: Option<usize>,
7690
/// Ignore all files & directories matching EXCLUDE
7791
#[arg(long, short, num_args = 1..)]
92+
#[merge(strategy = overwrite_vector)]
7893
pub exclude: Vec<String>,
7994
/// Exclude [bot] commits. Use <REGEX> to override the default pattern
8095
#[arg(
@@ -84,21 +99,27 @@ pub struct InfoCliOptions {
8499
default_missing_value = NO_BOTS_DEFAULT_REGEX_PATTERN,
85100
value_name = "REGEX"
86101
)]
102+
#[merge(strategy = overwrite)]
87103
pub no_bots: Option<MyRegex>,
88104
/// Ignores merge commits
89105
#[arg(long)]
106+
#[merge(strategy = merge::bool::overwrite_false)]
90107
pub no_merges: bool,
91108
/// Show the email address of each author
92109
#[arg(long, short = 'E')]
110+
#[merge(strategy = merge::bool::overwrite_false)]
93111
pub email: bool,
94112
/// Display repository URL as HTTP
95113
#[arg(long)]
114+
#[merge(strategy = merge::bool::overwrite_false)]
96115
pub http_url: bool,
97116
/// Hide token in repository URL
98117
#[arg(long)]
118+
#[merge(strategy = merge::bool::overwrite_false)]
99119
pub hide_token: bool,
100120
/// Count hidden files and directories
101121
#[arg(long)]
122+
#[merge(strategy = merge::bool::overwrite_false)]
102123
pub include_hidden: bool,
103124
/// Filters output by language type
104125
#[arg(
@@ -108,10 +129,11 @@ pub struct InfoCliOptions {
108129
short = 'T',
109130
value_enum,
110131
)]
132+
#[merge(strategy = overwrite_vector)]
111133
pub r#type: Vec<LanguageType>,
112134
}
113135

114-
#[derive(Clone, Debug, Args, PartialEq, Eq)]
136+
#[derive(Clone, Debug, Args, PartialEq, Eq, Merge, Deserialize, Serialize)]
115137
#[command(next_help_heading = "ASCII")]
116138
pub struct AsciiCliOptions {
117139
/// Takes a non-empty STRING as input to replace the ASCII logo
@@ -131,6 +153,7 @@ pub struct AsciiCliOptions {
131153
short = 'c',
132154
value_parser = value_parser!(u8).range(..16),
133155
)]
156+
#[merge(strategy = overwrite_vector)]
134157
pub ascii_colors: Vec<u8>,
135158
/// Which LANGUAGE's ascii art to print
136159
#[arg(
@@ -140,22 +163,25 @@ pub struct AsciiCliOptions {
140163
value_enum,
141164
hide_possible_values = true
142165
)]
166+
#[serde(skip)]
143167
pub ascii_language: Option<Language>,
144168
/// Specify when to use true color
145169
///
146170
/// If set to auto: true color will be enabled if supported by the terminal
147171
#[arg(long, default_value = "auto", value_name = "WHEN", value_enum)]
172+
#[merge(strategy = overwrite)]
148173
pub true_color: When,
149174
}
150175

151-
#[derive(Clone, Debug, Args, PartialEq, Eq)]
176+
#[derive(Clone, Debug, Args, PartialEq, Eq, Merge, Deserialize, Serialize)]
152177
#[command(next_help_heading = "IMAGE")]
153178
pub struct ImageCliOptions {
154179
/// Path to the IMAGE file
155180
#[arg(long, short, value_hint = ValueHint::FilePath)]
156181
pub image: Option<PathBuf>,
157182
/// Which image PROTOCOL to use
158183
#[arg(long, value_enum, requires = "image", value_name = "PROTOCOL")]
184+
#[serde(skip)]
159185
pub image_protocol: Option<ImageProtocol>,
160186
/// VALUE of color resolution to use with SIXEL backend
161187
#[arg(
@@ -166,10 +192,11 @@ pub struct ImageCliOptions {
166192
value_parser = PossibleValuesParser::new(COLOR_RESOLUTIONS)
167193
.map(|s| s.parse::<usize>().unwrap())
168194
)]
195+
#[merge(strategy = overwrite)]
169196
pub color_resolution: usize,
170197
}
171198

172-
#[derive(Clone, Debug, Args, PartialEq, Eq)]
199+
#[derive(Clone, Debug, Args, PartialEq, Eq, Merge, Deserialize, Serialize)]
173200
#[command(next_help_heading = "TEXT FORMATTING")]
174201
pub struct TextForamttingCliOptions {
175202
/// Changes the text colors (X X X...)
@@ -186,59 +213,71 @@ pub struct TextForamttingCliOptions {
186213
value_parser = value_parser!(u8).range(..16),
187214
num_args = 1..=6
188215
)]
216+
#[merge(strategy = overwrite_vector)]
189217
pub text_colors: Vec<u8>,
190218
/// Use ISO 8601 formatted timestamps
191219
#[arg(long, short = 'z')]
220+
#[merge(strategy = merge::bool::overwrite_false)]
192221
pub iso_time: bool,
193222
/// Which thousands SEPARATOR to use
194223
#[arg(long, value_name = "SEPARATOR", default_value = "plain", value_enum)]
224+
#[merge(strategy = overwrite)]
195225
pub number_separator: NumberSeparator,
196226
/// Turns off bold formatting
197227
#[arg(long)]
228+
#[merge(strategy = merge::bool::overwrite_false)]
198229
pub no_bold: bool,
199230
}
200-
#[derive(Clone, Debug, Args, PartialEq, Eq, Default)]
231+
#[derive(Clone, Debug, Args, PartialEq, Eq, Default, Merge, Deserialize, Serialize)]
201232
#[command(next_help_heading = "VISUALS")]
202233
pub struct VisualsCliOptions {
203234
/// Hides the color palette
204235
#[arg(long)]
236+
#[merge(strategy = merge::bool::overwrite_false)]
205237
pub no_color_palette: bool,
206238
/// Hides the ascii art or image if provided
207239
#[arg(long)]
240+
#[merge(strategy = merge::bool::overwrite_false)]
208241
pub no_art: bool,
209242
/// Use Nerd Font icons
210243
///
211244
/// Replaces language chips with Nerd Font icons
212245
#[arg(long)]
246+
#[merge(strategy = merge::bool::overwrite_false)]
213247
pub nerd_fonts: bool,
214248
}
215249

216-
#[derive(Clone, Debug, Args, PartialEq, Eq, Default)]
250+
#[derive(Clone, Debug, Args, PartialEq, Eq, Default, Merge, Deserialize, Serialize)]
217251
#[command(next_help_heading = "DEVELOPER")]
218252
pub struct DeveloperCliOptions {
219253
/// Outputs Onefetch in a specific format
220254
#[arg(long, short, value_name = "FORMAT", value_enum)]
255+
#[serde(skip)]
221256
pub output: Option<SerializationFormat>,
222257
/// If provided, outputs the completion file for given SHELL
223258
#[arg(long = "generate", value_name = "SHELL", value_enum)]
259+
#[serde(skip)]
224260
pub completion: Option<Shell>,
225261
}
226262

227-
#[derive(Clone, Debug, Args, PartialEq, Eq, Default)]
263+
#[derive(Clone, Debug, Args, PartialEq, Eq, Default, Merge, Deserialize, Serialize)]
228264
#[command(next_help_heading = "OTHER")]
229265
pub struct OtherCliOptions {
230266
/// Prints out supported languages
231267
#[arg(long, short)]
268+
#[merge(strategy = merge::bool::overwrite_false)]
232269
pub languages: bool,
233270
/// Prints out supported package managers
234271
#[arg(long, short)]
272+
#[merge(strategy = merge::bool::overwrite_false)]
235273
pub package_managers: bool,
236274
}
237275

238276
impl Default for CliOptions {
239277
fn default() -> CliOptions {
240278
CliOptions {
241279
input: PathBuf::from("."),
280+
config_path: Default::default(),
242281
info: InfoCliOptions::default(),
243282
text_formatting: TextForamttingCliOptions::default(),
244283
visuals: VisualsCliOptions::default(),
@@ -302,6 +341,15 @@ impl Default for ImageCliOptions {
302341
}
303342
}
304343

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+
305353
pub fn print_supported_languages() -> Result<()> {
306354
for l in Language::iter() {
307355
println!("{l}");
@@ -337,14 +385,14 @@ pub fn print_completions<G: Generator>(gen: G, cmd: &mut Command) {
337385
generate(gen, cmd, cmd.get_name().to_string(), &mut io::stdout());
338386
}
339387

340-
#[derive(clap::ValueEnum, Clone, PartialEq, Eq, Debug)]
388+
#[derive(clap::ValueEnum, Clone, PartialEq, Eq, Debug, Deserialize, Serialize)]
341389
pub enum When {
342390
Auto,
343391
Never,
344392
Always,
345393
}
346394

347-
#[derive(clap::ValueEnum, Clone, PartialEq, Eq, Debug, Serialize, Copy)]
395+
#[derive(clap::ValueEnum, Clone, PartialEq, Eq, Debug, Serialize, Deserialize, Copy)]
348396
pub enum NumberSeparator {
349397
Plain,
350398
Comma,
@@ -463,3 +511,37 @@ impl FromStr for MyRegex {
463511
Ok(MyRegex(Regex::new(s)?))
464512
}
465513
}
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+
}

src/configuration.rs

+46
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,46 @@
1+
use anyhow::{anyhow, Result};
2+
use crate::cli::CliOptions;
3+
use dirs::config_dir;
4+
use merge::Merge;
5+
use std::{fs::{self, create_dir, File}, path::Path};
6+
use std::io::{BufReader, Read};
7+
8+
pub fn read_cfg_file<P: AsRef<Path>>(path: &P) -> Result<CliOptions> {
9+
let file = File::open(&path)?;
10+
let mut buf_reader = BufReader::new(file);
11+
let mut contents = String::new();
12+
buf_reader.read_to_string(&mut contents)?;
13+
Ok(toml::from_str(contents.as_str()).unwrap())
14+
}
15+
16+
pub fn load_cfg<P: AsRef<Path>>(path: Option<&P>) -> Result<CliOptions> {
17+
match path {
18+
Some(config_path) => read_cfg_file(config_path),
19+
None => {
20+
let mut default_path = config_dir().unwrap();
21+
default_path.push("/onefetch/config.toml");
22+
if default_path.exists() {
23+
read_cfg_file(&default_path)
24+
} else {
25+
create_dir(&default_path)?;
26+
write_default_cfg(&default_path);
27+
Err(anyhow!("Configuration file at {:?} does not exist!", default_path))
28+
}
29+
}
30+
}
31+
}
32+
33+
pub fn write_default_cfg<P: AsRef<Path>>(default_path: &P) {
34+
let toml = toml::to_string(&CliOptions::default()).expect("Config should be serializable");
35+
fs::write(default_path, toml).expect("Unable to write file");
36+
}
37+
38+
pub fn populate(config: CliOptions) -> CliOptions{
39+
match load_cfg(config.config_path.as_ref()) {
40+
Ok(mut cfg) =>{
41+
cfg.merge(config);
42+
cfg
43+
},
44+
Err(_) => config
45+
}
46+
}

src/info/langs/language.rs

-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,5 @@
11
use crate::info::utils::info_field::InfoField;
22
use owo_colors::OwoColorize;
3-
use serde::Serialize;
43
use tokei;
54

65
include!(concat!(env!("OUT_DIR"), "/language.rs"));

src/info/langs/language.tera

+2-1
Original file line numberDiff line numberDiff line change
@@ -5,6 +5,7 @@ use owo_colors::{
55
use std::fmt;
66
use std::fmt::Write;
77
use strum::EnumIter;
8+
use serde::{Deserialize,Serialize};
89

910
pub struct Colors {
1011
basic_colors: Vec<DynColors>,
@@ -13,7 +14,7 @@ pub struct Colors {
1314

1415
const DEFAULT_CHIP_ICON: char = '\u{25CF}';
1516

16-
#[derive(Clone, PartialEq, Eq, Debug, clap::ValueEnum)]
17+
#[derive(Clone, PartialEq, Eq, Debug, clap::ValueEnum, Deserialize, Serialize)]
1718
pub enum LanguageType {
1819
Programming,
1920
Markup,

0 commit comments

Comments
 (0)