From 7a3aeaf080b7d0e4754649c61e3e4fac4865914b Mon Sep 17 00:00:00 2001 From: Michael Angerman <1809991+stormasm@users.noreply.github.com> Date: Sun, 13 Feb 2022 09:23:54 -0800 Subject: [PATCH] sort_by: coerce_compare now returns an Ordering (#4461) * coerce_compare now returns an Ordering which will enable mixed type comparison * arbitrary nushell sort order of Float / Int / String / Bool --- crates/nu-command/src/filters/sort_by.rs | 64 ++++++++++++++++-------- 1 file changed, 44 insertions(+), 20 deletions(-) diff --git a/crates/nu-command/src/filters/sort_by.rs b/crates/nu-command/src/filters/sort_by.rs index edf008955f..391aebb912 100644 --- a/crates/nu-command/src/filters/sort_by.rs +++ b/crates/nu-command/src/filters/sort_by.rs @@ -183,7 +183,6 @@ pub fn sort( config, ) .expect("sort_by Value::Record bug") - .compare() }); } _ => { @@ -199,11 +198,8 @@ pub fn sort( ); coerce_compare(&lowercase_left, &lowercase_right, call) .expect("sort_by default bug") - .compare() } else { - coerce_compare(a, b, call) - .expect("sort_by default bug") - .compare() + coerce_compare(a, b, call).expect("sort_by default bug") } }); } @@ -218,7 +214,7 @@ pub fn process( call: &Call, insensitive: bool, config: &Config, -) -> Result { +) -> Result { let left_value = left.get_data_by_key(column); let left_res = match left_value { @@ -280,36 +276,64 @@ pub fn process_floats(left: &f64, right: &f64) -> std::cmp::Ordering { } } -pub fn coerce_compare( - left: &Value, - right: &Value, - call: &Call, -) -> Result { - match (left, right) { +/* +Arbitrary Order of Values: + Floats + Ints + Strings + Bools +*/ + +pub fn coerce_compare(left: &Value, right: &Value, call: &Call) -> Result { + Ok(match (left, right) { (Value::Float { val: left, .. }, Value::Float { val: right, .. }) => { - Ok(CompareValues::Floats(*left, *right)) + CompareValues::Floats(*left, *right).compare() } (Value::Filesize { val: left, .. }, Value::Filesize { val: right, .. }) => { - Ok(CompareValues::Filesize(*left, *right)) + CompareValues::Filesize(*left, *right).compare() } (Value::Date { val: left, .. }, Value::Date { val: right, .. }) => { - Ok(CompareValues::Date(*left, *right)) + CompareValues::Date(*left, *right).compare() } (Value::Int { val: left, .. }, Value::Int { val: right, .. }) => { - Ok(CompareValues::Ints(*left, *right)) + CompareValues::Ints(*left, *right).compare() } (Value::String { val: left, .. }, Value::String { val: right, .. }) => { - Ok(CompareValues::String(left.clone(), right.clone())) + CompareValues::String(left.clone(), right.clone()).compare() } (Value::Bool { val: left, .. }, Value::Bool { val: right, .. }) => { - Ok(CompareValues::Booleans(*left, *right)) + CompareValues::Booleans(*left, *right).compare() } + // Floats will always come before Ints + (Value::Float { .. }, Value::Int { .. }) => Ordering::Less, + (Value::Int { .. }, Value::Float { .. }) => Ordering::Greater, + + // Floats will always come before Strings + (Value::Float { .. }, Value::String { .. }) => Ordering::Less, + (Value::String { .. }, Value::Float { .. }) => Ordering::Greater, + + // Floats will always come before Bools + (Value::Float { .. }, Value::Bool { .. }) => Ordering::Less, + (Value::Bool { .. }, Value::Float { .. }) => Ordering::Greater, + + // Ints will always come before strings + (Value::Int { .. }, Value::String { .. }) => Ordering::Less, + (Value::String { .. }, Value::Int { .. }) => Ordering::Greater, + + // Ints will always come before Bools + (Value::Int { .. }, Value::Bool { .. }) => Ordering::Less, + (Value::Bool { .. }, Value::Int { .. }) => Ordering::Greater, + + // Strings will always come before Bools + (Value::String { .. }, Value::Bool { .. }) => Ordering::Less, + (Value::Bool { .. }, Value::String { .. }) => Ordering::Greater, + _ => { let description = format!("not able to compare {:?} with {:?}\n", left, right); - Err(ShellError::TypeMismatch(description, call.head)) + return Err(ShellError::TypeMismatch(description, call.head)); } - } + }) } #[cfg(test)]