All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.hazelcast.org.apache.calcite.rel.rules.SemiJoinJoinTransposeRule Maven / Gradle / Ivy

There is a newer version: 5.5.0
Show newest version
/*
 * 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.plan.RelOptRule;
import com.hazelcast.org.apache.calcite.plan.RelOptRuleCall;
import com.hazelcast.org.apache.calcite.plan.RelOptUtil;
import com.hazelcast.org.apache.calcite.rel.RelNode;
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.core.RelFactories;
import com.hazelcast.org.apache.calcite.rel.logical.LogicalJoin;
import com.hazelcast.org.apache.calcite.rel.type.RelDataTypeField;
import com.hazelcast.org.apache.calcite.rex.RexNode;
import com.hazelcast.org.apache.calcite.tools.RelBuilderFactory;
import com.hazelcast.org.apache.calcite.util.ImmutableIntList;

import com.hazelcast.com.google.common.collect.ImmutableList;
import com.hazelcast.com.google.common.collect.ImmutableSet;

import java.util.ArrayList;
import java.util.List;

/**
 * Planner rule that pushes a {@code SemiJoin}
 * down in a tree past a {@link com.hazelcast.org.apache.calcite.rel.core.Join}
 * in order to trigger other rules that will convert {@code SemiJoin}s.
 *
 * 
    *
  • SemiJoin(LogicalJoin(X, Y), Z) → LogicalJoin(SemiJoin(X, Z), Y) *
  • SemiJoin(LogicalJoin(X, Y), Z) → LogicalJoin(X, SemiJoin(Y, Z)) *
* *

Whether this * first or second conversion is applied depends on which operands actually * participate in the semi-join.

*/ public class SemiJoinJoinTransposeRule extends RelOptRule implements TransformationRule { public static final SemiJoinJoinTransposeRule INSTANCE = new SemiJoinJoinTransposeRule(RelFactories.LOGICAL_BUILDER); //~ Constructors ----------------------------------------------------------- /** * Creates a SemiJoinJoinTransposeRule. */ public SemiJoinJoinTransposeRule(RelBuilderFactory relBuilderFactory) { super( operandJ(LogicalJoin.class, null, Join::isSemiJoin, some(operand(Join.class, any()))), relBuilderFactory, null); } //~ Methods ---------------------------------------------------------------- // implement RelOptRule public void onMatch(RelOptRuleCall call) { LogicalJoin semiJoin = call.rel(0); final Join join = call.rel(1); if (join.isSemiJoin()) { return; } final ImmutableIntList leftKeys = semiJoin.analyzeCondition().leftKeys; // X is the left child of the join below the semi-join // Y is the right child of the join below the semi-join // Z is the right child of the semi-join int nFieldsX = join.getLeft().getRowType().getFieldList().size(); int nFieldsY = join.getRight().getRowType().getFieldList().size(); int nFieldsZ = semiJoin.getRight().getRowType().getFieldList().size(); int nTotalFields = nFieldsX + nFieldsY + nFieldsZ; List fields = new ArrayList<>(); // create a list of fields for the full join result; note that // we can't simply use the fields from the semi-join because the // row-type of a semi-join only includes the left hand side fields List joinFields = semiJoin.getRowType().getFieldList(); for (int i = 0; i < (nFieldsX + nFieldsY); i++) { fields.add(joinFields.get(i)); } joinFields = semiJoin.getRight().getRowType().getFieldList(); for (int i = 0; i < nFieldsZ; i++) { fields.add(joinFields.get(i)); } // determine which operands below the semi-join are the actual // Rels that participate in the semi-join int nKeysFromX = 0; for (int leftKey : leftKeys) { if (leftKey < nFieldsX) { nKeysFromX++; } } // the keys must all originate from either the left or right; // otherwise, a semi-join wouldn't have been created assert (nKeysFromX == 0) || (nKeysFromX == leftKeys.size()); // need to convert the semi-join condition and possibly the keys RexNode newSemiJoinFilter; int[] adjustments = new int[nTotalFields]; if (nKeysFromX > 0) { // (X, Y, Z) --> (X, Z, Y) // semiJoin(X, Z) // pass 0 as Y's adjustment because there shouldn't be any // references to Y in the semi-join filter setJoinAdjustments( adjustments, nFieldsX, nFieldsY, nFieldsZ, 0, -nFieldsY); newSemiJoinFilter = semiJoin.getCondition().accept( new RelOptUtil.RexInputConverter( semiJoin.getCluster().getRexBuilder(), fields, adjustments)); } else { // (X, Y, Z) --> (X, Y, Z) // semiJoin(Y, Z) setJoinAdjustments( adjustments, nFieldsX, nFieldsY, nFieldsZ, -nFieldsX, -nFieldsX); newSemiJoinFilter = semiJoin.getCondition().accept( new RelOptUtil.RexInputConverter( semiJoin.getCluster().getRexBuilder(), fields, adjustments)); } // create the new join RelNode leftSemiJoinOp; if (nKeysFromX > 0) { leftSemiJoinOp = join.getLeft(); } else { leftSemiJoinOp = join.getRight(); } LogicalJoin newSemiJoin = LogicalJoin.create(leftSemiJoinOp, semiJoin.getRight(), // No need to copy the hints, the framework would try to do that. ImmutableList.of(), newSemiJoinFilter, ImmutableSet.of(), JoinRelType.SEMI); RelNode leftJoinRel; RelNode rightJoinRel; if (nKeysFromX > 0) { leftJoinRel = newSemiJoin; rightJoinRel = join.getRight(); } else { leftJoinRel = join.getLeft(); rightJoinRel = newSemiJoin; } RelNode newJoinRel = join.copy( join.getTraitSet(), join.getCondition(), leftJoinRel, rightJoinRel, join.getJoinType(), join.isSemiJoinDone()); call.transformTo(newJoinRel); } /** * Sets an array to reflect how much each index corresponding to a field * needs to be adjusted. The array corresponds to fields in a 3-way join * between (X, Y, and Z). X remains unchanged, but Y and Z need to be * adjusted by some fixed amount as determined by the input. * * @param adjustments array to be filled out * @param nFieldsX number of fields in X * @param nFieldsY number of fields in Y * @param nFieldsZ number of fields in Z * @param adjustY the amount to adjust Y by * @param adjustZ the amount to adjust Z by */ private void setJoinAdjustments( int[] adjustments, int nFieldsX, int nFieldsY, int nFieldsZ, int adjustY, int adjustZ) { for (int i = 0; i < nFieldsX; i++) { adjustments[i] = 0; } for (int i = nFieldsX; i < (nFieldsX + nFieldsY); i++) { adjustments[i] = adjustY; } for (int i = nFieldsX + nFieldsY; i < (nFieldsX + nFieldsY + nFieldsZ); i++) { adjustments[i] = adjustZ; } } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy