mod range; mod row; mod stream; mod unit; use chrono::{DateTime, FixedOffset}; use chrono_humanize::HumanTime; pub use range::*; pub use row::*; use serde::{Deserialize, Serialize}; pub use stream::*; pub use unit::*; use std::fmt::Debug; use crate::ast::{CellPath, PathMember, RangeInclusion}; use crate::{span, BlockId, Span, Type}; use crate::ShellError; /// Core structured values that pass through the pipeline in engine-q #[derive(Debug, Clone, Serialize, Deserialize)] pub enum Value { Bool { val: bool, span: Span, }, Int { val: i64, span: Span, }, Filesize { val: i64, span: Span, }, Duration { val: i64, span: Span, }, Date { val: DateTime, span: Span, }, Range { val: Box, span: Span, }, Float { val: f64, span: Span, }, String { val: String, span: Span, }, Record { cols: Vec, vals: Vec, span: Span, }, Stream { stream: ValueStream, span: Span, }, List { vals: Vec, span: Span, }, Block { val: BlockId, span: Span, }, Nothing { span: Span, }, Error { error: ShellError, }, Binary { val: Vec, span: Span, }, CellPath { val: CellPath, span: Span, }, } impl Value { pub fn as_string(&self) -> Result { match self { Value::String { val, .. } => Ok(val.to_string()), _ => Err(ShellError::CantConvert("string".into(), self.span())), } } /// Get the span for the current value pub fn span(&self) -> Span { match self { Value::Error { .. } => Span::unknown(), Value::Bool { span, .. } => *span, Value::Int { span, .. } => *span, Value::Float { span, .. } => *span, Value::Filesize { span, .. } => *span, Value::Duration { span, .. } => *span, Value::Date { span, .. } => *span, Value::Range { span, .. } => *span, Value::String { span, .. } => *span, Value::Record { span, .. } => *span, Value::List { span, .. } => *span, Value::Block { span, .. } => *span, Value::Stream { span, .. } => *span, Value::Nothing { span, .. } => *span, Value::Binary { span, .. } => *span, Value::CellPath { span, .. } => *span, } } /// Update the value with a new span pub fn with_span(mut self, new_span: Span) -> Value { match &mut self { Value::Bool { span, .. } => *span = new_span, Value::Int { span, .. } => *span = new_span, Value::Float { span, .. } => *span = new_span, Value::Filesize { span, .. } => *span = new_span, Value::Duration { span, .. } => *span = new_span, Value::Date { span, .. } => *span = new_span, Value::Range { span, .. } => *span = new_span, Value::String { span, .. } => *span = new_span, Value::Record { span, .. } => *span = new_span, Value::Stream { span, .. } => *span = new_span, Value::List { span, .. } => *span = new_span, Value::Block { span, .. } => *span = new_span, Value::Nothing { span, .. } => *span = new_span, Value::Error { .. } => {} Value::Binary { span, .. } => *span = new_span, Value::CellPath { span, .. } => *span = new_span, } self } /// Get the type of the current Value pub fn get_type(&self) -> Type { match self { Value::Bool { .. } => Type::Bool, Value::Int { .. } => Type::Int, Value::Float { .. } => Type::Float, Value::Filesize { .. } => Type::Filesize, Value::Duration { .. } => Type::Duration, Value::Date { .. } => Type::Date, Value::Range { .. } => Type::Range, Value::String { .. } => Type::String, Value::Record { cols, vals, .. } => { Type::Record(cols.clone(), vals.iter().map(|x| x.get_type()).collect()) } Value::List { .. } => Type::List(Box::new(Type::Unknown)), // FIXME Value::Nothing { .. } => Type::Nothing, Value::Block { .. } => Type::Block, Value::Stream { .. } => Type::ValueStream, Value::Error { .. } => Type::Error, Value::Binary { .. } => Type::Binary, Value::CellPath { .. } => Type::CellPath, } } /// Convert Value into string. Note that Streams will be consumed. pub fn into_string(self) -> String { match self { Value::Bool { val, .. } => val.to_string(), Value::Int { val, .. } => val.to_string(), Value::Float { val, .. } => val.to_string(), Value::Filesize { val, .. } => format_filesize(val), Value::Duration { val, .. } => format_duration(val), Value::Date { val, .. } => HumanTime::from(val).to_string(), Value::Range { val, .. } => { format!( "range: [{}]", val.into_iter() .map(|x| x.into_string()) .collect::>() .join(", ") ) } Value::String { val, .. } => val, Value::Stream { stream, .. } => stream.into_string(), Value::List { vals: val, .. } => format!( "[{}]", val.into_iter() .map(|x| x.into_string()) .collect::>() .join(", ") ), Value::Record { cols, vals, .. } => format!( "{{{}}}", cols.iter() .zip(vals.iter()) .map(|(x, y)| format!("{}: {}", x, y.clone().into_string())) .collect::>() .join(", ") ), Value::Block { val, .. } => format!("", val), Value::Nothing { .. } => String::new(), Value::Error { error } => format!("{:?}", error), Value::Binary { val, .. } => format!("{:?}", val), Value::CellPath { val, .. } => val.into_string(), } } pub fn collect_string(self) -> String { match self { Value::Bool { val, .. } => val.to_string(), Value::Int { val, .. } => val.to_string(), Value::Float { val, .. } => val.to_string(), Value::Filesize { val, .. } => format!("{} bytes", val), Value::Duration { val, .. } => format!("{} ns", val), Value::Date { val, .. } => format!("{:?}", val), Value::Range { val, .. } => val .into_iter() .map(|x| x.into_string()) .collect::>() .join(", "), Value::String { val, .. } => val, Value::Stream { stream, .. } => stream.collect_string(), Value::List { vals: val, .. } => val .into_iter() .map(|x| x.collect_string()) .collect::>() .join("\n"), Value::Record { vals, .. } => vals .into_iter() .map(|y| y.collect_string()) .collect::>() .join("\n"), Value::Block { val, .. } => format!("", val), Value::Nothing { .. } => String::new(), Value::Error { error } => format!("{:?}", error), Value::Binary { val, .. } => format!("{:?}", val), Value::CellPath { .. } => self.into_string(), } } /// Create a new `Nothing` value pub fn nothing() -> Value { Value::Nothing { span: Span::unknown(), } } /// Follow a given column path into the value: for example accessing nth elements in a stream or list pub fn follow_cell_path(self, cell_path: &[PathMember]) -> Result { let mut current = self; for member in cell_path { // FIXME: this uses a few extra clones for simplicity, but there may be a way // to traverse the path without them match member { PathMember::Int { val: count, span: origin_span, } => { // Treat a numeric path member as `nth ` match &mut current { Value::List { vals: val, .. } => { if let Some(item) = val.get(*count) { current = item.clone(); } else { return Err(ShellError::AccessBeyondEnd(val.len(), *origin_span)); } } Value::Stream { stream, .. } => { if let Some(item) = stream.nth(*count) { current = item; } else { return Err(ShellError::AccessBeyondEndOfStream(*origin_span)); } } x => { return Err(ShellError::IncompatiblePathAccess( format!("{}", x.get_type()), *origin_span, )) } } } PathMember::String { val: column_name, span: origin_span, } => match &mut current { Value::Record { cols, vals, .. } => { let mut found = false; for col in cols.iter().zip(vals.iter()) { if col.0 == column_name { current = col.1.clone(); found = true; break; } } if !found { return Err(ShellError::CantFindColumn(*origin_span)); } } Value::List { vals, span } => { let mut output = vec![]; for val in vals { if let Value::Record { cols, vals, .. } = val { for col in cols.iter().enumerate() { if col.1 == column_name { output.push(vals[col.0].clone()); } } } } current = Value::List { vals: output, span: *span, }; } Value::Stream { stream, span } => { let mut output = vec![]; for val in stream { if let Value::Record { cols, vals, .. } = val { for col in cols.iter().enumerate() { if col.1 == column_name { output.push(vals[col.0].clone()); } } } } current = Value::List { vals: output, span: *span, }; } x => { return Err(ShellError::IncompatiblePathAccess( format!("{}", x.get_type()), *origin_span, )) } }, } } Ok(current) } pub fn string(s: &str, span: Span) -> Value { Value::String { val: s.into(), span, } } pub fn is_true(&self) -> bool { matches!(self, Value::Bool { val: true, .. }) } pub fn columns(&self) -> Vec { match self { Value::Record { cols, .. } => cols.clone(), _ => vec![], } } pub fn map(self, span: Span, mut f: F) -> Value where Self: Sized, F: FnMut(Self) -> Value + 'static, { match self { Value::List { vals, .. } => Value::List { vals: vals.into_iter().map(f).collect(), span, }, Value::Stream { stream, .. } => Value::Stream { stream: stream.map(f).into_value_stream(), span, }, Value::Range { val, .. } => Value::Stream { stream: val.into_iter().map(f).into_value_stream(), span, }, v => { if v.as_string().is_ok() { Value::List { vals: vec![f(v)], span, } } else { Value::Error { error: ShellError::PipelineMismatch { expected: Type::String, expected_span: span, origin: v.span(), }, } } } } } pub fn flat_map(self, span: Span, mut f: F) -> Value where Self: Sized, U: IntoIterator, F: FnMut(Self) -> U + 'static, { match self { Value::List { vals, .. } => Value::List { vals: vals.into_iter().map(f).flatten().collect(), span, }, Value::Stream { stream, .. } => Value::Stream { stream: stream.map(f).flatten().into_value_stream(), span, }, Value::Range { val, .. } => Value::Stream { stream: val.into_iter().map(f).flatten().into_value_stream(), span, }, v => { if v.as_string().is_ok() { Value::List { vals: f(v).into_iter().collect(), span, } } else { Value::Error { error: ShellError::PipelineMismatch { expected: Type::String, expected_span: span, origin: v.span(), }, } } } } } } impl PartialEq for Value { fn eq(&self, other: &Self) -> bool { match (self, other) { (Value::Bool { val: lhs, .. }, Value::Bool { val: rhs, .. }) => lhs == rhs, (Value::Int { val: lhs, .. }, Value::Int { val: rhs, .. }) => lhs == rhs, (Value::Float { val: lhs, .. }, Value::Float { val: rhs, .. }) => lhs == rhs, (Value::String { val: lhs, .. }, Value::String { val: rhs, .. }) => lhs == rhs, (Value::Block { val: b1, .. }, Value::Block { val: b2, .. }) => b1 == b2, (Value::List { vals: vals_lhs, .. }, Value::List { vals: vals_rhs, .. }) => { for (lhs, rhs) in vals_lhs.iter().zip(vals_rhs) { if lhs != rhs { return false; } } true } ( Value::Record { cols: cols_lhs, vals: vals_lhs, .. }, Value::Record { cols: cols_rhs, vals: vals_rhs, .. }, ) => { if cols_lhs != cols_rhs { return false; } for (lhs, rhs) in vals_lhs.iter().zip(vals_rhs) { if lhs != rhs { return false; } } true } ( Value::Stream { stream: stream_lhs, .. }, Value::Stream { stream: stream_rhs, .. }, ) => { let vals_lhs: Vec = stream_lhs.clone().collect(); let vals_rhs: Vec = stream_rhs.clone().collect(); vals_lhs == vals_rhs } // Note: This may look a bit strange, but a Stream is still just a List, // it just happens to be in an iterator form instead of a concrete form. If the contained // values are the same then it should be treated as equal ( Value::Stream { stream: stream_lhs, .. }, Value::List { vals: stream_rhs, .. }, ) => { let vals_lhs: Vec = stream_lhs.clone().collect(); let vals_rhs: Vec = stream_rhs.clone().into_iter().into_value_stream().collect(); vals_lhs == vals_rhs } // Note: This may look a bit strange, but a Stream is still just a List, // it just happens to be in an iterator form instead of a concrete form. If the contained // values are the same then it should be treated as equal ( Value::List { vals: stream_lhs, .. }, Value::Stream { stream: stream_rhs, .. }, ) => { let vals_lhs: Vec = stream_lhs.clone().into_iter().into_value_stream().collect(); let vals_rhs: Vec = stream_rhs.clone().collect(); vals_lhs == vals_rhs } _ => false, } } } impl Value { pub fn add(&self, op: Span, rhs: &Value) -> Result { let span = span(&[self.span(), rhs.span()]); match (self, rhs) { (Value::Int { val: lhs, .. }, Value::Int { val: rhs, .. }) => Ok(Value::Int { val: lhs + rhs, span, }), (Value::Int { val: lhs, .. }, Value::Float { val: rhs, .. }) => Ok(Value::Float { val: *lhs as f64 + *rhs, span, }), (Value::Float { val: lhs, .. }, Value::Int { val: rhs, .. }) => Ok(Value::Float { val: *lhs + *rhs as f64, span, }), (Value::Float { val: lhs, .. }, Value::Float { val: rhs, .. }) => Ok(Value::Float { val: lhs + rhs, span, }), (Value::String { val: lhs, .. }, Value::String { val: rhs, .. }) => Ok(Value::String { val: lhs.to_string() + rhs, span, }), (Value::Duration { val: lhs, .. }, Value::Duration { val: rhs, .. }) => { Ok(Value::Duration { val: *lhs + *rhs, span, }) } (Value::Filesize { val: lhs, .. }, Value::Filesize { val: rhs, .. }) => { Ok(Value::Filesize { val: *lhs + *rhs, span, }) } _ => Err(ShellError::OperatorMismatch { op_span: op, lhs_ty: self.get_type(), lhs_span: self.span(), rhs_ty: rhs.get_type(), rhs_span: rhs.span(), }), } } pub fn sub(&self, op: Span, rhs: &Value) -> Result { let span = span(&[self.span(), rhs.span()]); match (self, rhs) { (Value::Int { val: lhs, .. }, Value::Int { val: rhs, .. }) => Ok(Value::Int { val: lhs - rhs, span, }), (Value::Int { val: lhs, .. }, Value::Float { val: rhs, .. }) => Ok(Value::Float { val: *lhs as f64 - *rhs, span, }), (Value::Float { val: lhs, .. }, Value::Int { val: rhs, .. }) => Ok(Value::Float { val: *lhs - *rhs as f64, span, }), (Value::Float { val: lhs, .. }, Value::Float { val: rhs, .. }) => Ok(Value::Float { val: lhs - rhs, span, }), (Value::Duration { val: lhs, .. }, Value::Duration { val: rhs, .. }) => { Ok(Value::Duration { val: *lhs - *rhs, span, }) } (Value::Filesize { val: lhs, .. }, Value::Filesize { val: rhs, .. }) => { Ok(Value::Filesize { val: *lhs - *rhs, span, }) } _ => Err(ShellError::OperatorMismatch { op_span: op, lhs_ty: self.get_type(), lhs_span: self.span(), rhs_ty: rhs.get_type(), rhs_span: rhs.span(), }), } } pub fn mul(&self, op: Span, rhs: &Value) -> Result { let span = span(&[self.span(), rhs.span()]); match (self, rhs) { (Value::Int { val: lhs, .. }, Value::Int { val: rhs, .. }) => Ok(Value::Int { val: lhs * rhs, span, }), (Value::Int { val: lhs, .. }, Value::Float { val: rhs, .. }) => Ok(Value::Float { val: *lhs as f64 * *rhs, span, }), (Value::Float { val: lhs, .. }, Value::Int { val: rhs, .. }) => Ok(Value::Float { val: *lhs * *rhs as f64, span, }), (Value::Float { val: lhs, .. }, Value::Float { val: rhs, .. }) => Ok(Value::Float { val: lhs * rhs, span, }), _ => Err(ShellError::OperatorMismatch { op_span: op, lhs_ty: self.get_type(), lhs_span: self.span(), rhs_ty: rhs.get_type(), rhs_span: rhs.span(), }), } } pub fn div(&self, op: Span, rhs: &Value) -> Result { let span = span(&[self.span(), rhs.span()]); match (self, rhs) { (Value::Int { val: lhs, .. }, Value::Int { val: rhs, .. }) => { if *rhs != 0 { if lhs % rhs == 0 { Ok(Value::Int { val: lhs / rhs, span, }) } else { Ok(Value::Float { val: (*lhs as f64) / (*rhs as f64), span, }) } } else { Err(ShellError::DivisionByZero(op)) } } (Value::Int { val: lhs, .. }, Value::Float { val: rhs, .. }) => { if *rhs != 0.0 { Ok(Value::Float { val: *lhs as f64 / *rhs, span, }) } else { Err(ShellError::DivisionByZero(op)) } } (Value::Float { val: lhs, .. }, Value::Int { val: rhs, .. }) => { if *rhs != 0 { Ok(Value::Float { val: *lhs / *rhs as f64, span, }) } else { Err(ShellError::DivisionByZero(op)) } } (Value::Float { val: lhs, .. }, Value::Float { val: rhs, .. }) => { if *rhs != 0.0 { Ok(Value::Float { val: lhs / rhs, span, }) } else { Err(ShellError::DivisionByZero(op)) } } _ => Err(ShellError::OperatorMismatch { op_span: op, lhs_ty: self.get_type(), lhs_span: self.span(), rhs_ty: rhs.get_type(), rhs_span: rhs.span(), }), } } pub fn lt(&self, op: Span, rhs: &Value) -> Result { let span = span(&[self.span(), rhs.span()]); match (self, rhs) { (Value::Int { val: lhs, .. }, Value::Int { val: rhs, .. }) => Ok(Value::Bool { val: lhs < rhs, span, }), (Value::Int { val: lhs, .. }, Value::Float { val: rhs, .. }) => Ok(Value::Bool { val: (*lhs as f64) < *rhs, span, }), (Value::Float { val: lhs, .. }, Value::Int { val: rhs, .. }) => Ok(Value::Bool { val: *lhs < *rhs as f64, span, }), (Value::Float { val: lhs, .. }, Value::Float { val: rhs, .. }) => Ok(Value::Bool { val: lhs < rhs, span, }), (Value::Duration { val: lhs, .. }, Value::Duration { val: rhs, .. }) => { Ok(Value::Bool { val: lhs < rhs, span, }) } (Value::Filesize { val: lhs, .. }, Value::Filesize { val: rhs, .. }) => { Ok(Value::Bool { val: lhs < rhs, span, }) } _ => Err(ShellError::OperatorMismatch { op_span: op, lhs_ty: self.get_type(), lhs_span: self.span(), rhs_ty: rhs.get_type(), rhs_span: rhs.span(), }), } } pub fn lte(&self, op: Span, rhs: &Value) -> Result { let span = span(&[self.span(), rhs.span()]); match (self, rhs) { (Value::Int { val: lhs, .. }, Value::Int { val: rhs, .. }) => Ok(Value::Bool { val: lhs <= rhs, span, }), (Value::Int { val: lhs, .. }, Value::Float { val: rhs, .. }) => Ok(Value::Bool { val: (*lhs as f64) <= *rhs, span, }), (Value::Float { val: lhs, .. }, Value::Int { val: rhs, .. }) => Ok(Value::Bool { val: *lhs <= *rhs as f64, span, }), (Value::Float { val: lhs, .. }, Value::Float { val: rhs, .. }) => Ok(Value::Bool { val: lhs <= rhs, span, }), (Value::Duration { val: lhs, .. }, Value::Duration { val: rhs, .. }) => { Ok(Value::Bool { val: lhs <= rhs, span, }) } (Value::Filesize { val: lhs, .. }, Value::Filesize { val: rhs, .. }) => { Ok(Value::Bool { val: lhs <= rhs, span, }) } _ => Err(ShellError::OperatorMismatch { op_span: op, lhs_ty: self.get_type(), lhs_span: self.span(), rhs_ty: rhs.get_type(), rhs_span: rhs.span(), }), } } pub fn gt(&self, op: Span, rhs: &Value) -> Result { let span = span(&[self.span(), rhs.span()]); match (self, rhs) { (Value::Int { val: lhs, .. }, Value::Int { val: rhs, .. }) => Ok(Value::Bool { val: lhs > rhs, span, }), (Value::Int { val: lhs, .. }, Value::Float { val: rhs, .. }) => Ok(Value::Bool { val: (*lhs as f64) > *rhs, span, }), (Value::Float { val: lhs, .. }, Value::Int { val: rhs, .. }) => Ok(Value::Bool { val: *lhs > *rhs as f64, span, }), (Value::Float { val: lhs, .. }, Value::Float { val: rhs, .. }) => Ok(Value::Bool { val: lhs > rhs, span, }), (Value::Duration { val: lhs, .. }, Value::Duration { val: rhs, .. }) => { Ok(Value::Bool { val: lhs > rhs, span, }) } (Value::Filesize { val: lhs, .. }, Value::Filesize { val: rhs, .. }) => { Ok(Value::Bool { val: lhs > rhs, span, }) } _ => Err(ShellError::OperatorMismatch { op_span: op, lhs_ty: self.get_type(), lhs_span: self.span(), rhs_ty: rhs.get_type(), rhs_span: rhs.span(), }), } } pub fn gte(&self, op: Span, rhs: &Value) -> Result { let span = span(&[self.span(), rhs.span()]); match (self, rhs) { (Value::Int { val: lhs, .. }, Value::Int { val: rhs, .. }) => Ok(Value::Bool { val: lhs >= rhs, span, }), (Value::Int { val: lhs, .. }, Value::Float { val: rhs, .. }) => Ok(Value::Bool { val: (*lhs as f64) >= *rhs, span, }), (Value::Float { val: lhs, .. }, Value::Int { val: rhs, .. }) => Ok(Value::Bool { val: *lhs >= *rhs as f64, span, }), (Value::Float { val: lhs, .. }, Value::Float { val: rhs, .. }) => Ok(Value::Bool { val: lhs >= rhs, span, }), (Value::Duration { val: lhs, .. }, Value::Duration { val: rhs, .. }) => { Ok(Value::Bool { val: lhs >= rhs, span, }) } (Value::Filesize { val: lhs, .. }, Value::Filesize { val: rhs, .. }) => { Ok(Value::Bool { val: lhs >= rhs, span, }) } _ => Err(ShellError::OperatorMismatch { op_span: op, lhs_ty: self.get_type(), lhs_span: self.span(), rhs_ty: rhs.get_type(), rhs_span: rhs.span(), }), } } pub fn eq(&self, op: Span, rhs: &Value) -> Result { let span = span(&[self.span(), rhs.span()]); match (self, rhs) { (Value::Int { val: lhs, .. }, Value::Int { val: rhs, .. }) => Ok(Value::Bool { val: lhs == rhs, span, }), (Value::String { val: lhs, .. }, Value::String { val: rhs, .. }) => Ok(Value::Bool { val: lhs == rhs, span, }), // FIXME: these should consider machine epsilon (Value::Int { val: lhs, .. }, Value::Float { val: rhs, .. }) => Ok(Value::Bool { val: (*lhs as f64) == *rhs, span, }), // FIXME: these should consider machine epsilon (Value::Float { val: lhs, .. }, Value::Int { val: rhs, .. }) => Ok(Value::Bool { val: *lhs == *rhs as f64, span, }), // FIXME: these should consider machine epsilon (Value::Float { val: lhs, .. }, Value::Float { val: rhs, .. }) => Ok(Value::Bool { val: lhs == rhs, span, }), (Value::Duration { val: lhs, .. }, Value::Duration { val: rhs, .. }) => { Ok(Value::Bool { val: lhs == rhs, span, }) } (Value::Filesize { val: lhs, .. }, Value::Filesize { val: rhs, .. }) => { Ok(Value::Bool { val: lhs == rhs, span, }) } (Value::List { vals: lhs, .. }, Value::List { vals: rhs, .. }) => Ok(Value::Bool { val: lhs == rhs, span, }), ( Value::Record { vals: lhs, cols: lhs_headers, .. }, Value::Record { vals: rhs, cols: rhs_headers, .. }, ) => Ok(Value::Bool { val: lhs_headers == rhs_headers && lhs == rhs, span, }), _ => Err(ShellError::OperatorMismatch { op_span: op, lhs_ty: self.get_type(), lhs_span: self.span(), rhs_ty: rhs.get_type(), rhs_span: rhs.span(), }), } } pub fn ne(&self, op: Span, rhs: &Value) -> Result { let span = span(&[self.span(), rhs.span()]); match (self, rhs) { (Value::Int { val: lhs, .. }, Value::Int { val: rhs, .. }) => Ok(Value::Bool { val: lhs != rhs, span, }), (Value::String { val: lhs, .. }, Value::String { val: rhs, .. }) => Ok(Value::Bool { val: lhs != rhs, span, }), // FIXME: these should consider machine epsilon (Value::Int { val: lhs, .. }, Value::Float { val: rhs, .. }) => Ok(Value::Bool { val: (*lhs as f64) != *rhs, span, }), // FIXME: these should consider machine epsilon (Value::Float { val: lhs, .. }, Value::Int { val: rhs, .. }) => Ok(Value::Bool { val: *lhs != *rhs as f64, span, }), // FIXME: these should consider machine epsilon (Value::Float { val: lhs, .. }, Value::Float { val: rhs, .. }) => Ok(Value::Bool { val: lhs != rhs, span, }), (Value::Duration { val: lhs, .. }, Value::Duration { val: rhs, .. }) => { Ok(Value::Bool { val: lhs != rhs, span, }) } (Value::Filesize { val: lhs, .. }, Value::Filesize { val: rhs, .. }) => { Ok(Value::Bool { val: lhs != rhs, span, }) } (Value::List { vals: lhs, .. }, Value::List { vals: rhs, .. }) => Ok(Value::Bool { val: lhs != rhs, span, }), ( Value::Record { vals: lhs, cols: lhs_headers, .. }, Value::Record { vals: rhs, cols: rhs_headers, .. }, ) => Ok(Value::Bool { val: lhs_headers != rhs_headers || lhs != rhs, span, }), _ => Err(ShellError::OperatorMismatch { op_span: op, lhs_ty: self.get_type(), lhs_span: self.span(), rhs_ty: rhs.get_type(), rhs_span: rhs.span(), }), } } pub fn r#in(&self, op: Span, rhs: &Value) -> Result { let span = span(&[self.span(), rhs.span()]); match (self, rhs) { (lhs, Value::Range { val: rhs, .. }) => Ok(Value::Bool { val: if matches!( rhs.incr.gt( Span::unknown(), &Value::Int { val: 0, span: Span::unknown() } ), Ok(Value::Bool { val: true, .. }) ) { matches!( lhs.gte(Span::unknown(), &rhs.from), Ok(Value::Bool { val: true, .. }) ) && match rhs.inclusion { RangeInclusion::Inclusive => matches!( lhs.lte(Span::unknown(), &rhs.to), Ok(Value::Bool { val: true, .. }) ), RangeInclusion::RightExclusive => matches!( lhs.lt(Span::unknown(), &rhs.to), Ok(Value::Bool { val: true, .. }) ), } } else { matches!( lhs.lte(Span::unknown(), &rhs.from), Ok(Value::Bool { val: true, .. }) ) && match rhs.inclusion { RangeInclusion::Inclusive => matches!( lhs.gte(Span::unknown(), &rhs.to), Ok(Value::Bool { val: true, .. }) ), RangeInclusion::RightExclusive => matches!( lhs.gt(Span::unknown(), &rhs.to), Ok(Value::Bool { val: true, .. }) ), } }, span, }), (Value::String { val: lhs, .. }, Value::String { val: rhs, .. }) => Ok(Value::Bool { val: rhs.contains(lhs), span, }), (lhs, Value::List { vals: rhs, .. }) => Ok(Value::Bool { val: rhs .iter() .any(|x| lhs.eq(Span::unknown(), x).map_or(false, |v| v.is_true())), span, }), _ => Err(ShellError::OperatorMismatch { op_span: op, lhs_ty: self.get_type(), lhs_span: self.span(), rhs_ty: rhs.get_type(), rhs_span: rhs.span(), }), } } } /// Format a duration in nanoseconds into a string pub fn format_duration(duration: i64) -> String { let (sign, duration) = if duration >= 0 { (1, duration) } else { (-1, -duration) }; let (micros, nanos): (i64, i64) = (duration / 1000, duration % 1000); let (millis, micros): (i64, i64) = (micros / 1000, micros % 1000); let (secs, millis): (i64, i64) = (millis / 1000, millis % 1000); let (mins, secs): (i64, i64) = (secs / 60, secs % 60); let (hours, mins): (i64, i64) = (mins / 60, mins % 60); let (days, hours): (i64, i64) = (hours / 24, hours % 24); let mut output_prep = vec![]; if days != 0 { output_prep.push(format!("{}day", days)); } if hours != 0 { output_prep.push(format!("{}hr", hours)); } if mins != 0 { output_prep.push(format!("{}min", mins)); } // output 0sec for zero duration if duration == 0 || secs != 0 { output_prep.push(format!("{}sec", secs)); } if millis != 0 { output_prep.push(format!("{}ms", millis)); } if micros != 0 { output_prep.push(format!("{}us", micros)); } if nanos != 0 { output_prep.push(format!("{}ns", nanos)); } format!( "{}{}", if sign == -1 { "-" } else { "" }, output_prep.join(" ") ) } fn format_filesize(num_bytes: i64) -> String { let byte = byte_unit::Byte::from_bytes(num_bytes as u128); if byte.get_bytes() == 0u128 { return "—".to_string(); } let byte = byte.get_appropriate_unit(false); match byte.get_unit() { byte_unit::ByteUnit::B => format!("{} B ", byte.get_value()), _ => byte.format(1), } }