From 8620635483ce3d1bf063b381e2bb0b4abf7b4856 Mon Sep 17 00:00:00 2001 From: Kev Wang Date: Tue, 17 Dec 2024 14:21:09 -0800 Subject: [PATCH] perf: filter null join key optimization rule (#3583) --- .../src/optimization/optimizer.rs | 8 +- .../rules/filter_null_join_key.rs | 300 ++++++++++++++++++ .../src/optimization/rules/mod.rs | 2 + 3 files changed, 307 insertions(+), 3 deletions(-) create mode 100644 src/daft-logical-plan/src/optimization/rules/filter_null_join_key.rs diff --git a/src/daft-logical-plan/src/optimization/optimizer.rs b/src/daft-logical-plan/src/optimization/optimizer.rs index 76f6251438..64ed0a65c9 100644 --- a/src/daft-logical-plan/src/optimization/optimizer.rs +++ b/src/daft-logical-plan/src/optimization/optimizer.rs @@ -6,9 +6,10 @@ use common_treenode::Transformed; use super::{ logical_plan_tracker::LogicalPlanTracker, rules::{ - DropRepartition, EliminateCrossJoin, EnrichWithStats, LiftProjectFromAgg, MaterializeScans, - OptimizerRule, PushDownFilter, PushDownLimit, PushDownProjection, SimplifyExpressionsRule, - SplitActorPoolProjects, UnnestPredicateSubquery, UnnestScalarSubquery, + DropRepartition, EliminateCrossJoin, EnrichWithStats, FilterNullJoinKey, + LiftProjectFromAgg, MaterializeScans, OptimizerRule, PushDownFilter, PushDownLimit, + PushDownProjection, SimplifyExpressionsRule, SplitActorPoolProjects, + UnnestPredicateSubquery, UnnestScalarSubquery, }, }; use crate::LogicalPlan; @@ -109,6 +110,7 @@ impl Optimizer { RuleBatch::new( vec![ Box::new(DropRepartition::new()), + Box::new(FilterNullJoinKey::new()), Box::new(PushDownFilter::new()), Box::new(PushDownProjection::new()), Box::new(EliminateCrossJoin::new()), diff --git a/src/daft-logical-plan/src/optimization/rules/filter_null_join_key.rs b/src/daft-logical-plan/src/optimization/rules/filter_null_join_key.rs new file mode 100644 index 0000000000..80aab76c18 --- /dev/null +++ b/src/daft-logical-plan/src/optimization/rules/filter_null_join_key.rs @@ -0,0 +1,300 @@ +use std::sync::Arc; + +use common_error::DaftResult; +use common_treenode::{Transformed, TreeNode}; +use daft_algebra::boolean::combine_conjunction; +use daft_core::join::JoinType; + +use super::OptimizerRule; +use crate::{ + ops::{Filter, Join}, + LogicalPlan, +}; + +/// Optimization rule for filtering out nulls from join keys. +/// +/// When a join will always discard null keys from a join side, +/// this rule inserts a filter before that side to remove rows where a join key is null. +/// This reduces the cardinality of the tables before a join to improve join performance, +/// and can also be pushed down with other rules to reduce source and intermediate output sizes. +/// +/// # Example +/// ```sql +/// SELECT * FROM left JOIN right ON left.x = right.y +/// ``` +/// turns into +/// ```sql +/// SELECT * +/// FROM (SELECT * FROM left WHERE x IS NOT NULL) AS non_null_left +/// JOIN (SELECT * FROM right WHERE x IS NOT NULL) AS non_null_right +/// ON non_null_left.x = non_null_right.y +/// ``` +/// +/// So if `left` was: +/// ``` +/// ╭───────╮ +/// │ x │ +/// │ --- │ +/// │ Int64 │ +/// ╞═══════╡ +/// │ 1 │ +/// ├╌╌╌╌╌╌╌┤ +/// │ 2 │ +/// ├╌╌╌╌╌╌╌┤ +/// │ None │ +/// ╰───────╯ +/// ``` +/// And `right` was: +/// ``` +/// ╭───────╮ +/// │ y │ +/// │ --- │ +/// │ Int64 │ +/// ╞═══════╡ +/// │ 1 │ +/// ├╌╌╌╌╌╌╌┤ +/// │ None │ +/// ├╌╌╌╌╌╌╌┤ +/// │ None │ +/// ╰───────╯ +/// ``` +/// the original query would join on all rows, whereas the new query would first filter out null rows and join on the following: +/// +/// `non_null_left`: +/// ``` +/// ╭───────╮ +/// │ x │ +/// │ --- │ +/// │ Int64 │ +/// ╞═══════╡ +/// │ 1 │ +/// ├╌╌╌╌╌╌╌┤ +/// │ 2 │ +/// ╰───────╯ +/// ``` +/// `non_null_right`: +/// ``` +/// ╭───────╮ +/// │ y │ +/// │ --- │ +/// │ Int64 │ +/// ╞═══════╡ +/// │ 1 │ +/// ╰───────╯ +/// ``` +#[derive(Default, Debug)] +pub struct FilterNullJoinKey {} + +impl FilterNullJoinKey { + pub fn new() -> Self { + Self {} + } +} + +impl OptimizerRule for FilterNullJoinKey { + fn try_optimize(&self, plan: Arc) -> DaftResult>> { + plan.transform(|node| { + if let LogicalPlan::Join(Join { + left, + right, + left_on, + right_on, + null_equals_nulls, + join_type, + .. + }) = node.as_ref() + { + let mut null_equals_nulls_iter = null_equals_nulls.as_ref().map_or_else( + || Box::new(std::iter::repeat(false)) as Box>, + |x| Box::new(x.clone().into_iter()), + ); + + let (can_filter_left, can_filter_right) = match join_type { + JoinType::Inner => (true, true), + JoinType::Left => (false, true), + JoinType::Right => (true, false), + JoinType::Outer => (false, false), + JoinType::Anti => (false, true), + JoinType::Semi => (true, true), + }; + + let left_null_pred = if can_filter_left { + combine_conjunction( + null_equals_nulls_iter + .by_ref() + .zip(left_on) + .filter(|(null_eq_null, _)| !null_eq_null) + .map(|(_, left_key)| left_key.clone().is_null().not()), + ) + } else { + None + }; + + let right_null_pred = if can_filter_right { + combine_conjunction( + null_equals_nulls_iter + .by_ref() + .zip(right_on) + .filter(|(null_eq_null, _)| !null_eq_null) + .map(|(_, right_key)| right_key.clone().is_null().not()), + ) + } else { + None + }; + + if left_null_pred.is_none() && right_null_pred.is_none() { + Ok(Transformed::no(node.clone())) + } else { + let new_left = if let Some(pred) = left_null_pred { + Arc::new(LogicalPlan::Filter(Filter::try_new(left.clone(), pred)?)) + } else { + left.clone() + }; + + let new_right = if let Some(pred) = right_null_pred { + Arc::new(LogicalPlan::Filter(Filter::try_new(right.clone(), pred)?)) + } else { + right.clone() + }; + + let new_join = Arc::new(node.with_new_children(&[new_left, new_right])); + + Ok(Transformed::yes(new_join)) + } + } else { + Ok(Transformed::no(node)) + } + }) + } +} + +#[cfg(test)] +mod tests { + use std::sync::Arc; + + use common_error::DaftResult; + use daft_core::prelude::*; + use daft_dsl::col; + + use crate::{ + optimization::{ + optimizer::{RuleBatch, RuleExecutionStrategy}, + rules::filter_null_join_key::FilterNullJoinKey, + test::assert_optimized_plan_with_rules_eq, + }, + test::{dummy_scan_node, dummy_scan_operator}, + LogicalPlan, + }; + + /// Helper that creates an optimizer with the FilterNullJoinKey rule registered, optimizes + /// the provided plan with said optimizer, and compares the optimized plan with + /// the provided expected plan. + fn assert_optimized_plan_eq( + plan: Arc, + expected: Arc, + ) -> DaftResult<()> { + assert_optimized_plan_with_rules_eq( + plan, + expected, + vec![RuleBatch::new( + vec![Box::new(FilterNullJoinKey::new())], + RuleExecutionStrategy::Once, + )], + ) + } + + #[test] + fn filter_keys_basic() -> DaftResult<()> { + let left_scan = dummy_scan_node(dummy_scan_operator(vec![ + Field::new("a", DataType::Int64), + Field::new("b", DataType::Utf8), + ])); + + let right_scan = dummy_scan_node(dummy_scan_operator(vec![ + Field::new("c", DataType::Int64), + Field::new("d", DataType::Utf8), + ])); + + let plan = left_scan + .join( + right_scan.clone(), + vec![col("a")], + vec![col("c")], + JoinType::Inner, + None, + None, + None, + false, + )? + .build(); + + let expected = left_scan + .filter(col("a").is_null().not())? + .clone() + .join( + right_scan.filter(col("c").is_null().not())?, + vec![col("a")], + vec![col("c")], + JoinType::Inner, + None, + None, + None, + false, + )? + .build(); + + assert_optimized_plan_eq(plan, expected)?; + + Ok(()) + } + + #[test] + fn filter_keys_null_equals_nulls() -> DaftResult<()> { + let left_scan = dummy_scan_node(dummy_scan_operator(vec![ + Field::new("a", DataType::Int64), + Field::new("b", DataType::Utf8), + Field::new("c", DataType::Boolean), + ])); + + let right_scan = dummy_scan_node(dummy_scan_operator(vec![ + Field::new("d", DataType::Int64), + Field::new("e", DataType::Utf8), + Field::new("f", DataType::Boolean), + ])); + + let plan = left_scan + .join_with_null_safe_equal( + right_scan.clone(), + vec![col("a"), col("b"), col("c")], + vec![col("d"), col("e"), col("f")], + Some(vec![false, true, false]), + JoinType::Left, + None, + None, + None, + false, + )? + .build(); + + let expected_predicate = col("d").is_null().not().and(col("f").is_null().not()); + + let expected = left_scan + .clone() + .join_with_null_safe_equal( + right_scan.filter(expected_predicate)?, + vec![col("a"), col("b"), col("c")], + vec![col("d"), col("e"), col("f")], + Some(vec![false, true, false]), + JoinType::Left, + None, + None, + None, + false, + )? + .build(); + + assert_optimized_plan_eq(plan, expected)?; + + Ok(()) + } +} diff --git a/src/daft-logical-plan/src/optimization/rules/mod.rs b/src/daft-logical-plan/src/optimization/rules/mod.rs index f540a77cb0..787c0e7ac6 100644 --- a/src/daft-logical-plan/src/optimization/rules/mod.rs +++ b/src/daft-logical-plan/src/optimization/rules/mod.rs @@ -1,6 +1,7 @@ mod drop_repartition; mod eliminate_cross_join; mod enrich_with_stats; +mod filter_null_join_key; mod lift_project_from_agg; mod materialize_scans; mod push_down_filter; @@ -15,6 +16,7 @@ mod unnest_subquery; pub use drop_repartition::DropRepartition; pub use eliminate_cross_join::EliminateCrossJoin; pub use enrich_with_stats::EnrichWithStats; +pub use filter_null_join_key::FilterNullJoinKey; pub use lift_project_from_agg::LiftProjectFromAgg; pub use materialize_scans::MaterializeScans; pub use push_down_filter::PushDownFilter;