use nu_cmd_base::input_handler::{operate, CellPathOnlyArgs}; use nu_engine::command_prelude::*; use nu_utils::get_system_locale; #[derive(Clone)] pub struct IntoFilesize; impl Command for IntoFilesize { fn name(&self) -> &str { "into filesize" } fn signature(&self) -> Signature { Signature::build("into filesize") .input_output_types(vec![ (Type::Int, Type::Filesize), (Type::Number, Type::Filesize), (Type::String, Type::Filesize), (Type::Filesize, Type::Filesize), (Type::table(), Type::table()), (Type::record(), Type::record()), ( Type::List(Box::new(Type::Int)), Type::List(Box::new(Type::Filesize)), ), ( Type::List(Box::new(Type::Number)), Type::List(Box::new(Type::Filesize)), ), ( Type::List(Box::new(Type::String)), Type::List(Box::new(Type::Filesize)), ), ( Type::List(Box::new(Type::Filesize)), Type::List(Box::new(Type::Filesize)), ), // Catch all for heterogeneous lists. ( Type::List(Box::new(Type::Any)), Type::List(Box::new(Type::Filesize)), ), ]) .allow_variants_without_examples(true) .rest( "rest", SyntaxShape::CellPath, "For a data structure input, convert data at the given cell paths.", ) .category(Category::Conversions) } fn description(&self) -> &str { "Convert value to filesize." } fn search_terms(&self) -> Vec<&str> { vec!["convert", "number", "bytes"] } fn run( &self, engine_state: &EngineState, stack: &mut Stack, call: &Call, input: PipelineData, ) -> Result { let cell_paths: Vec = call.rest(engine_state, stack, 0)?; let args = CellPathOnlyArgs::from(cell_paths); operate(action, args, input, call.head, engine_state.signals()) } fn examples(&self) -> Vec { vec![ Example { description: "Convert string to filesize in table", example: r#"[[device size]; ["/dev/sda1" "200"] ["/dev/loop0" "50"]] | into filesize size"#, result: Some(Value::test_list(vec![ Value::test_record(record! { "device" => Value::test_string("/dev/sda1"), "size" => Value::test_filesize(200), }), Value::test_record(record! { "device" => Value::test_string("/dev/loop0"), "size" => Value::test_filesize(50), }), ])), }, Example { description: "Convert string to filesize", example: "'2' | into filesize", result: Some(Value::test_filesize(2)), }, Example { description: "Convert float to filesize", example: "8.3 | into filesize", result: Some(Value::test_filesize(8)), }, Example { description: "Convert int to filesize", example: "5 | into filesize", result: Some(Value::test_filesize(5)), }, Example { description: "Convert file size to filesize", example: "4KB | into filesize", result: Some(Value::test_filesize(4000)), }, Example { description: "Convert string with unit to filesize", example: "'-1KB' | into filesize", result: Some(Value::test_filesize(-1000)), }, ] } } fn action(input: &Value, _args: &CellPathOnlyArgs, span: Span) -> Value { let value_span = input.span(); match input { Value::Filesize { .. } => input.clone(), Value::Int { val, .. } => Value::filesize(*val, value_span), Value::Float { val, .. } => Value::filesize(*val as i64, value_span), Value::String { val, .. } => match i64_from_string(val, value_span) { Ok(val) => Value::filesize(val, value_span), Err(error) => Value::error(error, value_span), }, Value::Nothing { .. } => Value::filesize(0, value_span), other => Value::error( ShellError::OnlySupportsThisInputType { exp_input_type: "string and int".into(), wrong_type: other.get_type().to_string(), dst_span: span, src_span: value_span, }, span, ), } } fn i64_from_string(a_string: &str, span: Span) -> Result { // Get the Locale so we know what the thousands separator is let locale = get_system_locale(); // Now that we know the locale, get the thousands separator and remove it // so strings like 1,123,456 can be parsed as 1123456 let no_comma_string = a_string.replace(locale.separator(), ""); let clean_string = no_comma_string.trim(); // Hadle negative file size if let Some(stripped_negative_string) = clean_string.strip_prefix('-') { match stripped_negative_string.parse::() { Ok(n) => i64_from_byte_size(n, true, span), Err(_) => Err(string_convert_error(span)), } } else if let Some(stripped_positive_string) = clean_string.strip_prefix('+') { match stripped_positive_string.parse::() { Ok(n) if stripped_positive_string.starts_with(|c: char| c.is_ascii_digit()) => { i64_from_byte_size(n, false, span) } _ => Err(string_convert_error(span)), } } else { match clean_string.parse::() { Ok(n) => i64_from_byte_size(n, false, span), Err(_) => Err(string_convert_error(span)), } } } fn i64_from_byte_size( byte_size: bytesize::ByteSize, is_negative: bool, span: Span, ) -> Result { match i64::try_from(byte_size.as_u64()) { Ok(n) => Ok(if is_negative { -n } else { n }), Err(_) => Err(string_convert_error(span)), } } fn string_convert_error(span: Span) -> ShellError { ShellError::CantConvert { to_type: "filesize".into(), from_type: "string".into(), span, help: None, } } #[cfg(test)] mod test { use super::*; #[test] fn test_examples() { use crate::test_examples; test_examples(IntoFilesize {}) } }