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

org.modeshape.jcr.query.optimize.RewriteIdentityJoins Maven / Gradle / Ivy

There is a newer version: 5.4.1.Final
Show newest version
/*
 * ModeShape (http://www.modeshape.org)
 *
 * Licensed 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 org.modeshape.jcr.query.optimize;

import java.util.HashMap;
import java.util.LinkedList;
import java.util.List;
import java.util.Map;
import org.modeshape.common.annotation.Immutable;
import org.modeshape.jcr.GraphI18n;
import org.modeshape.jcr.query.QueryContext;
import org.modeshape.jcr.query.model.Column;
import org.modeshape.jcr.query.model.EquiJoinCondition;
import org.modeshape.jcr.query.model.JoinCondition;
import org.modeshape.jcr.query.model.SameNodeJoinCondition;
import org.modeshape.jcr.query.model.SelectorName;
import org.modeshape.jcr.query.plan.PlanNode;
import org.modeshape.jcr.query.plan.PlanNode.Property;
import org.modeshape.jcr.query.plan.PlanNode.Type;
import org.modeshape.jcr.query.plan.PlanUtil;
import org.modeshape.jcr.query.validate.Schemata;
import org.modeshape.jcr.query.validate.Schemata.Table;

/**
 * An {@link OptimizerRule optimizer rule} that rewrites JOIN nodes that have {@link EquiJoinCondition equi-join criteria} where
 * the columns involved in the equi-join are all identity columns (that is, they form a
 * {@link org.modeshape.jcr.query.validate.Schemata.Table#getKeys() key} for the table). This rewrite only happens when the left
 * and right children of the JOIN node are both SOURCE nodes.
 * 

* The basic idea is that in these identity equi-join cases, the following structure: * *

 *          ...
 *           |
 *         JOIN
 *        /     \
 *       /       \
 *   SOURCE     SOURCE
 * 
* * is transformed into a simple SOURCE node: * *
 *        ...
 *         |
 *       SOURCE
 * 
* * Note that this rewriting removes a selector, and thus the nodes above the JOIN node that made use of the removed selector are * also modified to reference the remaining selector. *

*/ @Immutable public class RewriteIdentityJoins implements OptimizerRule { public static final RewriteIdentityJoins INSTANCE = new RewriteIdentityJoins(); @Override public PlanNode execute( QueryContext context, PlanNode plan, LinkedList ruleStack ) { if (!context.getHints().hasJoin) return plan; // For each of the JOIN nodes ... Map rewrittenSelectors = null; int rewrittenJoins = 0; int numJoins = 0; for (PlanNode joinNode : plan.findAllAtOrBelow(Type.JOIN)) { ++numJoins; JoinCondition condition = joinNode.getProperty(Property.JOIN_CONDITION, JoinCondition.class); if (condition instanceof EquiJoinCondition) { PlanNode leftNode = joinNode.getFirstChild().findAtOrBelow(Type.SOURCE); PlanNode rightNode = joinNode.getLastChild().findAtOrBelow(Type.SOURCE); assert leftNode != null; assert rightNode != null; EquiJoinCondition equiJoin = (EquiJoinCondition)condition; // Find the names (or aliases) of the tables ... Schemata schemata = context.getSchemata(); assert schemata != null; SelectorName leftTableName = leftNode.getProperty(Property.SOURCE_NAME, SelectorName.class); SelectorName rightTableName = rightNode.getProperty(Property.SOURCE_NAME, SelectorName.class); assert leftTableName != null; assert rightTableName != null; // Presumably the join condition is using at least one alias, but we only care about the actual name ... if (!leftTableName.equals(rightTableName)) { // The join is not joining the same table, so this doesn't meet the condition ... continue; } // Find the schemata columns referenced by the join condition ... Table table = schemata.getTable(leftTableName); if (table == null) { context.getProblems().addError(GraphI18n.tableDoesNotExist, leftTableName); continue; } String leftColumnName = equiJoin.getProperty1Name(); String rightColumnName = equiJoin.getProperty2Name(); Schemata.Column leftColumn = table.getColumn(leftColumnName); Schemata.Column rightColumn = table.getColumn(rightColumnName); if (leftColumn != null && rightColumn != null) { // Are the join columns (on both sides) keys? if (table.hasKey(leftColumn) && (rightColumn == leftColumn || table.hasKey(rightColumn))) { // It meets all the criteria, so rewrite this join node ... if (rewrittenSelectors == null) rewrittenSelectors = new HashMap(); rewriteJoinNode(context, joinNode, rewrittenSelectors); ++rewrittenJoins; } } else { // one or both of the columns must be residual properties ... // if (leftColumn == null) { // context.getProblems().addError(GraphI18n.columnDoesNotExistOnTable, leftColumnName, leftTableName); // continue; // } // if (rightColumn == null) { // context.getProblems().addError(GraphI18n.columnDoesNotExistOnTable, rightColumnName, leftTableName); // continue; // } } } else if (condition instanceof SameNodeJoinCondition) { SameNodeJoinCondition sameNodeCondition = (SameNodeJoinCondition)condition; if (sameNodeCondition.getSelector2Path() == null) { // It meets all the criteria, so rewrite this join node ... if (rewrittenSelectors == null) rewrittenSelectors = new HashMap(); rewriteJoinNode(context, joinNode, rewrittenSelectors); ++rewrittenJoins; } } } if (rewrittenSelectors != null && !rewrittenSelectors.isEmpty()) { // We re-wrote at least one JOIN, but since this only applies to JOIN nodes that meet certain criteria, // the rewriting may have changed JOIN nodes that did not meet this criteria into nodes that now meet // this criteria, so we need to re-run this rule... ruleStack.addFirst(this); // After this rule is done as is no longer needed, we need to try to push SELECTs and PROJECTs again ... if (!(ruleStack.peek() instanceof PushSelectCriteria)) { // We haven't already added these, so add them now ... ruleStack.addFirst(PushProjects.INSTANCE); if (context.getHints().hasCriteria) { ruleStack.addFirst(PushSelectCriteria.INSTANCE); } } // Now rewrite the various portions of the plan that make use of the now-removed selectors ... PlanUtil.replaceReferencesToRemovedSource(context, plan, rewrittenSelectors); assert rewrittenJoins > 0; if (rewrittenJoins == numJoins) { assert plan.findAllAtOrBelow(Type.JOIN).isEmpty(); context.getHints().hasJoin = false; } } return plan; } protected void rewriteJoinNode( QueryContext context, PlanNode joinNode, Map rewrittenSelectors ) { // Remove the right source node from the join node ... PlanNode rightChild = joinNode.getLastChild(); rightChild.removeFromParent(); PlanNode rightSource = rightChild.findAtOrBelow(Type.SOURCE); // Replace the join node with the left source node ... PlanNode leftChild = joinNode.getFirstChild(); joinNode.extractFromParent(); PlanNode leftSource = leftChild.findAtOrBelow(Type.SOURCE); // Combine the right PROJECT node with that on the left ... PlanNode rightProject = rightChild.findAtOrBelow(Type.PROJECT); if (rightProject != null) { PlanNode leftProject = leftChild.findAtOrBelow(Type.PROJECT); if (leftProject != null) { List leftColumns = leftProject.getPropertyAsList(Property.PROJECT_COLUMNS, Column.class); for (Column rightColumn : rightProject.getPropertyAsList(Property.PROJECT_COLUMNS, Column.class)) { if (!leftColumns.contains(rightColumn)) leftColumns.add(rightColumn); } } else { // Just create a project on the left side ... leftProject = new PlanNode(Type.PROJECT); leftProject.setProperty(Property.PROJECT_COLUMNS, rightProject.getProperty(Property.PROJECT_COLUMNS)); leftChild.getFirstChild().insertAsParent(leftProject); } } // Accumulate any SELECT nodes from the right side and add to the left ... PlanNode topRightSelect = rightChild.findAtOrBelow(Type.SELECT); if (topRightSelect != null) { PlanNode bottomRightSelect = topRightSelect; while (true) { if (bottomRightSelect.getFirstChild().isNot(Type.SELECT)) break; bottomRightSelect = bottomRightSelect.getFirstChild(); } topRightSelect.setParent(null); bottomRightSelect.removeAllChildren(); // Place just above the left source ... leftSource.getParent().addLastChild(topRightSelect); leftSource.setParent(bottomRightSelect); } // Now record that references to the right selector name should be removed ... SelectorName rightTableName = rightSource.getProperty(Property.SOURCE_NAME, SelectorName.class); SelectorName rightTableAlias = rightSource.getProperty(Property.SOURCE_ALIAS, SelectorName.class); SelectorName leftTableAlias = leftSource.getProperty(Property.SOURCE_ALIAS, SelectorName.class); if (leftTableAlias != null) { if (rightTableName != null) rewrittenSelectors.put(rightTableName, leftTableAlias); if (rightTableAlias != null) rewrittenSelectors.put(rightTableAlias, leftTableAlias); } else { SelectorName leftTableName = leftSource.getProperty(Property.SOURCE_NAME, SelectorName.class); assert leftTableName != null; if (rightTableName != null) rewrittenSelectors.put(rightTableName, leftTableName); if (rightTableAlias != null) rewrittenSelectors.put(rightTableAlias, leftTableName); } } @Override public String toString() { return getClass().getSimpleName(); } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy