use crate::ctx::Context; use crate::dbs::{Options, Transaction}; use crate::err::Error; use crate::idx::planner::plan::{IndexOperator, IndexOption}; use crate::sql::index::Index; use crate::sql::statements::DefineIndexStatement; use crate::sql::{Array, Cond, Expression, Idiom, Operator, Subquery, Table, Value, With}; use async_recursion::async_recursion; use std::collections::HashMap; use std::sync::Arc; pub(super) struct Tree {} impl Tree { /// Traverse all the conditions and extract every expression /// that can be resolved by an index. pub(super) async fn build<'a>( ctx: &'a Context<'_>, opt: &'a Options, txn: &'a Transaction, table: &'a Table, cond: &'a Option, with: &'a Option, ) -> Result)>, Error> { let with_indexes = match with { Some(With::Index(ixs)) => Vec::with_capacity(ixs.len()), _ => vec![], }; let mut b = TreeBuilder { ctx, opt, txn, table, with, indexes: None, index_lookup: Default::default(), index_map: IndexMap::default(), with_indexes, }; let mut res = None; if let Some(cond) = cond { res = Some((b.eval_value(&cond.0).await?, b.index_map, b.with_indexes)); } Ok(res) } } struct TreeBuilder<'a> { ctx: &'a Context<'a>, opt: &'a Options, txn: &'a Transaction, table: &'a Table, with: &'a Option, indexes: Option>, index_lookup: HashMap, index_map: IndexMap, with_indexes: Vec, } impl<'a> TreeBuilder<'a> { async fn find_index(&mut self, i: &Idiom) -> Result, Error> { if let Some(ir) = self.index_lookup.get(i) { return Ok(Some(*ir)); } if self.indexes.is_none() { let indexes = self .txn .clone() .lock() .await .all_tb_indexes(self.opt.ns(), self.opt.db(), &self.table.0) .await?; self.indexes = Some(indexes); } if let Some(indexes) = &self.indexes { for ix in indexes.as_ref() { if ix.cols.len() == 1 && ix.cols[0].eq(i) { let ir = self.index_lookup.len() as IndexRef; if let Some(With::Index(ixs)) = self.with { if ixs.contains(&ix.name.0) { self.with_indexes.push(ir); } } self.index_lookup.insert(i.clone(), ir); self.index_map.definitions.insert(ir, ix.clone()); return Ok(Some(ir)); } } } Ok(None) } #[cfg_attr(not(target_arch = "wasm32"), async_recursion)] #[cfg_attr(target_arch = "wasm32", async_recursion(?Send))] async fn eval_value(&mut self, v: &Value) -> Result { match v { Value::Expression(e) => self.eval_expression(e).await, Value::Idiom(i) => self.eval_idiom(i).await, Value::Strand(_) => Ok(Node::Scalar(v.to_owned())), Value::Number(_) => Ok(Node::Scalar(v.to_owned())), Value::Bool(_) => Ok(Node::Scalar(v.to_owned())), Value::Thing(_) => Ok(Node::Scalar(v.to_owned())), Value::Subquery(s) => self.eval_subquery(s).await, Value::Param(p) => { let v = p.compute(self.ctx, self.opt, self.txn, None).await?; self.eval_value(&v).await } _ => Ok(Node::Unsupported(format!("Unsupported value: {}", v))), } } async fn eval_idiom(&mut self, i: &Idiom) -> Result { Ok(if let Some(ix) = self.find_index(i).await? { Node::IndexedField(i.to_owned(), ix) } else { Node::NonIndexedField }) } async fn eval_expression(&mut self, e: &Expression) -> Result { match e { Expression::Unary { .. } => Ok(Node::Unsupported("unary expressions not supported".to_string())), Expression::Binary { l, o, r, } => { let left = self.eval_value(l).await?; let right = self.eval_value(r).await?; if let Some(io) = self.index_map.options.get(e) { return Ok(Node::Expression { io: Some(io.clone()), left: Box::new(left), right: Box::new(right), exp: Arc::new(e.clone()), }); } let mut io = None; if let Some((id, ix)) = left.is_indexed_field() { io = self.lookup_index_option(ix, o, id, &right, e); } else if let Some((id, ix)) = right.is_indexed_field() { io = self.lookup_index_option(ix, o, id, &left, e); }; Ok(Node::Expression { io, left: Box::new(left), right: Box::new(right), exp: Arc::new(e.clone()), }) } } } fn lookup_index_option( &mut self, ir: IndexRef, op: &Operator, id: &Idiom, v: &Node, e: &Expression, ) -> Option { if let Some(v) = v.is_scalar() { if let Some(ix) = self.index_map.definitions.get(&ir) { let op = match &ix.index { Index::Idx => Self::eval_index_operator(op, v), Index::Uniq => Self::eval_index_operator(op, v), Index::Search { .. } => { if let Operator::Matches(mr) = op { Some(IndexOperator::Matches(v.clone().to_raw_string(), *mr)) } else { None } } Index::MTree(_) => None, }; if let Some(op) = op { let io = IndexOption::new(ir, id.clone(), op); self.index_map.options.insert(Arc::new(e.clone()), io.clone()); return Some(io); } } } None } fn eval_index_operator(op: &Operator, v: &Value) -> Option { match op { Operator::Equal => Some(IndexOperator::Equality(Array::from(v.clone()))), Operator::LessThan | Operator::LessThanOrEqual | Operator::MoreThan | Operator::MoreThanOrEqual => Some(IndexOperator::RangePart(op.clone(), v.clone())), _ => None, } } async fn eval_subquery(&mut self, s: &Subquery) -> Result { match s { Subquery::Value(v) => self.eval_value(v).await, _ => Ok(Node::Unsupported(format!("Unsupported subquery: {}", s))), } } } pub(super) type IndexRef = u16; /// For each expression the a possible index option #[derive(Default)] pub(super) struct IndexMap { pub(super) options: HashMap, IndexOption>, pub(super) definitions: HashMap, } #[derive(Debug, Clone, Eq, PartialEq, Hash)] pub(super) enum Node { Expression { io: Option, left: Box, right: Box, exp: Arc, }, IndexedField(Idiom, IndexRef), NonIndexedField, Scalar(Value), Unsupported(String), } impl Node { pub(super) fn is_scalar(&self) -> Option<&Value> { if let Node::Scalar(v) = self { Some(v) } else { None } } pub(super) fn is_indexed_field(&self) -> Option<(&Idiom, IndexRef)> { if let Node::IndexedField(id, ix) = self { Some((id, *ix)) } else { None } } }