com.hazelcast.org.apache.calcite.rel.logical.LogicalJoin 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.logical;
import com.hazelcast.org.apache.calcite.plan.Convention;
import com.hazelcast.org.apache.calcite.plan.RelOptCluster;
import com.hazelcast.org.apache.calcite.plan.RelTraitSet;
import com.hazelcast.org.apache.calcite.rel.RelInput;
import com.hazelcast.org.apache.calcite.rel.RelNode;
import com.hazelcast.org.apache.calcite.rel.RelShuttle;
import com.hazelcast.org.apache.calcite.rel.RelWriter;
import com.hazelcast.org.apache.calcite.rel.core.CorrelationId;
import com.hazelcast.org.apache.calcite.rel.core.Join;
import com.hazelcast.org.apache.calcite.rel.core.JoinRelType;
import com.hazelcast.org.apache.calcite.rel.hint.RelHint;
import com.hazelcast.org.apache.calcite.rel.type.RelDataTypeField;
import com.hazelcast.org.apache.calcite.rex.RexNode;
import com.hazelcast.com.google.common.collect.ImmutableList;
import com.hazelcast.com.google.common.collect.ImmutableSet;
import java.util.ArrayList;
import java.util.List;
import java.util.Objects;
import java.util.Set;
/**
* Sub-class of {@link com.hazelcast.org.apache.calcite.rel.core.Join}
* not targeted at any particular engine or calling convention.
*
* Some rules:
*
*
* - {@link com.hazelcast.org.apache.calcite.rel.rules.JoinExtractFilterRule} converts an
* {@link LogicalJoin inner join} to a {@link LogicalFilter filter} on top of a
* {@link LogicalJoin cartesian inner join}.
*
*
- {@code net.sf.farrago.fennel.rel.FennelCartesianJoinRule}
* implements a LogicalJoin as a cartesian product.
*
*
*/
public final class LogicalJoin extends Join {
//~ Instance fields --------------------------------------------------------
// NOTE jvs 14-Mar-2006: Normally we don't use state like this
// to control rule firing, but due to the non-local nature of
// semijoin optimizations, it's pretty much required.
private final boolean semiJoinDone;
private final ImmutableList systemFieldList;
//~ Constructors -----------------------------------------------------------
/**
* Creates a LogicalJoin.
*
* Use {@link #create} unless you know what you're doing.
*
* @param cluster Cluster
* @param traitSet Trait set
* @param hints Hints
* @param left Left input
* @param right Right input
* @param condition Join condition
* @param joinType Join type
* @param variablesSet Set of variables that are set by the
* LHS and used by the RHS and are not available to
* nodes above this LogicalJoin in the tree
* @param semiJoinDone Whether this join has been translated to a
* semi-join
* @param systemFieldList List of system fields that will be prefixed to
* output row type; typically empty but must not be
* null
* @see #isSemiJoinDone()
*/
public LogicalJoin(
RelOptCluster cluster,
RelTraitSet traitSet,
List hints,
RelNode left,
RelNode right,
RexNode condition,
Set variablesSet,
JoinRelType joinType,
boolean semiJoinDone,
ImmutableList systemFieldList) {
super(cluster, traitSet, hints, left, right, condition, variablesSet, joinType);
this.semiJoinDone = semiJoinDone;
this.systemFieldList = Objects.requireNonNull(systemFieldList);
}
@Deprecated // to be removed before 2.0
public LogicalJoin(RelOptCluster cluster, RelTraitSet traitSet,
RelNode left, RelNode right, RexNode condition, Set variablesSet,
JoinRelType joinType, boolean semiJoinDone,
ImmutableList systemFieldList) {
this(cluster, traitSet, ImmutableList.of(), left, right, condition,
variablesSet, joinType, semiJoinDone, systemFieldList);
}
@Deprecated // to be removed before 2.0
public LogicalJoin(RelOptCluster cluster, RelTraitSet traitSet, RelNode left,
RelNode right, RexNode condition, JoinRelType joinType,
Set variablesStopped, boolean semiJoinDone,
ImmutableList systemFieldList) {
this(cluster, traitSet, ImmutableList.of(), left, right, condition,
CorrelationId.setOf(variablesStopped), joinType, semiJoinDone,
systemFieldList);
}
@Deprecated // to be removed before 2.0
public LogicalJoin(RelOptCluster cluster, RelNode left, RelNode right,
RexNode condition, JoinRelType joinType, Set variablesStopped) {
this(cluster, cluster.traitSetOf(Convention.NONE), ImmutableList.of(),
left, right, condition, CorrelationId.setOf(variablesStopped),
joinType, false, ImmutableList.of());
}
@Deprecated // to be removed before 2.0
public LogicalJoin(RelOptCluster cluster, RelNode left, RelNode right,
RexNode condition, JoinRelType joinType, Set variablesStopped,
boolean semiJoinDone, ImmutableList systemFieldList) {
this(cluster, cluster.traitSetOf(Convention.NONE), ImmutableList.of(),
left, right, condition, CorrelationId.setOf(variablesStopped), joinType,
semiJoinDone, systemFieldList);
}
/**
* Creates a LogicalJoin by parsing serialized output.
*/
public LogicalJoin(RelInput input) {
this(input.getCluster(), input.getCluster().traitSetOf(Convention.NONE),
new ArrayList<>(),
input.getInputs().get(0), input.getInputs().get(1),
input.getExpression("condition"), ImmutableSet.of(),
input.getEnum("joinType", JoinRelType.class), false,
ImmutableList.of());
}
/** Creates a LogicalJoin. */
public static LogicalJoin create(RelNode left, RelNode right, List hints,
RexNode condition, Set variablesSet, JoinRelType joinType) {
return create(left, right, hints, condition, variablesSet, joinType, false,
ImmutableList.of());
}
/** Creates a LogicalJoin, flagged with whether it has been translated to a
* semi-join. */
public static LogicalJoin create(RelNode left, RelNode right, List hints,
RexNode condition, Set variablesSet, JoinRelType joinType,
boolean semiJoinDone, ImmutableList systemFieldList) {
final RelOptCluster cluster = left.getCluster();
final RelTraitSet traitSet = cluster.traitSetOf(Convention.NONE);
return new LogicalJoin(cluster, traitSet, hints, left, right, condition,
variablesSet, joinType, semiJoinDone, systemFieldList);
}
//~ Methods ----------------------------------------------------------------
@Override public LogicalJoin copy(RelTraitSet traitSet, RexNode conditionExpr,
RelNode left, RelNode right, JoinRelType joinType, boolean semiJoinDone) {
assert traitSet.containsIfApplicable(Convention.NONE);
return new LogicalJoin(getCluster(),
getCluster().traitSetOf(Convention.NONE), hints, left, right, conditionExpr,
variablesSet, joinType, semiJoinDone, systemFieldList);
}
@Override public RelNode accept(RelShuttle shuttle) {
return shuttle.visit(this);
}
public RelWriter explainTerms(RelWriter pw) {
// Don't ever print semiJoinDone=false. This way, we
// don't clutter things up in optimizers that don't use semi-joins.
return super.explainTerms(pw)
.itemIf("semiJoinDone", semiJoinDone, semiJoinDone);
}
@Override public boolean isSemiJoinDone() {
return semiJoinDone;
}
public List getSystemFieldList() {
return systemFieldList;
}
@Override public RelNode withHints(List hintList) {
return new LogicalJoin(getCluster(), traitSet, hintList,
left, right, condition, variablesSet, joinType, semiJoinDone, systemFieldList);
}
}