use crate::ctx::Context; use crate::dbs::{Options, Transaction}; use crate::doc::CursorDoc; use crate::err::Error; use crate::sql::comment::mightbespace; use crate::sql::common::{closebracket, commas, openbracket}; use crate::sql::error::IResult; use crate::sql::fmt::{pretty_indent, Fmt, Pretty}; use crate::sql::number::Number; use crate::sql::operation::Operation; use crate::sql::value::{value, Value}; use nom::character::complete::char; use nom::combinator::opt; use nom::multi::separated_list0; use serde::{Deserialize, Serialize}; use std::collections::HashSet; use std::fmt::{self, Display, Formatter, Write}; use std::ops; use std::ops::Deref; use std::ops::DerefMut; pub(crate) const TOKEN: &str = "$surrealdb::private::sql::Array"; #[derive(Clone, Debug, Default, Eq, Ord, PartialEq, PartialOrd, Serialize, Deserialize, Hash)] #[serde(rename = "$surrealdb::private::sql::Array")] pub struct Array(pub Vec); impl From for Array { fn from(v: Value) -> Self { vec![v].into() } } impl From> for Array { fn from(v: Vec) -> Self { Self(v) } } impl From> for Array { fn from(v: Vec) -> Self { Self(v.into_iter().map(Value::from).collect()) } } impl From> for Array { fn from(v: Vec) -> Self { Self(v.into_iter().map(Value::from).collect()) } } impl From> for Array { fn from(v: Vec<&str>) -> Self { Self(v.into_iter().map(Value::from).collect()) } } impl From> for Array { fn from(v: Vec) -> Self { Self(v.into_iter().map(Value::from).collect()) } } impl From> for Array { fn from(v: Vec) -> Self { Self(v.into_iter().map(Value::from).collect()) } } impl From> for Array { fn from(v: Vec) -> Self { Self(v.into_iter().map(Value::from).collect()) } } impl From for Vec { fn from(s: Array) -> Self { s.0 } } impl FromIterator for Array { fn from_iter>(iter: I) -> Self { Array(iter.into_iter().collect()) } } impl Deref for Array { type Target = Vec; fn deref(&self) -> &Self::Target { &self.0 } } impl DerefMut for Array { fn deref_mut(&mut self) -> &mut Self::Target { &mut self.0 } } impl IntoIterator for Array { type Item = Value; type IntoIter = std::vec::IntoIter; fn into_iter(self) -> Self::IntoIter { self.0.into_iter() } } impl Array { pub fn new() -> Self { Self::default() } pub fn with_capacity(len: usize) -> Self { Self(Vec::with_capacity(len)) } } impl Array { /// Process this type returning a computed simple Value pub(crate) async fn compute( &self, ctx: &Context<'_>, opt: &Options, txn: &Transaction, doc: Option<&CursorDoc<'_>>, ) -> Result { let mut x = Self::with_capacity(self.len()); for v in self.iter() { match v.compute(ctx, opt, txn, doc).await { Ok(v) => x.push(v), Err(e) => return Err(e), }; } Ok(Value::Array(x)) } pub(crate) fn is_all_none_or_null(&self) -> bool { self.0.iter().all(|v| v.is_none_or_null()) } } impl Display for Array { fn fmt(&self, f: &mut Formatter) -> fmt::Result { let mut f = Pretty::from(f); f.write_char('[')?; if !self.is_empty() { let indent = pretty_indent(); write!(f, "{}", Fmt::pretty_comma_separated(self.as_slice()))?; drop(indent); } f.write_char(']') } } // ------------------------------ impl ops::Add for Array { type Output = Self; fn add(mut self, other: Value) -> Self { self.0.push(other); self } } impl ops::Add for Array { type Output = Self; fn add(mut self, mut other: Self) -> Self { self.0.append(&mut other.0); self } } // ------------------------------ impl ops::Sub for Array { type Output = Self; fn sub(mut self, other: Value) -> Self { if let Some(p) = self.0.iter().position(|x| *x == other) { self.0.remove(p); } self } } impl ops::Sub for Array { type Output = Self; fn sub(mut self, other: Self) -> Self { for v in other.0 { if let Some(p) = self.0.iter().position(|x| *x == v) { self.0.remove(p); } } self } } // ------------------------------ pub trait Abolish { fn abolish(&mut self, f: F) where F: FnMut(usize) -> bool; } impl Abolish for Vec { fn abolish(&mut self, mut f: F) where F: FnMut(usize) -> bool, { let mut i = 0; // FIXME: use drain_filter once stabilized (https://github.com/rust-lang/rust/issues/43244) // to avoid negation of the predicate return value. self.retain(|_| { let retain = !f(i); i += 1; retain }); } } // ------------------------------ pub(crate) trait Clump { fn clump(self, clump_size: usize) -> T; } impl Clump for Array { fn clump(self, clump_size: usize) -> Array { self.0 .chunks(clump_size) .map::(|chunk| chunk.to_vec().into()) .collect::>() .into() } } // ------------------------------ pub(crate) trait Combine { fn combine(self, other: T) -> T; } impl Combine for Array { fn combine(self, other: Self) -> Array { let mut out = Self::with_capacity(self.len().saturating_mul(other.len())); for a in self.iter() { for b in other.iter() { out.push(vec![a.clone(), b.clone()].into()); } } out } } // ------------------------------ pub(crate) trait Complement { fn complement(self, other: T) -> T; } impl Complement for Array { fn complement(self, other: Self) -> Array { let mut out = Array::new(); for v in self.into_iter() { if !other.contains(&v) { out.push(v) } } out } } // ------------------------------ pub(crate) trait Concat { fn concat(self, other: T) -> T; } impl Concat for Array { fn concat(mut self, mut other: Array) -> Array { self.append(&mut other); self } } // ------------------------------ pub(crate) trait Difference { fn difference(self, other: T) -> T; } impl Difference for Array { fn difference(self, mut other: Array) -> Array { let mut out = Array::new(); for v in self.into_iter() { if let Some(pos) = other.iter().position(|w| v == *w) { other.remove(pos); } else { out.push(v); } } out.append(&mut other); out } } // ------------------------------ pub(crate) trait Flatten { fn flatten(self) -> T; } impl Flatten for Array { fn flatten(self) -> Array { let mut out = Array::new(); for v in self.into_iter() { match v { Value::Array(mut a) => out.append(&mut a), _ => out.push(v), } } out } } // ------------------------------ pub(crate) trait Intersect { fn intersect(self, other: T) -> T; } impl Intersect for Array { fn intersect(self, mut other: Self) -> Self { let mut out = Self::new(); for v in self.0.into_iter() { if let Some(pos) = other.iter().position(|w| v == *w) { other.remove(pos); out.push(v); } } out } } // ------------------------------ // Documented with the assumption that it is just for arrays. pub(crate) trait Matches { /// Returns an array complimenting the origional where each value is true or false /// depending on whether it is == to the compared value. /// /// Admittedly, this is most often going to be used in `count(array::matches($arr, $val))` /// to count the number of times an element appears in an array but it's nice to have /// this in addition. fn matches(self, compare_val: Value) -> T; } impl Matches for Array { fn matches(self, compare_val: Value) -> Array { self.iter().map(|arr_val| (arr_val == &compare_val).into()).collect::>().into() } } // ------------------------------ // Documented with the assumption that it is just for arrays. pub(crate) trait Transpose { /// Stacks arrays on top of each other. This can serve as 2d array transposition. /// /// The input array can contain regular values which are treated as arrays with /// a single element. /// /// It's best to think of the function as creating a layered structure of the arrays /// rather than transposing them when the input is not a 2d array. See the examples /// for what happense when the input arrays are not all the same size. /// /// Here's a diagram: /// [0, 1, 2, 3], [4, 5, 6] /// -> /// [0 | 1 | 2 | 3] /// [4 | 5 | 6 ] /// ^ ^ ^ ^ /// [0, 4] [1, 5] [2, 6] [3] /// /// # Examples /// /// ```ignore /// fn array(sql: &str) -> Array { /// unimplemented!(); /// } /// /// // Example of `transpose` doing what it says on the tin. /// assert_eq!(array("[[0, 1], [2, 3]]").transpose(), array("[[0, 2], [1, 3]]")); /// // `transpose` can be thought of layering arrays on top of each other so when /// // one array runs out, it stops appearing in the output. /// assert_eq!(array("[[0, 1], [2]]").transpose(), array("[[0, 2], [1]]")); /// assert_eq!(array("[0, 1, 2]").transpose(), array("[[0, 1, 2]]")); /// ``` fn transpose(self) -> T; } impl Transpose for Array { fn transpose(self) -> Array { if self.len() == 0 { return self; } // I'm sure there's a way more efficient way to do this that I don't know about. // The new array will be at *least* this large so we can start there; let mut transposed_vec = Vec::::with_capacity(self.len()); let mut iters = self .iter() .map(|v| { if let Value::Array(arr) = v { Box::new(arr.iter().cloned()) as Box> } else { Box::new(std::iter::once(v).cloned()) as Box> } }) .collect::>(); // We know there is at least one element in the array therefore iters is not empty. // This is safe. let longest_length = iters.iter().map(|i| i.len()).max().unwrap(); for _ in 0..longest_length { transposed_vec .push(iters.iter_mut().filter_map(|i| i.next()).collect::>().into()); } transposed_vec.into() } } // ------------------------------ pub(crate) trait Union { fn union(self, other: T) -> T; } impl Union for Array { fn union(mut self, mut other: Self) -> Array { self.append(&mut other); self.uniq() } } // ------------------------------ pub(crate) trait Uniq { fn uniq(self) -> T; } impl Uniq for Array { fn uniq(mut self) -> Array { let mut set: HashSet<&Value> = HashSet::new(); let mut to_remove: Vec = Vec::new(); for (i, item) in self.iter().enumerate() { if !set.insert(item) { to_remove.push(i); } } for i in to_remove.iter().rev() { self.remove(*i); } self } } // ------------------------------ pub fn array(i: &str) -> IResult<&str, Array> { let (i, _) = openbracket(i)?; let (i, v) = separated_list0(commas, value)(i)?; let (i, _) = mightbespace(i)?; let (i, _) = opt(char(','))(i)?; let (i, _) = closebracket(i)?; Ok((i, Array(v))) } #[cfg(test)] mod tests { use super::*; #[test] fn array_empty() { let sql = "[]"; let res = array(sql); assert!(res.is_ok()); let out = res.unwrap().1; assert_eq!("[]", format!("{}", out)); assert_eq!(out.0.len(), 0); } #[test] fn array_normal() { let sql = "[1,2,3]"; let res = array(sql); assert!(res.is_ok()); let out = res.unwrap().1; assert_eq!("[1, 2, 3]", format!("{}", out)); assert_eq!(out.0.len(), 3); } #[test] fn array_commas() { let sql = "[1,2,3,]"; let res = array(sql); assert!(res.is_ok()); let out = res.unwrap().1; assert_eq!("[1, 2, 3]", format!("{}", out)); assert_eq!(out.0.len(), 3); } #[test] fn array_expression() { let sql = "[1,2,3+1]"; let res = array(sql); assert!(res.is_ok()); let out = res.unwrap().1; assert_eq!("[1, 2, 3 + 1]", format!("{}", out)); assert_eq!(out.0.len(), 3); } #[test] fn array_fnc_clump() { fn test(input_sql: &str, clump_size: usize, expected_result: &str) { let arr_result = array(input_sql); assert!(arr_result.is_ok()); let arr = arr_result.unwrap().1; let clumped_arr = arr.clump(clump_size); assert_eq!(format!("{}", clumped_arr), expected_result); } test("[0, 1, 2, 3]", 2, "[[0, 1], [2, 3]]"); test("[0, 1, 2, 3, 4, 5]", 3, "[[0, 1, 2], [3, 4, 5]]"); test("[0, 1, 2]", 2, "[[0, 1], [2]]"); test("[]", 2, "[]"); } #[test] fn array_fnc_transpose() { fn test(input_sql: &str, expected_result: &str) { let arr_result = array(input_sql); assert!(arr_result.is_ok()); let arr = arr_result.unwrap().1; let transposed_arr = arr.transpose(); assert_eq!(format!("{}", transposed_arr), expected_result); } test("[[0, 1], [2, 3]]", "[[0, 2], [1, 3]]"); test("[[0, 1], [2]]", "[[0, 2], [1]]"); test("[[0, 1, 2], [true, false]]", "[[0, true], [1, false], [2]]"); test("[[0, 1], [2, 3], [4, 5]]", "[[0, 2, 4], [1, 3, 5]]"); } #[test] fn array_fnc_uniq_normal() { let sql = "[1,2,1,3,3,4]"; let res = array(sql); assert!(res.is_ok()); let out = res.unwrap().1.uniq(); assert_eq!("[1, 2, 3, 4]", format!("{}", out)); assert_eq!(out.0.len(), 4); } }