com.hazelcast.org.apache.calcite.rel.rules.LoptJoinTree Maven / Gradle / Ivy
/*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF licenses this file to you under the Apache License, Version 2.0
* (the "License"); you may not use this file except in compliance with
* the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package com.hazelcast.org.apache.calcite.rel.rules;
import com.hazelcast.org.apache.calcite.rel.RelNode;
import com.hazelcast.org.apache.calcite.rel.core.Join;
import com.hazelcast.org.checkerframework.checker.initialization.qual.NotOnlyInitialized;
import com.hazelcast.org.checkerframework.checker.initialization.qual.UnderInitialization;
import java.util.ArrayList;
import java.util.List;
import java.util.Objects;
/**
* Utility class used to store a {@link com.hazelcast.org.apache.calcite.rel.core.Join} tree
* and the factors that make up the tree.
*
* Because {@link RelNode}s can be duplicated in a query
* when you have a self-join, factor ids are needed to distinguish between the
* different join inputs that correspond to identical tables. The class
* associates factor ids with a join tree, matching the order of the factor ids
* with the order of those factors in the join tree.
*/
public class LoptJoinTree {
//~ Instance fields --------------------------------------------------------
@NotOnlyInitialized
private final BinaryTree factorTree;
private final RelNode joinTree;
private final boolean removableSelfJoin;
//~ Constructors -----------------------------------------------------------
/**
* Creates a join-tree consisting of a single node.
*
* @param joinTree RelNode corresponding to the single node
* @param factorId factor id of the node
*/
@SuppressWarnings("argument.type.incompatible")
public LoptJoinTree(RelNode joinTree, int factorId) {
this.joinTree = joinTree;
this.factorTree = new Leaf(factorId, this);
this.removableSelfJoin = false;
}
/**
* Associates the factor ids with a join-tree.
*
* @param joinTree RelNodes corresponding to the join tree
* @param factorTree tree of the factor ids
* @param removableSelfJoin whether the join corresponds to a removable
* self-join
*/
public LoptJoinTree(
RelNode joinTree,
BinaryTree factorTree,
boolean removableSelfJoin) {
this.joinTree = joinTree;
this.factorTree = factorTree;
this.removableSelfJoin = removableSelfJoin;
}
/**
* Associates the factor ids with a join-tree given the factors corresponding
* to the left and right subtrees of the join.
*
* @param joinTree RelNodes corresponding to the join tree
* @param leftFactorTree tree of the factor ids for left subtree
* @param rightFactorTree tree of the factor ids for the right subtree
*/
public LoptJoinTree(
RelNode joinTree,
BinaryTree leftFactorTree,
BinaryTree rightFactorTree) {
this(joinTree, leftFactorTree, rightFactorTree, false);
}
/**
* Associates the factor ids with a join-tree given the factors corresponding
* to the left and right subtrees of the join. Also indicates whether the
* join is a removable self-join.
*
* @param joinTree RelNodes corresponding to the join tree
* @param leftFactorTree tree of the factor ids for left subtree
* @param rightFactorTree tree of the factor ids for the right subtree
* @param removableSelfJoin true if the join is a removable self-join
*/
public LoptJoinTree(
RelNode joinTree,
BinaryTree leftFactorTree,
BinaryTree rightFactorTree,
boolean removableSelfJoin) {
factorTree = new Node(leftFactorTree, rightFactorTree, this);
this.joinTree = joinTree;
this.removableSelfJoin = removableSelfJoin;
}
//~ Methods ----------------------------------------------------------------
public RelNode getJoinTree() {
return joinTree;
}
public LoptJoinTree getLeft() {
final Node node = (Node) factorTree;
return new LoptJoinTree(
((Join) joinTree).getLeft(),
node.getLeft(),
node.getLeft().getParent().isRemovableSelfJoin());
}
public LoptJoinTree getRight() {
final Node node = (Node) factorTree;
return new LoptJoinTree(
((Join) joinTree).getRight(),
node.getRight(),
node.getRight().getParent().isRemovableSelfJoin());
}
public BinaryTree getFactorTree() {
return factorTree;
}
public List getTreeOrder() {
List treeOrder = new ArrayList<>();
getTreeOrder(treeOrder);
return treeOrder;
}
public void getTreeOrder(List treeOrder) {
factorTree.getTreeOrder(treeOrder);
}
public boolean isRemovableSelfJoin() {
return removableSelfJoin;
}
//~ Inner Classes ----------------------------------------------------------
/**
* Simple binary tree class that stores an id in the leaf nodes and keeps
* track of the parent LoptJoinTree object associated with the binary tree.
*/
protected abstract static class BinaryTree {
@NotOnlyInitialized
private final LoptJoinTree parent;
protected BinaryTree(@UnderInitialization LoptJoinTree parent) {
this.parent = parent;
}
public LoptJoinTree getParent() {
return parent;
}
public abstract void getTreeOrder(List treeOrder);
}
/** Binary tree node that has no children. */
protected static class Leaf extends BinaryTree {
private final int id;
public Leaf(int rootId, @UnderInitialization LoptJoinTree parent) {
super(parent);
this.id = rootId;
}
/** Returns the id associated with a leaf node in a binary tree. */
public int getId() {
return id;
}
@Override public void getTreeOrder(List treeOrder) {
treeOrder.add(id);
}
}
/** Binary tree node that has two children. */
protected static class Node extends BinaryTree {
private final BinaryTree left;
private final BinaryTree right;
public Node(BinaryTree left, BinaryTree right, @UnderInitialization LoptJoinTree parent) {
super(parent);
this.left = Objects.requireNonNull(left, "left");
this.right = Objects.requireNonNull(right, "right");
}
public BinaryTree getLeft() {
return left;
}
public BinaryTree getRight() {
return right;
}
@Override public void getTreeOrder(List treeOrder) {
left.getTreeOrder(treeOrder);
right.getTreeOrder(treeOrder);
}
}
}