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
use std::{
collections::{BTreeMap, BTreeSet},
convert::TryInto,
};
use mit_lint::{Lint, Lints, CONFIG_KEY_PREFIX};
use thiserror::Error as ThisError;
use crate::{external, external::Vcs};
pub fn read_from_toml_or_else_vcs(config: &str, vcs: &mut dyn Vcs) -> Result<Lints, Error> {
let vcs_lints = try_from_vcs(vcs)?;
let config: BTreeMap<String, BTreeMap<String, BTreeMap<String, bool>>> =
toml::from_str(config)?;
let lint_prefix = CONFIG_KEY_PREFIX.split('.').collect::<Vec<_>>();
let namespace = (*lint_prefix.get(0).unwrap()).to_string();
let config = match config.get(&namespace) {
None => return Ok(vcs_lints),
Some(lints) => lints,
};
let group = (*lint_prefix.get(1).unwrap()).to_string();
let lint_names = match config.get(&group) {
None => return Ok(vcs_lints),
Some(lints) => lints,
};
let to_add: Lints = lint_names
.iter()
.filter_map(|(key, value)| if *value { Some(key as &str) } else { None })
.collect::<Vec<&str>>()
.try_into()?;
let to_remove: Lints = lint_names
.iter()
.filter_map(|(key, value)| if *value { None } else { Some(key as &str) })
.collect::<Vec<&str>>()
.try_into()?;
Ok(vcs_lints.subtract(&to_remove).merge(&to_add))
}
fn try_from_vcs(config: &mut dyn Vcs) -> Result<Lints, Error> {
Ok(Lints::new(
Lint::all_lints()
.filter_map(|lint| {
get_config_or_default(config, lint, lint.enabled_by_default()).transpose()
})
.collect::<Result<BTreeSet<Lint>, Error>>()?,
))
}
fn get_config_or_default(
config: &dyn Vcs,
lint: Lint,
default: bool,
) -> Result<Option<Lint>, Error> {
Ok(config
.get_bool(&lint.config_key())?
.or(Some(default))
.filter(|lint_value| lint_value == &true)
.map(|_| lint))
}
#[derive(ThisError, Debug)]
pub enum Error {
#[error("lint config reading failed: {0}")]
VcsIo(#[from] external::Error),
#[error("{0}")]
Toml(#[from] toml::de::Error),
#[error("{0}")]
MitLint(#[from] mit_lint::Error),
}