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
use crate::commands::math::utils::run_with_numerical_functions_on_stream; use crate::prelude::*; use bigdecimal::One; use nu_engine::WholeStreamCommand; use nu_errors::ShellError; use nu_protocol::{Signature, UntaggedValue, Value}; pub struct SubCommand; impl WholeStreamCommand for SubCommand { fn name(&self) -> &str { "math floor" } fn signature(&self) -> Signature { Signature::build("math floor") } fn usage(&self) -> &str { "Applies the floor function to a list of numbers" } fn run(&self, args: CommandArgs) -> Result<OutputStream, ShellError> { let input = args.input; run_with_numerical_functions_on_stream( input, floor_int, floor_big_int, floor_big_decimal, floor_default, ) } fn examples(&self) -> Vec<Example> { vec![Example { description: "Apply the floor function to a list of numbers", example: "echo [1.5 2.3 -3.1] | math floor", result: Some(vec![ UntaggedValue::big_int(1).into(), UntaggedValue::big_int(2).into(), UntaggedValue::big_int(-4).into(), ]), }] } } fn floor_int(val: i64) -> Value { UntaggedValue::int(val).into() } fn floor_big_int(val: BigInt) -> Value { UntaggedValue::big_int(val).into() } fn floor_big_decimal(val: BigDecimal) -> Value { let mut maybe_floored = val.round(0); if maybe_floored > val { maybe_floored -= BigDecimal::one(); } let (floored, _) = maybe_floored.into_bigint_and_exponent(); UntaggedValue::big_int(floored).into() } fn floor_default(_: UntaggedValue) -> Value { UntaggedValue::Error(ShellError::unexpected( "Only numerical values are supported", )) .into() } #[cfg(test)] mod tests { use super::ShellError; use super::SubCommand; #[test] fn examples_work_as_expected() -> Result<(), ShellError> { use crate::examples::test as test_examples; test_examples(SubCommand {}) } }