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

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

/*
 * ModeShape (http://www.modeshape.org)
 * See the COPYRIGHT.txt file distributed with this work for information
 * regarding copyright ownership.  Some portions may be licensed
 * to Red Hat, Inc. under one or more contributor license agreements.
 * See the AUTHORS.txt file in the distribution for a full listing of 
 * individual contributors.
 *
 * ModeShape is free software. Unless otherwise indicated, all code in ModeShape
 * is licensed to you under the terms of the GNU Lesser General Public License as
 * published by the Free Software Foundation; either version 2.1 of
 * the License, or (at your option) any later version.
 * 
 * ModeShape is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
 * Lesser General Public License for more details.
 *
 * You should have received a copy of the GNU Lesser General Public
 * License along with this software; if not, write to the Free
 * Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA
 * 02110-1301 USA, or see the FSF site: http://www.fsf.org.
 */
package org.modeshape.jcr.query.optimize;

import java.util.LinkedList;
import java.util.List;
import java.util.Set;
import org.modeshape.common.annotation.Immutable;
import org.modeshape.jcr.query.QueryContext;
import org.modeshape.jcr.query.model.ChildNodeJoinCondition;
import org.modeshape.jcr.query.model.DescendantNodeJoinCondition;
import org.modeshape.jcr.query.model.DynamicOperand;
import org.modeshape.jcr.query.model.EquiJoinCondition;
import org.modeshape.jcr.query.model.JoinCondition;
import org.modeshape.jcr.query.model.Order;
import org.modeshape.jcr.query.model.Ordering;
import org.modeshape.jcr.query.model.PropertyValue;
import org.modeshape.jcr.query.model.SameNodeJoinCondition;
import org.modeshape.jcr.query.model.SelectorName;
import org.modeshape.jcr.query.plan.JoinAlgorithm;
import org.modeshape.jcr.query.plan.PlanNode;
import org.modeshape.jcr.query.plan.PlanNode.Property;
import org.modeshape.jcr.query.plan.PlanNode.Type;

/**
 * An {@link OptimizerRule optimizer rule} that choose the appropriate join algorithm and sets up any prerequisites, based upon
 * the {@link JoinCondition}.
 * 

* There are two static instances that can be used (or the equivalent can be instantiated or subclassed using the constructor): * one that only uses {@link JoinAlgorithm#NESTED_LOOP nested-loop}, and another that will attempt to use * {@link JoinAlgorithm#MERGE merge} where possible. Both instances ignore any existing {@link Property#JOIN_ALGORITHM} property * value set on the JOIN node. *

*

* For example, the {@link #USE_ONLY_NESTED_JOIN_ALGORITHM} instance will convert this simple tree: * *

 *          ...
 *           |
 *         JOIN
 *        /     \
 *      ...     ...
 * 
* * into this: * *
 *          ...
 *           |
 *         JOIN ({@link Property#JOIN_ALGORITHM JOIN_ALGORITHM}={@link JoinAlgorithm#NESTED_LOOP NESTED_LOOP})
 *        /     \
 *      ...     ...
 * 
* * On the other hand, the {@link #USE_BEST_JOIN_ALGORITHM} instance will do a couple of different things, depending upon the input * plan. *
    *
  1. If the condition is a {@link DescendantNodeJoinCondition}, then the join algorithm will always be * {@link JoinAlgorithm#NESTED_LOOP}.
  2. *
  3. Otherwise, the rule will use the {@link JoinAlgorithm#MERGE} algorithm and will change this structure: * *
     *          ...
     *           |
     *         JOIN
     *        /     \
     *      ...     ...
     * 
    * * into this: * *
     *          ...
     *           |
     *         JOIN ({@link Property#JOIN_ALGORITHM JOIN_ALGORITHM}={@link JoinAlgorithm#MERGE MERGE})
     *        /     \
     *       /       \
     * DUP_REMOVE  DUP_REMOVE
     *     |           |
     *    SORT       SORT
     *     |           |
     *    ...         ...
     * 
    * *
  4. *
*

*/ @Immutable public class ChooseJoinAlgorithm implements OptimizerRule { public static final ChooseJoinAlgorithm USE_ONLY_NESTED_JOIN_ALGORITHM = new ChooseJoinAlgorithm(true); public static final ChooseJoinAlgorithm USE_BEST_JOIN_ALGORITHM = new ChooseJoinAlgorithm(false); private final boolean useOnlyNested; protected ChooseJoinAlgorithm( boolean useOnlyNested ) { this.useOnlyNested = useOnlyNested; } @Override public PlanNode execute( QueryContext context, PlanNode plan, LinkedList ruleStack ) { // For each of the JOIN nodes ... for (PlanNode joinNode : plan.findAllAtOrBelow(Type.JOIN)) { JoinCondition condition = joinNode.getProperty(Property.JOIN_CONDITION, JoinCondition.class); if (useOnlyNested) { joinNode.setProperty(Property.JOIN_ALGORITHM, JoinAlgorithm.NESTED_LOOP); break; } if (condition instanceof DescendantNodeJoinCondition) { // It has to be a nest-loop join ... joinNode.setProperty(Property.JOIN_ALGORITHM, JoinAlgorithm.NESTED_LOOP); } else { joinNode.setProperty(Property.JOIN_ALGORITHM, JoinAlgorithm.MERGE); assert joinNode.getChildCount() == 2; // We can try to use the merge join, but we need to sort and remove remove duplicates ... // on the left and right children of the join ... Set leftSelectors = joinNode.getFirstChild().getSelectors(); Set rightSelectors = joinNode.getLastChild().getSelectors(); List leftSortBy = new LinkedList(); List rightSortBy = new LinkedList(); createOrderBysForJoinCondition(condition, leftSelectors, leftSortBy, rightSelectors, rightSortBy); PlanNode leftSort = new PlanNode(Type.SORT, leftSelectors); leftSort.setProperty(Property.SORT_ORDER_BY, leftSortBy); joinNode.getFirstChild().insertAsParent(leftSort); if (joinNode.getFirstChild().findAllAtOrBelow(Type.DUP_REMOVE).isEmpty()) { // There is no duplicate removal below the left-hand side of the join, so insert it ... PlanNode leftDupRemoval = new PlanNode(Type.DUP_REMOVE, leftSelectors); joinNode.getFirstChild().insertAsParent(leftDupRemoval); } // There is no sort below the right-hand side of the join, so insert it ... PlanNode rightSort = new PlanNode(Type.SORT, rightSelectors); rightSort.setProperty(Property.SORT_ORDER_BY, rightSortBy); joinNode.getLastChild().insertAsParent(rightSort); if (joinNode.getLastChild().findAllAtOrBelow(Type.DUP_REMOVE).isEmpty()) { // There is no duplicate removal below the right-hand side of the join, so insert it ... PlanNode rightDupRemoval = new PlanNode(Type.DUP_REMOVE, rightSelectors); joinNode.getLastChild().insertAsParent(rightDupRemoval); } } } return plan; } protected void createOrderBysForJoinCondition( JoinCondition condition, Set leftSelectors, List leftSortBy, Set rightSelectors, List rightSortBy ) { if (condition instanceof SameNodeJoinCondition) { SameNodeJoinCondition joinCondition = (SameNodeJoinCondition)condition; SelectorName name1 = joinCondition.selector1Name(); SelectorName name2 = joinCondition.selector2Name(); if (leftSelectors.contains(name1)) { leftSortBy.add(name1); rightSortBy.add(name2); } else { leftSortBy.add(name2); rightSortBy.add(name1); } } else if (condition instanceof ChildNodeJoinCondition) { ChildNodeJoinCondition joinCondition = (ChildNodeJoinCondition)condition; SelectorName childName = joinCondition.childSelectorName(); SelectorName parentName = joinCondition.parentSelectorName(); if (leftSelectors.contains(childName)) { leftSortBy.add(childName); rightSortBy.add(parentName); } else { leftSortBy.add(parentName); rightSortBy.add(childName); } } else if (condition instanceof EquiJoinCondition) { EquiJoinCondition joinCondition = (EquiJoinCondition)condition; SelectorName selector1 = joinCondition.selector1Name(); SelectorName selector2 = joinCondition.selector2Name(); String property1 = joinCondition.getProperty1Name(); String property2 = joinCondition.getProperty2Name(); // Create the Ordering for the first selector/property pair ... DynamicOperand operand1 = new PropertyValue(selector1, property1); Ordering ordering1 = new Ordering(operand1, Order.ASCENDING); // Create the Ordering for the second selector/property pair ... DynamicOperand operand2 = new PropertyValue(selector2, property2); Ordering ordering2 = new Ordering(operand2, Order.ASCENDING); if (leftSelectors.contains(selector1)) { leftSortBy.add(ordering1); rightSortBy.add(ordering2); } else { leftSortBy.add(ordering2); rightSortBy.add(ordering1); } } else { assert false; throw new IllegalArgumentException(); } } @Override public String toString() { return getClass().getSimpleName(); } }