use nu_protocol::ast::Call; use nu_protocol::engine::{Command, EngineState, Stack}; use nu_protocol::{Category, Example, PipelineData, ShellError, Signature, Span, Type, Value}; #[derive(Clone)] pub struct SubCommand; impl Command for SubCommand { fn name(&self) -> &str { "math abs" } fn signature(&self) -> Signature { Signature::build("math abs") .input_output_types(vec![(Type::Number, Type::Number)]) .vectorizes_over_list(true) .category(Category::Math) } fn usage(&self) -> &str { "Returns the absolute value of a number." } fn search_terms(&self) -> Vec<&str> { vec!["absolute", "modulus", "positive", "distance"] } fn run( &self, engine_state: &EngineState, _stack: &mut Stack, call: &Call, input: PipelineData, ) -> Result { let head = call.head; input.map( move |value| abs_helper(value, head), engine_state.ctrlc.clone(), ) } fn examples(&self) -> Vec { vec![Example { description: "Compute absolute value of each number in a list of numbers", example: "[-50 -100.0 25] | math abs", result: Some(Value::List { vals: vec![ Value::test_int(50), Value::test_float(100.0), Value::test_int(25), ], span: Span::test_data(), }), }] } } fn abs_helper(val: Value, head: Span) -> Value { match val { Value::Int { val, span } => Value::int(val.abs(), span), Value::Float { val, span } => Value::Float { val: val.abs(), span, }, Value::Duration { val, span } => Value::Duration { val: val.abs(), span, }, Value::Error { .. } => val, other => Value::Error { error: Box::new(ShellError::OnlySupportsThisInputType { exp_input_type: "numeric".into(), wrong_type: other.get_type().to_string(), dst_span: head, src_span: other.expect_span(), }), }, } } #[cfg(test)] mod test { use super::*; #[test] fn test_examples() { use crate::test_examples; test_examples(SubCommand {}) } }