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
108
109
110
111
112
113
114
extern crate getopts;
extern crate regex;
#[derive(Clone)]
pub struct Opt {
node: String,
short: Option<String>,
long: Option<String>,
pub env: Option<String>,
flags: u32,
pub default: Option<String>,
pub format: Option<String>,
usage: Option<String>,
}
pub enum OptFlag {
Optional = 1,
Multiple = 2,
NoArg = 4,
OptionalArg = 8,
}
impl Opt {
pub fn new(
node: &str,
short: Option<&str>,
long: Option<&str>,
env: Option<&str>,
flags: u32,
default: Option<&str>,
format: Option<&str>,
usage: Option<&str>,
) -> Opt {
Opt {
node: node.to_owned(),
short: short.map(String::from),
long: long.map(String::from),
env: env.map(String::from),
flags: flags,
default: default.map(String::from),
format: format.map(String::from),
usage: usage.map(String::from),
}
}
pub fn to_getopts(&self, options: &mut getopts::Options) {
if self.short.is_some() || self.long.is_some() {
let mut hasarg = getopts::HasArg::Yes;
if self.flags & OptFlag::NoArg as u32 != 0 {
hasarg = getopts::HasArg::No;
} else if self.flags & OptFlag::OptionalArg as u32 != 0 {
hasarg = getopts::HasArg::Maybe;
}
options.opt(
self.short.as_ref().map_or("", String::as_str),
self.long.as_ref().map_or("", String::as_str),
self.usage.as_ref().map_or("", String::as_str),
"",
hasarg,
getopts::Occur::Multi,
);
}
}
pub fn node_depth(&self) -> usize {
self.node.matches('.').count()
}
pub fn get_branch_keys(&self) -> Vec<String> {
let keys = self.node.split('.').collect::<Vec<_>>();
keys[..keys.len() - 1]
.iter()
.map(|&s| s.to_owned())
.collect()
}
pub fn get_leaf_key(&self) -> Option<String> {
match self.node.as_ref() {
"" => None,
_ => Some(self.node.split('.').last().unwrap().to_owned())
}
}
pub fn get_name(&self) -> &Option<String> {
match self.long {
Some(_) => &self.long,
_ => &self.short,
}
}
}