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

com.hazelcast.org.apache.calcite.interpreter.Interpreter 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.interpreter;

import com.hazelcast.org.apache.calcite.DataContext;
import com.hazelcast.org.apache.calcite.DataContexts;
import com.hazelcast.org.apache.calcite.adapter.java.JavaTypeFactory;
import com.hazelcast.org.apache.calcite.config.CalciteSystemProperty;
import com.hazelcast.org.apache.calcite.linq4j.AbstractEnumerable;
import com.hazelcast.org.apache.calcite.linq4j.Enumerable;
import com.hazelcast.org.apache.calcite.linq4j.Enumerator;
import com.hazelcast.org.apache.calcite.linq4j.Linq4j;
import com.hazelcast.org.apache.calcite.linq4j.Ord;
import com.hazelcast.org.apache.calcite.linq4j.TransformedEnumerator;
import com.hazelcast.org.apache.calcite.plan.RelOptCluster;
import com.hazelcast.org.apache.calcite.plan.hep.HepPlanner;
import com.hazelcast.org.apache.calcite.plan.hep.HepProgram;
import com.hazelcast.org.apache.calcite.plan.hep.HepProgramBuilder;
import com.hazelcast.org.apache.calcite.rel.RelNode;
import com.hazelcast.org.apache.calcite.rel.RelVisitor;
import com.hazelcast.org.apache.calcite.rel.rules.CoreRules;
import com.hazelcast.org.apache.calcite.rel.type.RelDataType;
import com.hazelcast.org.apache.calcite.rel.type.RelDataTypeFactory;
import com.hazelcast.org.apache.calcite.rex.RexNode;
import com.hazelcast.org.apache.calcite.util.Pair;
import com.hazelcast.org.apache.calcite.util.ReflectUtil;
import com.hazelcast.org.apache.calcite.util.ReflectiveVisitDispatcher;
import com.hazelcast.org.apache.calcite.util.ReflectiveVisitor;
import com.hazelcast.org.apache.calcite.util.Util;

import com.hazelcast.com.google.common.collect.ImmutableList;
import com.hazelcast.com.google.common.collect.ImmutableMap;
import com.hazelcast.com.google.common.collect.Iterables;
import com.hazelcast.com.google.common.collect.LinkedHashMultimap;
import com.hazelcast.com.google.common.collect.Lists;
import com.hazelcast.com.google.common.collect.Multimap;

import com.hazelcast.org.checkerframework.checker.initialization.qual.NotOnlyInitialized;
import com.hazelcast.org.checkerframework.checker.initialization.qual.UnknownInitialization;
import com.hazelcast.org.checkerframework.checker.nullness.qual.Nullable;

import java.util.ArrayDeque;
import java.util.ArrayList;
import java.util.Collection;
import java.util.HashMap;
import java.util.Iterator;
import java.util.LinkedHashMap;
import java.util.List;
import java.util.Map;
import java.util.NoSuchElementException;

import static java.util.Objects.requireNonNull;

/**
 * Interpreter.
 *
 * 

Contains the context for interpreting relational expressions. In * particular it holds working state while the data flow graph is being * assembled. */ public class Interpreter extends AbstractEnumerable<@Nullable Object[]> implements AutoCloseable { private final Map nodes; private final DataContext dataContext; private final RelNode rootRel; /** Creates an Interpreter. */ public Interpreter(DataContext dataContext, RelNode rootRel) { this.dataContext = requireNonNull(dataContext, "dataContext"); final RelNode rel = optimize(rootRel); final CompilerImpl compiler = new Nodes.CoreCompiler(this, rootRel.getCluster()); @SuppressWarnings("method.invocation.invalid") Pair> pair = compiler.visitRoot(rel); this.rootRel = pair.left; this.nodes = ImmutableMap.copyOf(pair.right); } private static RelNode optimize(RelNode rootRel) { final HepProgram hepProgram = new HepProgramBuilder() .addRuleInstance(CoreRules.CALC_SPLIT) .addRuleInstance(CoreRules.FILTER_SCAN) .addRuleInstance(CoreRules.FILTER_INTERPRETER_SCAN) .addRuleInstance(CoreRules.PROJECT_TABLE_SCAN) .addRuleInstance(CoreRules.PROJECT_INTERPRETER_TABLE_SCAN) .addRuleInstance(CoreRules.AGGREGATE_REDUCE_FUNCTIONS) .build(); final HepPlanner planner = new HepPlanner(hepProgram); planner.setRoot(rootRel); rootRel = planner.findBestExp(); return rootRel; } @Override public Enumerator<@Nullable Object[]> enumerator() { start(); final NodeInfo nodeInfo = requireNonNull(nodes.get(rootRel), () -> "nodeInfo for " + rootRel); final Enumerator rows; if (nodeInfo.rowEnumerable != null) { rows = nodeInfo.rowEnumerable.enumerator(); } else { final ArrayDeque queue = Iterables.getOnlyElement(nodeInfo.sinks.values()).list; rows = Linq4j.iterableEnumerator(queue); } return new TransformedEnumerator(rows) { @Override protected @Nullable Object[] transform(Row row) { return row.getValues(); } }; } @SuppressWarnings("CatchAndPrintStackTrace") private void start() { // We rely on the nodes being ordered leaves first. for (Map.Entry entry : nodes.entrySet()) { final NodeInfo nodeInfo = entry.getValue(); try { assert nodeInfo.node != null : "node must not be null for nodeInfo, rel=" + nodeInfo.rel; nodeInfo.node.run(); } catch (InterruptedException e) { e.printStackTrace(); } } } @Override public void close() { nodes.values().forEach(NodeInfo::close); } /** Information about a node registered in the data flow graph. */ private static class NodeInfo { final RelNode rel; final Map sinks = new LinkedHashMap<>(); final @Nullable Enumerable rowEnumerable; @Nullable Node node; NodeInfo(RelNode rel, @Nullable Enumerable rowEnumerable) { this.rel = rel; this.rowEnumerable = rowEnumerable; } void close() { if (node != null) { final Node n = node; node = null; n.close(); } } } /** * A {@link Source} that is just backed by an {@link Enumerator}. The {@link Enumerator} is closed * when it is finished or by calling {@link #close()}. */ private static class EnumeratorSource implements Source { private final Enumerator enumerator; EnumeratorSource(final Enumerator enumerator) { this.enumerator = requireNonNull(enumerator, "enumerator"); } @Override public @Nullable Row receive() { if (enumerator.moveNext()) { return enumerator.current(); } // close the enumerator once we have gone through everything enumerator.close(); return null; } @Override public void close() { enumerator.close(); } } /** Implementation of {@link Sink} using a {@link java.util.ArrayDeque}. */ private static class ListSink implements Sink { final ArrayDeque list; private ListSink(ArrayDeque list) { this.list = list; } @Override public void send(Row row) throws InterruptedException { list.add(row); } @Override public void end() throws InterruptedException { } @SuppressWarnings("deprecation") @Override public void setSourceEnumerable(Enumerable enumerable) throws InterruptedException { // just copy over the source into the local list final Enumerator enumerator = enumerable.enumerator(); while (enumerator.moveNext()) { this.send(enumerator.current()); } enumerator.close(); } } /** Implementation of {@link Source} using a {@link java.util.ArrayDeque}. */ private static class ListSource implements Source { private final ArrayDeque list; private @Nullable Iterator iterator; ListSource(ArrayDeque list) { this.list = list; } @Override public @Nullable Row receive() { try { if (iterator == null) { iterator = list.iterator(); } return iterator.next(); } catch (NoSuchElementException e) { iterator = null; return null; } } @Override public void close() { // noop } } /** Implementation of {@link Sink} using a {@link java.util.ArrayDeque}. */ private static class DuplicatingSink implements Sink { private List> queues; private DuplicatingSink(List> queues) { this.queues = ImmutableList.copyOf(queues); } @Override public void send(Row row) throws InterruptedException { for (ArrayDeque queue : queues) { queue.add(row); } } @Override public void end() throws InterruptedException { } @SuppressWarnings("deprecation") @Override public void setSourceEnumerable(Enumerable enumerable) throws InterruptedException { // just copy over the source into the local list final Enumerator enumerator = enumerable.enumerator(); while (enumerator.moveNext()) { this.send(enumerator.current()); } enumerator.close(); } } /** * Walks over a tree of {@link com.hazelcast.org.apache.calcite.rel.RelNode} and, for each, * creates a {@link com.hazelcast.org.apache.calcite.interpreter.Node} that can be * executed in the interpreter. * *

The compiler looks for methods of the form "visit(XxxRel)". * A "visit" method must create an appropriate {@link Node} and put it into * the {@link #node} field. * *

If you wish to handle more kinds of relational expressions, add extra * "visit" methods in this or a sub-class, and they will be found and called * via reflection. */ static class CompilerImpl extends RelVisitor implements Compiler, ReflectiveVisitor { final ScalarCompiler scalarCompiler; private final ReflectiveVisitDispatcher dispatcher = ReflectUtil.createDispatcher(CompilerImpl.class, RelNode.class); @NotOnlyInitialized protected final Interpreter interpreter; protected @Nullable RelNode rootRel; protected @Nullable RelNode rel; protected @Nullable Node node; final Map nodes = new LinkedHashMap<>(); final Map> relInputs = new HashMap<>(); final Multimap outEdges = LinkedHashMultimap.create(); private static final String REWRITE_METHOD_NAME = "rewrite"; private static final String VISIT_METHOD_NAME = "visit"; CompilerImpl(@UnknownInitialization Interpreter interpreter, RelOptCluster cluster) { this.interpreter = interpreter; this.scalarCompiler = new JaninoRexCompiler(cluster.getRexBuilder()); } /** Visits the tree, starting from the root {@code p}. */ Pair> visitRoot(RelNode p) { rootRel = p; visit(p, 0, null); return Pair.of(requireNonNull(rootRel, "rootRel"), nodes); } @Override public void visit(RelNode p, int ordinal, @Nullable RelNode parent) { for (;;) { rel = null; boolean found = dispatcher.invokeVisitor(this, p, REWRITE_METHOD_NAME); if (!found) { throw new AssertionError( "interpreter: no implementation for rewrite"); } if (rel == null) { break; } if (CalciteSystemProperty.DEBUG.value()) { System.out.println("Interpreter: rewrite " + p + " to " + rel); } p = requireNonNull(rel, "rel"); if (parent != null) { List inputs = relInputs.get(parent); if (inputs == null) { inputs = Lists.newArrayList(parent.getInputs()); relInputs.put(parent, inputs); } inputs.set(ordinal, p); } else { rootRel = p; } } // rewrite children first (from left to right) final List inputs = relInputs.get(p); RelNode finalP = p; Ord.forEach(Util.first(inputs, p.getInputs()), (r, i) -> outEdges.put(r, new Edge(finalP, i))); if (inputs != null) { for (int i = 0; i < inputs.size(); i++) { RelNode input = inputs.get(i); visit(input, i, p); } } else { p.childrenAccept(this); } node = null; boolean found = dispatcher.invokeVisitor(this, p, VISIT_METHOD_NAME); if (!found) { if (p instanceof InterpretableRel) { InterpretableRel interpretableRel = (InterpretableRel) p; node = interpretableRel.implement( new InterpretableRel.InterpreterImplementor(this, null, DataContexts.EMPTY)); } else { // Probably need to add a visit(XxxRel) method to CoreCompiler. throw new AssertionError("interpreter: no implementation for " + p.getClass()); } } final NodeInfo nodeInfo = nodes.get(p); assert nodeInfo != null; nodeInfo.node = node; if (inputs != null) { for (int i = 0; i < inputs.size(); i++) { final RelNode input = inputs.get(i); visit(input, i, p); } } } /** Fallback rewrite method. * *

Overriding methods (each with a different sub-class of {@link RelNode} * as its argument type) sets the {@link #rel} field if intends to * rewrite. */ public void rewrite(RelNode r) { } @Override public Scalar compile(List nodes, @Nullable RelDataType inputRowType) { if (inputRowType == null) { inputRowType = getTypeFactory().builder() .build(); } return scalarCompiler.compile(nodes, inputRowType) .apply(interpreter.dataContext); } private JavaTypeFactory getTypeFactory() { return interpreter.dataContext.getTypeFactory(); } @Override public RelDataType combinedRowType(List inputs) { final RelDataTypeFactory.Builder builder = getTypeFactory().builder(); for (RelNode input : inputs) { builder.addAll(input.getRowType().getFieldList()); } return builder.build(); } @Override public Source source(RelNode rel, int ordinal) { final RelNode input = getInput(rel, ordinal); final Edge edge = new Edge(rel, ordinal); final Collection edges = outEdges.get(input); final NodeInfo nodeInfo = nodes.get(input); if (nodeInfo == null) { throw new AssertionError("should be registered: " + rel); } if (nodeInfo.rowEnumerable != null) { return new EnumeratorSource(nodeInfo.rowEnumerable.enumerator()); } assert nodeInfo.sinks.size() == edges.size(); final ListSink sink = nodeInfo.sinks.get(edge); if (sink != null) { return new ListSource(sink.list); } throw new IllegalStateException( "Got a sink " + sink + " to which there is no match source type!"); } private RelNode getInput(RelNode rel, int ordinal) { final List inputs = relInputs.get(rel); if (inputs != null) { return inputs.get(ordinal); } return rel.getInput(ordinal); } @Override public Sink sink(RelNode rel) { final Collection edges = outEdges.get(rel); final Collection edges2 = edges.isEmpty() ? ImmutableList.of(new Edge(null, 0)) : edges; NodeInfo nodeInfo = nodes.get(rel); if (nodeInfo == null) { nodeInfo = new NodeInfo(rel, null); nodes.put(rel, nodeInfo); for (Edge edge : edges2) { nodeInfo.sinks.put(edge, new ListSink(new ArrayDeque<>())); } } else { for (Edge edge : edges2) { if (nodeInfo.sinks.containsKey(edge)) { continue; } nodeInfo.sinks.put(edge, new ListSink(new ArrayDeque<>())); } } if (edges.size() == 1) { return Iterables.getOnlyElement(nodeInfo.sinks.values()); } else { final List> queues = new ArrayList<>(); for (ListSink sink : nodeInfo.sinks.values()) { queues.add(sink.list); } return new DuplicatingSink(queues); } } @Override public void enumerable(RelNode rel, Enumerable rowEnumerable) { NodeInfo nodeInfo = new NodeInfo(rel, rowEnumerable); nodes.put(rel, nodeInfo); } @Override public Context createContext() { return new Context(getDataContext()); } @Override public DataContext getDataContext() { return interpreter.dataContext; } } /** Edge between a {@link RelNode} and one of its inputs. */ static class Edge extends Pair<@Nullable RelNode, Integer> { Edge(@Nullable RelNode parent, int ordinal) { super(parent, ordinal); } } /** Converts a list of expressions to a scalar that can compute their * values. */ interface ScalarCompiler { Scalar.Producer compile(List nodes, RelDataType inputRowType); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy