2019-05-29 04:02:36 +00:00
|
|
|
use crate::errors::ShellError;
|
|
|
|
use crate::object::Value;
|
|
|
|
use crate::prelude::*;
|
|
|
|
|
2019-05-31 19:15:29 +00:00
|
|
|
fn get_member(path: &str, obj: &Value) -> Option<Value> {
|
|
|
|
let mut current = obj;
|
|
|
|
for p in path.split(".") {
|
|
|
|
match current.get_data_by_key(p) {
|
|
|
|
Some(v) => current = v,
|
|
|
|
None => {
|
2019-06-01 05:50:16 +00:00
|
|
|
return Some(Value::Error(Box::new(ShellError::string(format!(
|
|
|
|
"Object field name not found: {}",
|
|
|
|
p
|
|
|
|
)))))
|
2019-05-31 19:15:29 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Some(current.copy())
|
|
|
|
}
|
|
|
|
|
2019-06-05 01:53:38 +00:00
|
|
|
pub fn get(args: CommandArgs) -> Result<OutputStream, ShellError> {
|
2019-06-07 22:35:07 +00:00
|
|
|
if args.positional.len() == 0 {
|
|
|
|
if let Some(span) = args.name_span {
|
|
|
|
return Err(ShellError::labeled_error(
|
|
|
|
"Get requires a field or field path",
|
|
|
|
"needs parameter",
|
|
|
|
span,
|
|
|
|
));
|
|
|
|
} else {
|
|
|
|
return Err(ShellError::string("get requires fields."));
|
|
|
|
}
|
2019-05-29 04:02:36 +00:00
|
|
|
}
|
|
|
|
|
2019-06-01 05:50:16 +00:00
|
|
|
let fields: Result<Vec<String>, _> = args.positional.iter().map(|a| a.as_string()).collect();
|
2019-05-29 04:02:36 +00:00
|
|
|
let fields = fields?;
|
|
|
|
|
|
|
|
let stream = args
|
|
|
|
.input
|
|
|
|
.map(move |item| {
|
|
|
|
let mut result = VecDeque::new();
|
|
|
|
for field in &fields {
|
2019-05-31 19:15:29 +00:00
|
|
|
match get_member(field, &item) {
|
2019-05-29 04:02:36 +00:00
|
|
|
Some(Value::List(l)) => {
|
|
|
|
for item in l {
|
|
|
|
result.push_back(ReturnValue::Value(item.copy()));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Some(x) => result.push_back(ReturnValue::Value(x.copy())),
|
|
|
|
None => {}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
result
|
|
|
|
})
|
|
|
|
.flatten();
|
|
|
|
|
|
|
|
Ok(stream.boxed())
|
|
|
|
}
|