risingwave_frontend/optimizer/plan_node/
plan_tree_node.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
15use smallvec::SmallVec;
16
17use super::PlanRef;
18use crate::utils::ColIndexMapping;
19
20/// The trait [`PlanNode`](super::PlanNode) really need about tree structure and used by optimizer
21/// framework. every plan node should impl it.
22///
23/// The trait [`PlanTreeNodeLeaf`], [`PlanTreeNodeUnary`] and [`PlanTreeNodeBinary`], is just
24/// special cases for [`PlanTreeNode`]. as long as you impl these trait for a plan node, we can
25/// easily impl the [`PlanTreeNode`] which is really need by framework with helper macros
26/// `impl_plan_tree_node_for_leaf`, `impl_plan_tree_node_for_unary` and
27/// `impl_plan_tree_node_for_binary`. We can't auto `impl PlanTreeNode for
28/// PlanTreeNodeLeaf/Unary/Binary`, because compiler doesn't know they are disjoint and thinks there
29/// are conflicting implementation.
30///
31/// And due to these three traits need not be used as dyn, it can return `Self` type, which is
32/// useful when implement rules and visitors. So we highly recommend not impl the [`PlanTreeNode`]
33/// trait directly, instead use these tree trait and impl [`PlanTreeNode`] use these helper
34/// macros.
35pub trait PlanTreeNode {
36    /// Get input nodes of the plan.
37    fn inputs(&self) -> SmallVec<[PlanRef; 2]>;
38
39    /// Clone the node with a list of new inputs.
40    fn clone_with_inputs(&self, inputs: &[PlanRef]) -> PlanRef;
41}
42
43/// See [`PlanTreeNode`](super)
44pub trait PlanTreeNodeLeaf: Clone {}
45
46/// See [`PlanTreeNode`](super)
47pub trait PlanTreeNodeUnary {
48    fn input(&self) -> PlanRef;
49    #[must_use]
50    fn clone_with_input(&self, input: PlanRef) -> Self;
51
52    /// Rewrites the plan node according to the schema change of its input node during rewriting.
53    ///
54    /// This function can be used to implement [`prune_col`](super::ColPrunable::prune_col) or
55    /// [`logical_rewrite_for_stream`](super::ToStream::logical_rewrite_for_stream)
56    #[must_use]
57    fn rewrite_with_input(
58        &self,
59        _input: PlanRef,
60        _input_col_change: ColIndexMapping,
61    ) -> (Self, ColIndexMapping)
62    where
63        Self: Sized,
64    {
65        unimplemented!()
66    }
67}
68
69/// See [`PlanTreeNode`](super)
70pub trait PlanTreeNodeBinary {
71    fn left(&self) -> PlanRef;
72    fn right(&self) -> PlanRef;
73
74    #[must_use]
75    fn clone_with_left_right(&self, left: PlanRef, right: PlanRef) -> Self;
76
77    /// Rewrites the plan node according to the schema change of its input nodes during rewriting.
78    ///
79    /// This function can be used to implement [`prune_col`](super::ColPrunable::prune_col) or
80    /// [`logical_rewrite_for_stream`](super::ToStream::logical_rewrite_for_stream)
81    #[must_use]
82    fn rewrite_with_left_right(
83        &self,
84        _left: PlanRef,
85        _left_col_change: ColIndexMapping,
86        _right: PlanRef,
87        _right_col_change: ColIndexMapping,
88    ) -> (Self, ColIndexMapping)
89    where
90        Self: Sized,
91    {
92        unimplemented!()
93    }
94}
95
96macro_rules! impl_plan_tree_node_for_leaf {
97    ($leaf_node_type:ident) => {
98        impl crate::optimizer::plan_node::PlanTreeNode for $leaf_node_type {
99            fn inputs(&self) -> smallvec::SmallVec<[crate::optimizer::PlanRef; 2]> {
100                smallvec::smallvec![]
101            }
102
103            /// Clone the node with a list of new inputs.
104            fn clone_with_inputs(
105                &self,
106                inputs: &[crate::optimizer::PlanRef],
107            ) -> crate::optimizer::PlanRef {
108                assert_eq!(inputs.len(), 0);
109                self.clone().into()
110            }
111        }
112    };
113}
114
115macro_rules! impl_plan_tree_node_for_unary {
116    ($unary_node_type:ident) => {
117        impl crate::optimizer::plan_node::PlanTreeNode for $unary_node_type {
118            fn inputs(&self) -> smallvec::SmallVec<[crate::optimizer::PlanRef; 2]> {
119                smallvec::smallvec![self.input()]
120            }
121
122            /// Clone the node with a list of new inputs.
123            fn clone_with_inputs(
124                &self,
125                inputs: &[crate::optimizer::PlanRef],
126            ) -> crate::optimizer::PlanRef {
127                assert_eq!(inputs.len(), 1);
128                self.clone_with_input(inputs[0].clone()).into()
129            }
130        }
131    };
132}
133
134macro_rules! impl_plan_tree_node_for_binary {
135    ($binary_node_type:ident) => {
136        impl crate::optimizer::plan_node::PlanTreeNode for $binary_node_type {
137            fn inputs(&self) -> smallvec::SmallVec<[crate::optimizer::PlanRef; 2]> {
138                smallvec::smallvec![self.left(), self.right()]
139            }
140
141            fn clone_with_inputs(
142                &self,
143                inputs: &[crate::optimizer::PlanRef],
144            ) -> crate::optimizer::PlanRef {
145                assert_eq!(inputs.len(), 2);
146                self.clone_with_left_right(inputs[0].clone(), inputs[1].clone())
147                    .into()
148            }
149        }
150    };
151}