Vladsz83 commented on code in PR #11935:
URL: https://github.com/apache/ignite/pull/11935#discussion_r2007301991


##########
modules/calcite/src/main/java/org/apache/ignite/internal/processors/query/calcite/prepare/PlannerHelper.java:
##########
@@ -129,6 +152,179 @@ public static IgniteRel optimize(SqlNode sqlNode, 
IgnitePlanner planner, IgniteL
         }
     }
 
+    /** */
+    private static RelNode actualTopLevelJoinTypeHints(RelNode rel, 
List<RelHint> topLevelHints) {
+        assert rel instanceof Hintable;
+
+        List<RelHint> curHints = ((Hintable)rel).getHints();
+
+        List<RelHint> res = new ArrayList<>(topLevelHints.size());
+
+        for (RelHint topHint : topLevelHints) {
+            assert topHint.inheritPath.isEmpty();
+
+            if (!JOIN_TYPE_HINT_NAMES.contains(topHint.hintName))
+                continue;
+
+            if (curHints.isEmpty()) {
+                res.add(topHint);
+
+                continue;
+            }
+
+            for (RelHint curHint : curHints) {
+                // Consider only hints of the join types and which differ by 
name or parameters.
+                if (!JOIN_TYPE_HINT_NAMES.contains(curHint.hintName)
+                    || topHint.equals(curHint.inheritPath.isEmpty() ? curHint 
: curHint.copy(Collections.emptyList())))
+                    continue;
+
+                res.add(topHint);
+            }
+        }
+
+        if (!res.isEmpty()) {
+            rel = ((Hintable)rel).withHints(res);
+
+            if (!curHints.isEmpty())
+                rel = ((Hintable)rel).attachHints(curHints);
+        }
+
+        return rel;
+    }
+
+    /**
+     * Tries to optimize joins order.
+     *
+     * @see JoinToMultiJoinRule
+     * @see IgniteMultiJoinOptimizeRule
+     *
+     * @return An node with optimized joins or original {@code root} if didn't 
optimize.
+     */
+    private static RelNode optimizeJoinsOrder(IgnitePlanner planner, RelNode 
root, List<RelHint> topLevelHints) {
+        List<Join> joins = findNodes(root, Join.class, false);
+
+        // No original joins found, nothing to optimize.
+        if (joins.isEmpty())
+            return root;
+
+        int disabledCnt = 0;
+
+        // If all the joins have the forced order, no need to optimize the 
joins order at all.
+        for (Join join : joins) {
+            for (RelHint hint : join.getHints()) {
+                if 
(HintDefinition.ENFORCE_JOIN_ORDER.name().equals(hint.hintName)) {
+                    ++disabledCnt;
+
+                    break;
+                }
+            }
+        }
+
+        if (disabledCnt == joins.size())
+            return root;
+
+        RelNode res = planner.transform(PlannerPhase.HEP_OPTIMIZE_JOIN_ORDER, 
root.getTraitSet(), root);
+
+        // Still has a MultiJoin, didn't manage to collect one flat join to 
optimize.
+        if (!findNodes(res, MultiJoin.class, true).isEmpty())
+            return root;
+
+        // If a new joins order was proposed, no need to launch another join 
order optimizations.
+        
planner.setDisabledRules(HintDefinition.ENFORCE_JOIN_ORDER.disabledRules().stream().map(RelOptRule::toString)
+            .collect(Collectors.toList()));
+
+        if (!topLevelHints.isEmpty()) {
+            res = actualTopLevelJoinTypeHints(res, topLevelHints);
+
+            restoreJoinTypeHints(res);
+        }
+
+        return res;
+    }
+
+    /**
+     * A join type hint might be assigned to a query root (top-level hint) or 
to a table. Originally, SELECT-level hints
+     * are propagated and assigned to following Joins and TableScans. We lose 
assigned to Join nodes ones
+     * in {@link JoinToMultiJoinRule} and have to reassign them from top-level 
hints.
+     */
+    private static void restoreJoinTypeHints(RelNode root) {
+        RelShuttle visitor = new RelHomogeneousShuttle() {
+            /** Hints to assign on current tree level. */
+            private final List<List<RelHint>> hintsStack = new ArrayList<>();
+
+            /** Current hint inheritance path. It is important for hint 
priority. */
+            private final List<Integer> inputsStack = new ArrayList<>();
+
+            /** {@inheritDoc} */
+            @Override public RelNode visit(RelNode rel) {
+                // Leaf TableScans have no inputs. And we are interrested only 
in Joins.
+                if (rel.getInputs().isEmpty())
+                    return rel;
+
+                List<RelHint> curHints = Collections.emptyList();
+
+                if ((rel instanceof Hintable) && !(rel instanceof Join) && 
!((Hintable)rel).getHints().isEmpty()) {

Review Comment:
   We get joins without hints and repeat hints pushing to join nodes from the 
top nodes. If join has a hint, it was already inherited from above. No need to 
duplicate. Thus, we assign hints to joins, but do not consider already 
assigned. Current join type hint will be pushed further.



-- 
This is an automated message from the Apache Git Service.
To respond to the message, please log on to GitHub and use the
URL above to go to the specific comment.

To unsubscribe, e-mail: notifications-unsubscr...@ignite.apache.org

For queries about this service, please contact Infrastructure at:
us...@infra.apache.org

Reply via email to