risingwave_frontend/optimizer/rule/
except_to_anti_join_rule.rsuse risingwave_pb::plan_common::JoinType;
use super::{BoxedRule, Rule};
use crate::optimizer::plan_node::generic::Agg;
use crate::optimizer::plan_node::{LogicalExcept, LogicalJoin, PlanTreeNode};
use crate::optimizer::rule::IntersectToSemiJoinRule;
use crate::optimizer::PlanRef;
pub struct ExceptToAntiJoinRule {}
impl Rule for ExceptToAntiJoinRule {
fn apply(&self, plan: PlanRef) -> Option<PlanRef> {
let logical_except: &LogicalExcept = plan.as_logical_except()?;
let all = logical_except.all();
if all {
return None;
}
let inputs = logical_except.inputs();
let join = inputs
.into_iter()
.fold(None, |left, right| match left {
None => Some(right),
Some(left) => {
let on =
IntersectToSemiJoinRule::gen_null_safe_equal(left.clone(), right.clone());
Some(LogicalJoin::create(left, right, JoinType::LeftAnti, on))
}
})
.unwrap();
Some(Agg::new(vec![], (0..join.schema().len()).collect(), join).into())
}
}
impl ExceptToAntiJoinRule {
pub fn create() -> BoxedRule {
Box::new(ExceptToAntiJoinRule {})
}
}