Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add argument lifetime to BaseTableExpr's evaluate #448

Open
wants to merge 7 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
5 changes: 5 additions & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -9,13 +9,18 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
### Changed
- Adds quotes to the attributes of PartiQL tuple's debug output so it can be read and transformed using Kotlin `partiql-cli`
- [breaking] Changes the interface to `EvalPlan` to accept an `EvalContext`
- [breaking] Changes `EvaluationError` to not implement `Clone`
- [breaking] Changes the structure of `EvalPlan`
- [breaking] Changes the interface of `BaseTableExpr`'s `evaluate` to allow arguments to be held by the returned iterator

### Added
- Add `partiql-extension-visualize` for visualizing AST and logical plan
- Add a `SessionContext` containing both a system-level and a user-level context object usable by expression evaluation

### Fixed
- Fixed `ORDER BY`'s ability to see into projection aliases
- Fixed errors in `BaseTableExpr`s get added to the evaluation context
- Fixed certain errors surfacing in Permissive evaluation mode, when they should only be present in Strict mode

## [0.6.0] - 2023-10-31
### Changed
Expand Down
9 changes: 6 additions & 3 deletions extension/partiql-extension-ion-functions/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -99,11 +99,14 @@ impl BaseTableFunctionInfo for ReadIonFunction {
pub(crate) struct EvalFnReadIon {}

impl BaseTableExpr for EvalFnReadIon {
fn evaluate<'c>(
fn evaluate<'a, 'c>(
&self,
args: &[Cow<Value>],
args: &'a [Cow<Value>],
Copy link

@sadderchris sadderchris Mar 7, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

There's also an implicit lifetime on the Cow as well (the type is actually Cow<'_, Value>). Can you check this doesn't also need to be named?

As a nit: you may want to throw #![warn(rust_2018_idioms)] or even #![deny(rust_2018_idioms)] at the top of your crates' lib.rs to enforce that lifetimes always get written out (unless you feel really strongly about the added verbosity - IMO the benefits of this lint outweigh the downsides). It's easy to forget about implicit lifetimes otherwise.

_ctx: &'c dyn SessionContext<'c>,
) -> BaseTableExprResult<'c> {
) -> BaseTableExprResult<'a>
where
'c: 'a,
{
if let Some(arg1) = args.first() {
match arg1.as_ref() {
Value::String(path) => parse_ion_file(path),
Expand Down
9 changes: 6 additions & 3 deletions partiql-catalog/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -46,17 +46,20 @@ pub struct ObjectId {
}

pub type BaseTableExprResultError = Box<dyn Error>;

pub type BaseTableExprResultValueIter<'a> =
Box<dyn 'a + Iterator<Item = Result<Value, BaseTableExprResultError>>>;
pub type BaseTableExprResult<'a> =
Result<BaseTableExprResultValueIter<'a>, BaseTableExprResultError>;

pub trait BaseTableExpr: Debug {
fn evaluate<'c>(
fn evaluate<'a, 'c>(
&self,
args: &[Cow<Value>],
args: &'a [Cow<Value>],
ctx: &'c dyn SessionContext<'c>,
) -> BaseTableExprResult<'c>;
) -> BaseTableExprResult<'a>
where
'c: 'a;
}

pub trait BaseTableFunctionInfo: Debug {
Expand Down
15 changes: 13 additions & 2 deletions partiql-eval/src/error.rs
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
use crate::eval::evaluable::Evaluable;
use crate::eval::expr::EvalExpr;
use crate::eval::EvalContext;
use partiql_catalog::BaseTableExprResultError;
use partiql_value::{Tuple, Value};
use std::borrow::Cow;
use thiserror::Error;
Expand Down Expand Up @@ -30,7 +31,7 @@ pub struct EvalErr {
}

/// An error that can happen during evaluation.
#[derive(Error, Debug, Clone, PartialEq, Eq, Hash)]
#[derive(Error, Debug)]
#[non_exhaustive]
pub enum EvaluationError {
/// Internal error that was not due to user input or API violation.
Expand All @@ -42,9 +43,19 @@ pub enum EvaluationError {
/// Feature has not yet been implemented.
#[error("Not yet implemented: {0}")]
NotYetImplemented(String),

/// Error originating in an extension
#[error("Base Table Expression Error: {0}")]
ExtensioResultError(BaseTableExprResultError),
}

impl From<BaseTableExprResultError> for EvaluationError {
fn from(e: BaseTableExprResultError) -> Self {
EvaluationError::ExtensioResultError(e)
}
}

/// Used when an error occurs during the the logical to eval plan conversion. Allows the conversion
/// Used when an error occurs during the logical to eval plan conversion. Allows the conversion
/// to continue in order to report multiple errors.
#[derive(Debug)]
pub(crate) struct ErrorNode {}
Expand Down
8 changes: 4 additions & 4 deletions partiql-eval/src/eval/expr/base_table.rs
Original file line number Diff line number Diff line change
Expand Up @@ -37,14 +37,14 @@ impl EvalExpr for EvalFnBaseTableExpr {
let bag: Result<Bag, _> = it.collect();
match bag {
Ok(b) => Value::from(b),
Err(_) => {
// TODO hook into pending eval errors
Err(err) => {
ctx.add_error(err.into());
Missing
}
}
}
Err(_) => {
// TODO hook into pending eval errors
Err(err) => {
ctx.add_error(err.into());
Missing
}
};
Expand Down
23 changes: 15 additions & 8 deletions partiql-eval/src/eval/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -23,6 +23,7 @@ use petgraph::visit::EdgeRef;
use unicase::UniCase;

use crate::eval::evaluable::{EvalType, Evaluable};
use crate::plan::EvaluationMode;

pub(crate) mod eval_expr_wrapper;
pub mod evaluable;
Expand All @@ -31,11 +32,14 @@ pub mod expr;
/// Represents a PartiQL evaluation query plan which is a plan that can be evaluated to produce
/// a result. The plan uses a directed `petgraph::StableGraph`.
#[derive(Debug)]
pub struct EvalPlan(pub StableGraph<Box<dyn Evaluable>, u8, Directed>);
pub struct EvalPlan {
mode: EvaluationMode,
plan_graph: StableGraph<Box<dyn Evaluable>, u8, Directed>,
}

impl Default for EvalPlan {
fn default() -> Self {
Self::new()
Self::new(EvaluationMode::Permissive, Default::default())
}
}

Expand All @@ -48,13 +52,16 @@ fn err_illegal_state(msg: impl AsRef<str>) -> EvalErr {

impl EvalPlan {
/// Creates a new evaluation plan.
fn new() -> Self {
EvalPlan(StableGraph::<Box<dyn Evaluable>, u8, Directed>::new())
pub fn new(
mode: EvaluationMode,
plan_graph: StableGraph<Box<dyn Evaluable>, u8, Directed>,
) -> Self {
EvalPlan { mode, plan_graph }
}

#[inline]
fn plan_graph(&mut self) -> &mut StableGraph<Box<dyn Evaluable>, u8> {
&mut self.0
&mut self.plan_graph
}

#[inline]
Expand All @@ -73,7 +80,7 @@ impl EvalPlan {
// that all v ∈ V \{v0} are reachable from v0. Note that this is the definition of trees
// without the condition |E| = |V | − 1. Hence, all trees are DAGs.
// Reference: https://link.springer.com/article/10.1007/s00450-009-0061-0
let ops = toposort(&self.0, None).map_err(|e| EvalErr {
let ops = toposort(&self.plan_graph, None).map_err(|e| EvalErr {
errors: vec![EvaluationError::InvalidEvaluationPlan(format!(
"Malformed evaluation plan detected: {e:?}"
))],
Expand Down Expand Up @@ -101,7 +108,7 @@ impl EvalPlan {
result = Some(src.evaluate(ctx));

// return on first evaluation error
if ctx.has_errors() {
if ctx.has_errors() && self.mode == EvaluationMode::Strict {
return Err(EvalErr {
errors: ctx.errors(),
});
Expand All @@ -127,7 +134,7 @@ impl EvalPlan {
}

pub fn to_dot_graph(&self) -> String {
format!("{:?}", Dot::with_config(&self.0, &[]))
format!("{:?}", Dot::with_config(&self.plan_graph, &[]))
}
}

Expand Down
19 changes: 12 additions & 7 deletions partiql-eval/src/plan.rs
Original file line number Diff line number Diff line change
Expand Up @@ -49,6 +49,7 @@ macro_rules! correct_num_args_or_err {
};
}

#[derive(Debug, Eq, PartialEq)]
pub enum EvaluationMode {
Strict,
Permissive,
Expand Down Expand Up @@ -129,22 +130,26 @@ impl<'c> EvaluatorPlanner<'c> {
fn plan_eval<const STRICT: bool>(&mut self, lg: &LogicalPlan<BindingsOp>) -> EvalPlan {
let flows = lg.flows();

let mut graph: StableGraph<_, _> = Default::default();
let mut plan_graph: StableGraph<_, _> = Default::default();
let mut seen = HashMap::new();

for (s, d, branch_num) in flows {
let mut add_node = |op_id: &OpId| {
let logical_op = lg.operator(*op_id).unwrap();
*seen
.entry(*op_id)
.or_insert_with(|| graph.add_node(self.get_eval_node::<{ STRICT }>(logical_op)))
*seen.entry(*op_id).or_insert_with(|| {
plan_graph.add_node(self.get_eval_node::<{ STRICT }>(logical_op))
})
};

let (s, d) = (add_node(s), add_node(d));
graph.add_edge(s, d, *branch_num);
plan_graph.add_edge(s, d, *branch_num);
}

EvalPlan(graph)
let mode = if STRICT {
EvaluationMode::Strict
} else {
EvaluationMode::Permissive
};
EvalPlan::new(mode, plan_graph)
}

fn get_eval_node<const STRICT: bool>(&mut self, be: &BindingsOp) -> Box<dyn Evaluable> {
Expand Down
2 changes: 2 additions & 0 deletions partiql/Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -39,6 +39,8 @@ itertools = "0.12"
criterion = "0.5"
rand = "0.8"

assert_matches = "1.5"

[[bench]]
name = "bench_eval_multi_like"
harness = false
Expand Down
Loading
Loading