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
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
use std::fmt;
#[derive(Debug, Clone, Eq, PartialEq)]
pub enum RuleConfigurationError {
UnexpectedProperty(String),
MissingProperty(String),
BooleanExpected(String),
StringExpected(String),
UsizeExpected(String),
FloatExpected(String),
StringListExpected(String),
UnexpectedValueType(String),
UnexpectedValue { property: String, message: String },
PropertyCollision(Vec<String>),
}
fn enumerate_properties(properties: &[String]) -> String {
let last_index = properties.len().saturating_sub(1);
properties
.iter()
.enumerate()
.map(|(i, name)| {
if i == 0 {
format!("`{}`", name)
} else if i == last_index {
format!(" and `{}`", name)
} else {
format!(", `{}`", name)
}
})
.collect()
}
impl fmt::Display for RuleConfigurationError {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
use RuleConfigurationError::*;
match self {
UnexpectedProperty(property) => write!(f, "unexpected field '{}'", property),
MissingProperty(property) => write!(f, "missing required field '{}'", property),
BooleanExpected(property) => {
write!(f, "boolean value expected for field '{}'", property)
}
StringExpected(property) => write!(f, "string value expected for field '{}'", property),
UsizeExpected(property) => {
write!(f, "unsigned integer expected for field '{}'", property)
}
FloatExpected(property) => write!(f, "float value expected for field '{}'", property),
StringListExpected(property) => {
write!(f, "list of string expected for field '{}'", property)
}
UnexpectedValueType(property) => write!(f, "unexpected type for field '{}'", property),
UnexpectedValue { property, message } => {
write!(f, "unexpected value for field '{}': {}", property, message)
}
PropertyCollision(properties) => write!(
f,
"the properties {} cannot be defined together",
enumerate_properties(properties)
),
}
}
}
#[cfg(test)]
mod test {
use super::*;
#[test]
fn enumerate_one_property() {
assert_eq!(enumerate_properties(&["prop".to_owned()]), "`prop`")
}
#[test]
fn enumerate_two_properties() {
assert_eq!(
enumerate_properties(&["prop".to_owned(), "prop2".to_owned()]),
"`prop` and `prop2`"
)
}
#[test]
fn enumerate_three_properties() {
assert_eq!(
enumerate_properties(&["prop".to_owned(), "prop2".to_owned(), "prop3".to_owned()]),
"`prop`, `prop2` and `prop3`"
)
}
}