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
use anyhow::{anyhow, Result};
use std::io::Read;
use std::str::FromStr;
pub fn parse_key_val<T: FromStr>(s: &str) -> Result<(T, T)>
where
<T as FromStr>::Err: std::error::Error + Send + Sync + 'static,
{
let mut iter = s.splitn(2, '=');
let first = iter.next();
let second = iter.next();
match (first, second) {
(Some(first), Some(second)) => Ok((first.parse()?, second.parse()?)),
_ => Err(anyhow!("Invalid KEY=value: no `=` found in `{}`", s)),
}
}
pub fn get_input(values: &[(String, String)]) -> Vec<(String, String)> {
let mut result = Vec::new();
result.extend_from_slice(values);
let mut buffer = String::new();
if atty::isnt(atty::Stream::Stdin) {
std::io::stdin()
.read_to_string(&mut buffer)
.expect("Could not read stdin");
if !buffer.is_empty() {
if let Some(b) = buffer.strip_suffix("\n") {
buffer = b.to_string();
}
if let Ok(keyval) = parse_key_val(&buffer) {
result.push(keyval);
}
}
}
result
}
#[cfg(test)]
mod tests {
mod parse_key_val {
use super::super::*;
use pretty_assertions::assert_eq;
#[test]
fn it_works_with_one_equals() {
let (key, value): (String, String) = parse_key_val("meow=bar").unwrap();
assert_eq!(key, "meow".to_string());
assert_eq!(value, "bar".to_string());
}
#[test]
fn it_works_with_2_plus_equals() {
let (key, value): (String, String) = parse_key_val("meow=bar=bar2=bar3").unwrap();
assert_eq!(key, "meow".to_string());
assert_eq!(value, "bar=bar2=bar3".to_string());
}
#[test]
fn it_fails() {
let result: Result<(String, String)> = parse_key_val("meow");
let error = result.err().unwrap();
assert_eq!(
format!("{}", error),
"Invalid KEY=value: no `=` found in `meow`".to_string()
);
}
}
}