risingwave_frontend/optimizer/plan_rewriter/mod.rs
1// Copyright 2025 RisingWave Labs
2//
3// Licensed under the Apache License, Version 2.0 (the "License");
4// you may not use this file except in compliance with the License.
5// You may obtain a copy of the License at
6//
7// http://www.apache.org/licenses/LICENSE-2.0
8//
9// Unless required by applicable law or agreed to in writing, software
10// distributed under the License is distributed on an "AS IS" BASIS,
11// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12// See the License for the specific language governing permissions and
13// limitations under the License.
14
15mod plan_cloner;
16mod share_source_rewriter;
17
18use itertools::Itertools;
19use paste::paste;
20pub use plan_cloner::*;
21pub use share_source_rewriter::*;
22
23use crate::for_all_plan_nodes;
24use crate::optimizer::plan_node::*;
25
26macro_rules! def_rewrite {
27 ( $convention:ident, Share ) => {
28 paste! {
29 /// When we use the plan rewriter, we need to take care of the share operator,
30 /// because our plan is a DAG rather than a tree.
31 /// Make sure this method can keep the shape of DAG.
32 fn [<rewrite_ $convention:snake _ share>](&mut self, plan: &[<$convention Share>]) -> PlanRef;
33 }
34 };
35
36 ( $convention:ident, $name:ident ) => {
37 paste! {
38 #[doc = "Visit [`" [<$convention $name>] "`] , the function should rewrite the inputs."]
39 fn [<rewrite_ $convention:snake _ $name:snake>](&mut self, plan: &[<$convention $name>]) -> PlanRef {
40 let new_inputs = plan
41 .inputs()
42 .into_iter()
43 .map(|input| self.rewrite(input.clone()))
44 .collect_vec();
45 plan.clone_with_inputs(&new_inputs)
46 }
47 }
48 };
49}
50
51/// Define `PlanRewriter` trait.
52macro_rules! def_rewriter {
53 ($({ $convention:ident, $name:ident }),*) => {
54
55 /// it's kind of like a [`PlanVisitor<PlanRef>`](super::plan_visitor::PlanVisitor), but with default behaviour of each rewrite method
56 pub trait PlanRewriter {
57 paste! {
58 fn rewrite(&mut self, plan: PlanRef) -> PlanRef{
59 use risingwave_common::util::recursive::{tracker, Recurse};
60 use crate::session::current::notice_to_user;
61
62 tracker!().recurse(|t| {
63 if t.depth_reaches(PLAN_DEPTH_THRESHOLD) {
64 notice_to_user(PLAN_TOO_DEEP_NOTICE);
65 }
66
67 match plan.node_type() {
68 $(
69 PlanNodeType::[<$convention $name>] => self.[<rewrite_ $convention:snake _ $name:snake>](plan.downcast_ref::<[<$convention $name>]>().unwrap()),
70 )*
71 }
72 })
73 }
74
75 $(
76 def_rewrite! {$convention, $name}
77 )*
78 }
79 }
80 };
81}
82for_all_plan_nodes! { def_rewriter }