use super::variance::compute_variance as variance; use crate::math::utils::run_with_function; 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 stddev" } fn signature(&self) -> Signature { Signature::build("math stddev") .input_output_types(vec![(Type::List(Box::new(Type::Number)), Type::Number)]) .switch( "sample", "calculate sample standard deviation (i.e. using N-1 as the denominator)", Some('s'), ) .category(Category::Math) } fn usage(&self) -> &str { "Returns the standard deviation of a list of numbers, or of each column in a table" } fn search_terms(&self) -> Vec<&str> { vec![ "SD", "standard", "deviation", "dispersion", "variation", "statistics", ] } fn run( &self, _engine_state: &EngineState, _stack: &mut Stack, call: &Call, input: PipelineData, ) -> Result { let sample = call.has_flag("sample"); run_with_function(call, input, compute_stddev(sample)) } fn examples(&self) -> Vec { vec![ Example { description: "Compute the standard deviation of a list of numbers", example: "[1 2 3 4 5] | math stddev", result: Some(Value::Float { val: std::f64::consts::SQRT_2, span: Span::test_data(), }), }, Example { description: "Compute the sample standard deviation of a list of numbers", example: "[1 2 3 4 5] | math stddev -s", result: Some(Value::Float { val: 1.5811388300841898, span: Span::test_data(), }), }, ] } } pub fn compute_stddev(sample: bool) -> impl Fn(&[Value], &Span) -> Result { move |values: &[Value], span: &Span| { let variance = variance(sample)(values, span); match variance { Ok(Value::Float { val, span }) => Ok(Value::Float { val: val.sqrt(), span }), Ok(Value::Int { val, span }) => Ok(Value::Float { val: (val as f64).sqrt(), span }), Err(ShellError::UnsupportedInput(_, err_span)) => Err(ShellError::UnsupportedInput( "Attempted to compute the standard deviation with an item that cannot be used for that.".to_string(), err_span, )), other => other } } } #[cfg(test)] mod test { use super::*; #[test] fn test_examples() { use crate::test_examples; test_examples(SubCommand {}) } }