pub(crate) mod shape; use bigdecimal::BigDecimal; use chrono::{DateTime, Utc}; use derive_new::new; use nu_errors::ShellError; use nu_protocol::{ hir, Primitive, ShellTypeName, SpannedTypeName, TaggedDictBuilder, UntaggedValue, Value, }; use nu_source::{Span, Tag}; use nu_value_ext::ValueExt; use num_bigint::BigInt; use num_traits::Zero; use query_interface::{interfaces, vtable_for, ObjectHash}; use serde::{Deserialize, Serialize}; #[derive(Debug, Ord, PartialOrd, Eq, PartialEq, Clone, new, Serialize)] pub struct Operation { pub(crate) left: Value, pub(crate) operator: hir::Operator, pub(crate) right: Value, } #[derive(Debug, Ord, PartialOrd, Eq, PartialEq, Clone, Hash, Serialize, Deserialize, new)] pub struct Block { pub(crate) commands: hir::Commands, pub(crate) tag: Tag, } interfaces!(Block: dyn ObjectHash); #[derive(Serialize, Deserialize)] pub enum Switch { Present, Absent, } impl std::convert::TryFrom> for Switch { type Error = ShellError; fn try_from(value: Option<&Value>) -> Result { match value { None => Ok(Switch::Absent), Some(value) => match &value.value { UntaggedValue::Primitive(Primitive::Boolean(true)) => Ok(Switch::Present), _ => Err(ShellError::type_error("Boolean", value.spanned_type_name())), }, } } } pub fn select_fields(obj: &Value, fields: &[String], tag: impl Into) -> Value { let mut out = TaggedDictBuilder::new(tag); let descs = obj.data_descriptors(); for column_name in fields { match descs.iter().find(|d| *d == column_name) { None => out.insert_untagged(column_name, UntaggedValue::nothing()), Some(desc) => out.insert_value(desc.clone(), obj.get_data(desc).borrow().clone()), } } out.into_value() } pub fn reject_fields(obj: &Value, fields: &[String], tag: impl Into) -> Value { let mut out = TaggedDictBuilder::new(tag); let descs = obj.data_descriptors(); for desc in descs { if fields.iter().any(|field| *field == desc) { continue; } else { out.insert_value(desc.clone(), obj.get_data(&desc).borrow().clone()) } } out.into_value() } pub enum CompareValues { Ints(BigInt, BigInt), Decimals(BigDecimal, BigDecimal), String(String, String), Date(DateTime, DateTime), DateDuration(DateTime, BigInt), Booleans(bool, bool), } impl CompareValues { pub fn compare(&self) -> std::cmp::Ordering { match self { CompareValues::Ints(left, right) => left.cmp(right), CompareValues::Decimals(left, right) => left.cmp(right), CompareValues::String(left, right) => left.cmp(right), CompareValues::Date(left, right) => left.cmp(right), CompareValues::DateDuration(left, right) => { // FIXME: Not sure if I could do something better with the Span. let duration = Primitive::into_chrono_duration( Primitive::Duration(right.clone()), Span::unknown(), ) .expect("Could not convert nushell Duration into chrono Duration."); let right: DateTime = Utc::now() .checked_sub_signed(duration) .expect("Data overflow"); right.cmp(left) } CompareValues::Booleans(left, right) => left.cmp(right), } } } pub fn coerce_compare( left: &UntaggedValue, right: &UntaggedValue, ) -> Result { match (left, right) { (UntaggedValue::Primitive(left), UntaggedValue::Primitive(right)) => { coerce_compare_primitive(left, right) } _ => Err((left.type_name(), right.type_name())), } } pub fn coerce_compare_primitive( left: &Primitive, right: &Primitive, ) -> Result { use Primitive::*; Ok(match (left, right) { (Int(left), Int(right)) => CompareValues::Ints(left.clone(), right.clone()), (Int(left), Decimal(right)) => { CompareValues::Decimals(BigDecimal::zero() + left, right.clone()) } (Int(left), Filesize(right)) => CompareValues::Ints(left.clone(), BigInt::from(*right)), (Decimal(left), Decimal(right)) => CompareValues::Decimals(left.clone(), right.clone()), (Decimal(left), Int(right)) => { CompareValues::Decimals(left.clone(), BigDecimal::zero() + right) } (Decimal(left), Filesize(right)) => { CompareValues::Decimals(left.clone(), BigDecimal::from(*right)) } (Filesize(left), Filesize(right)) => { CompareValues::Ints(BigInt::from(*left), BigInt::from(*right)) } (Filesize(left), Int(right)) => CompareValues::Ints(BigInt::from(*left), right.clone()), (Filesize(left), Decimal(right)) => { CompareValues::Decimals(BigDecimal::from(*left), right.clone()) } (Nothing, Nothing) => CompareValues::Booleans(true, true), (String(left), String(right)) => CompareValues::String(left.clone(), right.clone()), (Line(left), String(right)) => CompareValues::String(left.clone(), right.clone()), (String(left), Line(right)) => CompareValues::String(left.clone(), right.clone()), (Line(left), Line(right)) => CompareValues::String(left.clone(), right.clone()), (Date(left), Date(right)) => CompareValues::Date(*left, *right), (Date(left), Duration(right)) => CompareValues::DateDuration(*left, right.clone()), (Boolean(left), Boolean(right)) => CompareValues::Booleans(*left, *right), _ => return Err((left.type_name(), right.type_name())), }) } #[cfg(test)] mod tests { use indexmap::{indexmap, IndexMap}; use nu_errors::ShellError; use nu_protocol::{ColumnPath as ColumnPathValue, PathMember, UntaggedValue, Value}; use nu_source::*; use nu_value_ext::{as_column_path, ValueExt}; use num_bigint::BigInt; fn string(input: impl Into) -> Value { crate::utils::helpers::string(input) } fn int(input: impl Into) -> Value { crate::utils::helpers::int(input) } fn row(entries: IndexMap) -> Value { crate::utils::helpers::row(entries) } fn table(list: &[Value]) -> Value { crate::utils::helpers::table(list) } fn error_callback( reason: &'static str, ) -> impl FnOnce(&Value, &PathMember, ShellError) -> ShellError { move |_obj_source, _column_path_tried, _err| ShellError::unimplemented(reason) } fn column_path(paths: &[Value]) -> Result, ShellError> { as_column_path(&table(paths)) } #[test] fn gets_matching_field_from_a_row() -> Result<(), ShellError> { let row = UntaggedValue::row(indexmap! { "amigos".into() => table(&[string("andres"),string("jonathan"),string("yehuda")]) }) .into_untagged_value(); assert_eq!( row.get_data_by_key("amigos".spanned_unknown()) .ok_or_else(|| ShellError::unexpected("Failure during testing"))?, table(&[string("andres"), string("jonathan"), string("yehuda")]) ); Ok(()) } #[test] fn gets_matching_field_from_nested_rows_inside_a_row() -> Result<(), ShellError> { let field_path = column_path(&[string("package"), string("version")]); let (version, tag) = string("0.4.0").into_parts(); let value = UntaggedValue::row(indexmap! { "package".into() => row(indexmap! { "name".into() => string("nu"), "version".into() => string("0.4.0") }) }); assert_eq!( *value.into_value(tag).get_data_by_column_path( &field_path?.item, Box::new(error_callback("package.version")) )?, version ); Ok(()) } #[test] fn gets_first_matching_field_from_rows_with_same_field_inside_a_table() -> Result<(), ShellError> { let field_path = column_path(&[string("package"), string("authors"), string("name")]); let (_, tag) = string("Andrés N. Robalino").into_parts(); let value = UntaggedValue::row(indexmap! { "package".into() => row(indexmap! { "name".into() => string("nu"), "version".into() => string("0.4.0"), "authors".into() => table(&[ row(indexmap!{"name".into() => string("Andrés N. Robalino")}), row(indexmap!{"name".into() => string("Jonathan Turner")}), row(indexmap!{"name".into() => string("Yehuda Katz")}) ]) }) }); assert_eq!( value.into_value(tag).get_data_by_column_path( &field_path?.item, Box::new(error_callback("package.authors.name")) )?, table(&[ string("Andrés N. Robalino"), string("Jonathan Turner"), string("Yehuda Katz") ]) ); Ok(()) } #[test] fn column_path_that_contains_just_a_number_gets_a_row_from_a_table() -> Result<(), ShellError> { let field_path = column_path(&[string("package"), string("authors"), int(0)]); let (_, tag) = string("Andrés N. Robalino").into_parts(); let value = UntaggedValue::row(indexmap! { "package".into() => row(indexmap! { "name".into() => string("nu"), "version".into() => string("0.4.0"), "authors".into() => table(&[ row(indexmap!{"name".into() => string("Andrés N. Robalino")}), row(indexmap!{"name".into() => string("Jonathan Turner")}), row(indexmap!{"name".into() => string("Yehuda Katz")}) ]) }) }); assert_eq!( *value.into_value(tag).get_data_by_column_path( &field_path?.item, Box::new(error_callback("package.authors.0")) )?, UntaggedValue::row(indexmap! { "name".into() => string("Andrés N. Robalino") }) ); Ok(()) } #[test] fn column_path_that_contains_just_a_number_gets_a_row_from_a_row() -> Result<(), ShellError> { let field_path = column_path(&[string("package"), string("authors"), string("0")]); let (_, tag) = string("Andrés N. Robalino").into_parts(); let value = UntaggedValue::row(indexmap! { "package".into() => row(indexmap! { "name".into() => string("nu"), "version".into() => string("0.4.0"), "authors".into() => row(indexmap! { "0".into() => row(indexmap!{"name".into() => string("Andrés N. Robalino")}), "1".into() => row(indexmap!{"name".into() => string("Jonathan Turner")}), "2".into() => row(indexmap!{"name".into() => string("Yehuda Katz")}), }) }) }); assert_eq!( *value.into_value(tag).get_data_by_column_path( &field_path?.item, Box::new(error_callback("package.authors.\"0\"")) )?, UntaggedValue::row(indexmap! { "name".into() => string("Andrés N. Robalino") }) ); Ok(()) } #[test] fn replaces_matching_field_from_a_row() -> Result<(), ShellError> { let field_path = column_path(&[string("amigos")]); let sample = UntaggedValue::row(indexmap! { "amigos".into() => table(&[ string("andres"), string("jonathan"), string("yehuda"), ]), }); let replacement = string("jonas"); let actual = sample .into_untagged_value() .replace_data_at_column_path(&field_path?.item, replacement) .ok_or_else(|| ShellError::untagged_runtime_error("Could not replace column"))?; assert_eq!(actual, row(indexmap! {"amigos".into() => string("jonas")})); Ok(()) } #[test] fn replaces_matching_field_from_nested_rows_inside_a_row() -> Result<(), ShellError> { let field_path = column_path(&[ string("package"), string("authors"), string("los.3.caballeros"), ]); let sample = UntaggedValue::row(indexmap! { "package".into() => row(indexmap! { "authors".into() => row(indexmap! { "los.3.mosqueteros".into() => table(&[string("andres::yehuda::jonathan")]), "los.3.amigos".into() => table(&[string("andres::yehuda::jonathan")]), "los.3.caballeros".into() => table(&[string("andres::yehuda::jonathan")]) }) }) }); let replacement = table(&[string("yehuda::jonathan::andres")]); let tag = replacement.tag.clone(); let actual = sample .into_value(&tag) .replace_data_at_column_path(&field_path?.item, replacement.clone()) .ok_or_else(|| { ShellError::labeled_error( "Could not replace column", "could not replace column", &tag, ) })?; assert_eq!( actual, UntaggedValue::row(indexmap! { "package".into() => row(indexmap! { "authors".into() => row(indexmap! { "los.3.mosqueteros".into() => table(&[string("andres::yehuda::jonathan")]), "los.3.amigos".into() => table(&[string("andres::yehuda::jonathan")]), "los.3.caballeros".into() => replacement})})}) .into_value(tag) ); Ok(()) } #[test] fn replaces_matching_field_from_rows_inside_a_table() -> Result<(), ShellError> { let field_path = column_path(&[ string("shell_policy"), string("releases"), string("nu.version.arepa"), ]); let sample = UntaggedValue::row(indexmap! { "shell_policy".into() => row(indexmap! { "releases".into() => table(&[ row(indexmap! { "nu.version.arepa".into() => row(indexmap! { "code".into() => string("0.4.0"), "tag_line".into() => string("GitHub-era") }) }), row(indexmap! { "nu.version.taco".into() => row(indexmap! { "code".into() => string("0.3.0"), "tag_line".into() => string("GitHub-era") }) }), row(indexmap! { "nu.version.stable".into() => row(indexmap! { "code".into() => string("0.2.0"), "tag_line".into() => string("GitHub-era") }) }) ]) }) }); let replacement = row(indexmap! { "code".into() => string("0.5.0"), "tag_line".into() => string("CABALLEROS") }); let tag = replacement.tag.clone(); let actual = sample .into_value(tag.clone()) .replace_data_at_column_path(&field_path?.item, replacement.clone()) .ok_or_else(|| { ShellError::labeled_error( "Could not replace column", "could not replace column", &tag, ) })?; assert_eq!( actual, UntaggedValue::row(indexmap! { "shell_policy".into() => row(indexmap! { "releases".into() => table(&[ row(indexmap! { "nu.version.arepa".into() => replacement }), row(indexmap! { "nu.version.taco".into() => row(indexmap! { "code".into() => string("0.3.0"), "tag_line".into() => string("GitHub-era") }) }), row(indexmap! { "nu.version.stable".into() => row(indexmap! { "code".into() => string("0.2.0"), "tag_line".into() => string("GitHub-era") }) }) ]) }) }).into_value(&tag) ); Ok(()) } }