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

com.hazelcast.org.apache.calcite.adapter.enumerable.EnumerableMatch 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.adapter.enumerable;

import com.hazelcast.org.apache.calcite.adapter.java.JavaTypeFactory;
import com.hazelcast.org.apache.calcite.linq4j.MemoryFactory;
import com.hazelcast.org.apache.calcite.linq4j.Ord;
import com.hazelcast.org.apache.calcite.linq4j.tree.BlockBuilder;
import com.hazelcast.org.apache.calcite.linq4j.tree.BlockStatement;
import com.hazelcast.org.apache.calcite.linq4j.tree.Expression;
import com.hazelcast.org.apache.calcite.linq4j.tree.Expressions;
import com.hazelcast.org.apache.calcite.linq4j.tree.MemberDeclaration;
import com.hazelcast.org.apache.calcite.linq4j.tree.ParameterExpression;
import com.hazelcast.org.apache.calcite.linq4j.tree.Types;
import com.hazelcast.org.apache.calcite.plan.RelOptCluster;
import com.hazelcast.org.apache.calcite.plan.RelTraitSet;
import com.hazelcast.org.apache.calcite.rel.RelCollation;
import com.hazelcast.org.apache.calcite.rel.RelNode;
import com.hazelcast.org.apache.calcite.rel.core.Match;
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.RexBuilder;
import com.hazelcast.org.apache.calcite.rex.RexCall;
import com.hazelcast.org.apache.calcite.rex.RexLiteral;
import com.hazelcast.org.apache.calcite.rex.RexNode;
import com.hazelcast.org.apache.calcite.rex.RexProgramBuilder;
import com.hazelcast.org.apache.calcite.rex.RexSubQuery;
import com.hazelcast.org.apache.calcite.rex.RexVisitorImpl;
import com.hazelcast.org.apache.calcite.runtime.Enumerables;
import com.hazelcast.org.apache.calcite.sql.SqlMatchFunction;
import com.hazelcast.org.apache.calcite.util.BuiltInMethod;
import com.hazelcast.org.apache.calcite.util.ImmutableBitSet;

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

import com.hazelcast.org.checkerframework.checker.nullness.qual.Nullable;

import java.lang.reflect.Modifier;
import java.lang.reflect.Type;
import java.util.ArrayList;
import java.util.List;
import java.util.Map;
import java.util.SortedSet;
import java.util.function.Consumer;
import java.util.function.Function;
import java.util.function.Predicate;

import static com.hazelcast.org.apache.calcite.adapter.enumerable.EnumUtils.NO_EXPRS;

import static java.util.Objects.requireNonNull;

/** Implementation of {@link com.hazelcast.org.apache.calcite.rel.core.Match} in
 * {@link com.hazelcast.org.apache.calcite.adapter.enumerable.EnumerableConvention enumerable calling convention}. */
public class EnumerableMatch extends Match implements EnumerableRel {
  /**
   * Creates an EnumerableMatch.
   *
   * 

Use {@link #create} unless you know what you're doing. */ public EnumerableMatch(RelOptCluster cluster, RelTraitSet traitSet, RelNode input, RelDataType rowType, RexNode pattern, boolean strictStart, boolean strictEnd, Map patternDefinitions, Map measures, RexNode after, Map> subsets, boolean allRows, ImmutableBitSet partitionKeys, RelCollation orderKeys, @Nullable RexNode interval) { super(cluster, traitSet, input, rowType, pattern, strictStart, strictEnd, patternDefinitions, measures, after, subsets, allRows, partitionKeys, orderKeys, interval); } /** Creates an EnumerableMatch. */ public static EnumerableMatch create(RelNode input, RelDataType rowType, RexNode pattern, boolean strictStart, boolean strictEnd, Map patternDefinitions, Map measures, RexNode after, Map> subsets, boolean allRows, ImmutableBitSet partitionKeys, RelCollation orderKeys, @Nullable RexNode interval) { final RelOptCluster cluster = input.getCluster(); final RelTraitSet traitSet = cluster.traitSetOf(EnumerableConvention.INSTANCE); return new EnumerableMatch(cluster, traitSet, input, rowType, pattern, strictStart, strictEnd, patternDefinitions, measures, after, subsets, allRows, partitionKeys, orderKeys, interval); } @Override public RelNode copy(RelTraitSet traitSet, List inputs) { return new EnumerableMatch(getCluster(), traitSet, inputs.get(0), getRowType(), pattern, strictStart, strictEnd, patternDefinitions, measures, after, subsets, allRows, partitionKeys, orderKeys, interval); } @Override public EnumerableRel.Result implement(EnumerableRelImplementor implementor, EnumerableRel.Prefer pref) { final BlockBuilder builder = new BlockBuilder(); final EnumerableRel input = (EnumerableRel) getInput(); final Result result = implementor.visitChild(this, 0, input, pref); final PhysType physType = PhysTypeImpl.of(implementor.getTypeFactory(), input.getRowType(), result.format); final Expression inputExp = builder.append("input", result.block); PhysType inputPhysType = result.physType; final PhysType keyPhysType = inputPhysType.project(partitionKeys.asList(), JavaRowFormat.LIST); final ParameterExpression row_ = Expressions.parameter(inputPhysType.getJavaRowType(), "row_"); final Expression keySelector_ = builder.append("keySelector", inputPhysType.generateSelector(row_, partitionKeys.asList(), keyPhysType.getFormat())); final RelDataTypeFactory.Builder typeBuilder = implementor.getTypeFactory().builder(); measures.forEach((name, value) -> typeBuilder.add(name, value.getType()).nullable(true)); final PhysType emitType = PhysTypeImpl.of(implementor.getTypeFactory(), typeBuilder.build(), result.format); final Expression matcher_ = implementMatcher(implementor, physType, builder, row_); final Expression emitter_ = implementEmitter(implementor, emitType, physType); final MaxHistoryFutureVisitor visitor = new MaxHistoryFutureVisitor(); patternDefinitions.values().forEach(pd -> pd.accept(visitor)); // Fetch // Calculate how many steps we need to look back or forward int history = visitor.getHistory(); int future = visitor.getFuture(); builder.add( Expressions.return_(null, Expressions.call(BuiltInMethod.MATCH.method, inputExp, keySelector_, matcher_, emitter_, Expressions.constant(history), Expressions.constant(future)))); return implementor.result(emitType, builder.toBlock()); } private Expression implementEmitter(EnumerableRelImplementor implementor, PhysType physType, PhysType inputPhysType) { final ParameterExpression rows_ = Expressions.parameter(Types.of(List.class, inputPhysType.getJavaRowType()), "rows"); final ParameterExpression rowStates_ = Expressions.parameter(List.class, "rowStates"); final ParameterExpression symbols_ = Expressions.parameter(List.class, "symbols"); final ParameterExpression match_ = Expressions.parameter(int.class, "match"); final ParameterExpression consumer_ = Expressions.parameter(Consumer.class, "consumer"); final ParameterExpression i_ = Expressions.parameter(int.class, "i"); final ParameterExpression row_ = Expressions.parameter(inputPhysType.getJavaRowType(), "row"); final BlockBuilder builder2 = new BlockBuilder(); // Add loop variable initialization builder2.add( Expressions.declare(0, row_, EnumUtils.convert( Expressions.call(rows_, BuiltInMethod.LIST_GET.method, i_), inputPhysType.getJavaRowType()))); RexBuilder rexBuilder = new RexBuilder(implementor.getTypeFactory()); RexProgramBuilder rexProgramBuilder = new RexProgramBuilder(inputPhysType.getRowType(), rexBuilder); for (Map.Entry entry : measures.entrySet()) { rexProgramBuilder.addProject(entry.getValue(), entry.getKey()); } final RexToLixTranslator translator = RexToLixTranslator.forAggregation( (JavaTypeFactory) getCluster().getTypeFactory(), builder2, new PassedRowsInputGetter(row_, rows_, inputPhysType), implementor.getConformance()); final ParameterExpression result_ = Expressions.parameter(physType.getJavaRowType()); builder2.add( Expressions.declare(Modifier.FINAL, result_, Expressions.new_(physType.getJavaRowType()))); Ord.forEach(measures.values(), (measure, i) -> builder2.add( Expressions.statement( Expressions.assign(physType.fieldReference(result_, i), implementMeasure(translator, rows_, symbols_, i_, row_, measure))))); builder2.add( Expressions.statement( Expressions.call(consumer_, BuiltInMethod.CONSUMER_ACCEPT.method, result_))); final BlockBuilder builder = new BlockBuilder(); // Loop Length // we have to use an explicit for (int i = ...) loop, as we need to know later // which of the matched rows are already passed (in MatchUtils), so foreach cannot be used builder.add( Expressions.for_( Expressions.declare(0, i_, Expressions.constant(0)), Expressions.lessThan(i_, Expressions.call(rows_, BuiltInMethod.COLLECTION_SIZE.method)), Expressions.preIncrementAssign(i_), builder2.toBlock())); return Expressions.new_( Types.of(Enumerables.Emitter.class), NO_EXPRS, Expressions.list( EnumUtils.overridingMethodDecl( BuiltInMethod.EMITTER_EMIT.method, ImmutableList.of(rows_, rowStates_, symbols_, match_, consumer_), builder.toBlock()))); } private static Expression implementMeasure(RexToLixTranslator translator, ParameterExpression rows_, ParameterExpression symbols_, ParameterExpression i_, ParameterExpression row_, RexNode value) { final SqlMatchFunction matchFunction; final MatchImplementor matchImplementor; switch (value.getKind()) { case LAST: case PREV: case CLASSIFIER: matchFunction = (SqlMatchFunction) ((RexCall) value).getOperator(); matchImplementor = RexImpTable.INSTANCE.get(matchFunction); // Work with the implementor return matchImplementor.implement(translator, (RexCall) value, row_, rows_, symbols_, i_); case RUNNING: case FINAL: // See [CALCITE-3341], this should be changed a bit, to implement // FINAL behavior final List operands = ((RexCall) value).getOperands(); assert operands.size() == 1; switch (operands.get(0).getKind()) { case LAST: case PREV: case CLASSIFIER: final RexCall call = (RexCall) operands.get(0); matchFunction = (SqlMatchFunction) call.getOperator(); matchImplementor = RexImpTable.INSTANCE.get(matchFunction); // Work with the implementor requireNonNull((PassedRowsInputGetter) translator.inputGetter, "inputGetter") .setIndex(null); return matchImplementor.implement(translator, call, row_, rows_, symbols_, i_); default: break; } return translator.translate(operands.get(0)); default: return translator.translate(value); } } private Expression implementMatcher(EnumerableRelImplementor implementor, PhysType physType, BlockBuilder builder, ParameterExpression row_) { final Expression patternBuilder_ = builder.append("patternBuilder", Expressions.call(BuiltInMethod.PATTERN_BUILDER.method)); final Expression automaton_ = builder.append("automaton", Expressions.call(implementPattern(patternBuilder_, pattern), BuiltInMethod.PATTERN_TO_AUTOMATON.method)); Expression matcherBuilder_ = builder.append("matcherBuilder", Expressions.call(BuiltInMethod.MATCHER_BUILDER.method, automaton_)); final BlockBuilder builder2 = new BlockBuilder(); // Wrap a MemoryEnumerable around for (Map.Entry entry : patternDefinitions.entrySet()) { // Translate REX to Expressions RexBuilder rexBuilder = new RexBuilder(implementor.getTypeFactory()); RexProgramBuilder rexProgramBuilder = new RexProgramBuilder(physType.getRowType(), rexBuilder); rexProgramBuilder.addCondition(entry.getValue()); final RexToLixTranslator.InputGetter inputGetter1 = new PrevInputGetter(row_, physType); final Expression condition = RexToLixTranslator .translateCondition(rexProgramBuilder.getProgram(), (JavaTypeFactory) getCluster().getTypeFactory(), builder2, inputGetter1, implementor.allCorrelateVariables, implementor.getConformance()); builder2.add(Expressions.return_(null, condition)); final Expression predicate_ = implementPredicate(physType, row_, builder2.toBlock()); matcherBuilder_ = Expressions.call(matcherBuilder_, BuiltInMethod.MATCHER_BUILDER_ADD.method, Expressions.constant(entry.getKey()), predicate_); } return builder.append("matcher", Expressions.call(matcherBuilder_, BuiltInMethod.MATCHER_BUILDER_BUILD.method)); } /** Generates code for a predicate. */ private static Expression implementPredicate(PhysType physType, ParameterExpression rows_, BlockStatement body) { final List memberDeclarations = new ArrayList<>(); ParameterExpression row_ = Expressions.parameter( Types.of(MemoryFactory.Memory.class, physType.getJavaRowType()), "row_"); Expressions.assign(row_, Expressions.call(rows_, BuiltInMethod.MEMORY_GET0.method)); // Implement the Predicate here based on the pattern definition // Add a predicate method: // // public boolean test(E row, List rows) { // return ...; // } memberDeclarations.add( EnumUtils.overridingMethodDecl( BuiltInMethod.PREDICATE_TEST.method, ImmutableList.of(row_), body)); if (EnumerableRules.BRIDGE_METHODS) { // Add a bridge method: // // public boolean test(Object row, Object rows) { // return this.test(row, (List) rows); // } final ParameterExpression row0_ = Expressions.parameter(Object.class, "row"); @SuppressWarnings("unused") final ParameterExpression rowsO_ = Expressions.parameter(Object.class, "rows"); BlockBuilder bridgeBody = new BlockBuilder(); bridgeBody.add( Expressions.return_(null, Expressions.call( Expressions.parameter(Comparable.class, "this"), BuiltInMethod.PREDICATE_TEST.method, Expressions.convert_(row0_, Types.of(MemoryFactory.Memory.class, physType.getJavaRowType()))))); memberDeclarations.add( EnumUtils.overridingMethodDecl( BuiltInMethod.PREDICATE_TEST.method, ImmutableList.of(row0_), bridgeBody.toBlock())); } return Expressions.new_(Types.of(Predicate.class), NO_EXPRS, memberDeclarations); } /** Generates code for a pattern. * *

For example, for the pattern {@code (A B)}, generates * {@code patternBuilder.symbol("A").symbol("B").seq()}. */ private static Expression implementPattern(Expression patternBuilder_, RexNode pattern) { switch (pattern.getKind()) { case LITERAL: final String symbol = ((RexLiteral) pattern).getValueAs(String.class); return Expressions.call(patternBuilder_, BuiltInMethod.PATTERN_BUILDER_SYMBOL.method, Expressions.constant(symbol)); case PATTERN_CONCAT: final RexCall concat = (RexCall) pattern; for (Ord operand : Ord.zip(concat.operands)) { patternBuilder_ = implementPattern(patternBuilder_, operand.e); if (operand.i > 0) { patternBuilder_ = Expressions.call(patternBuilder_, BuiltInMethod.PATTERN_BUILDER_SEQ.method); } } return patternBuilder_; default: throw new AssertionError("unknown kind: " + pattern); } } /** * Visitor that finds out how much "history" we need in the past and future. */ private static class MaxHistoryFutureVisitor extends RexVisitorImpl { private int history; private int future; protected MaxHistoryFutureVisitor() { super(true); } public int getHistory() { return history; } public int getFuture() { return future; } @Override public Void visitCall(RexCall call) { call.operands.forEach(o -> o.accept(this)); final RexLiteral operand; switch (call.op.kind) { case PREV: operand = (RexLiteral) call.getOperands().get(1); final int prev = requireNonNull(operand.getValueAs(Integer.class), () -> "operand in " + call); this.history = Math.max(this.history, prev); break; case NEXT: operand = (RexLiteral) call.getOperands().get(1); final int next = requireNonNull(operand.getValueAs(Integer.class), () -> "operand in " + call); this.future = Math.max(this.future, next); break; default: break; } return null; } @Override public Void visitSubQuery(RexSubQuery subQuery) { return null; } } /** * A special Getter that is able to return a field from a list of objects. */ static class PassedRowsInputGetter implements RexToLixTranslator.InputGetter { private @Nullable Expression index; private final ParameterExpression row; private final ParameterExpression passedRows; private final Function generator; private final PhysType physType; PassedRowsInputGetter(ParameterExpression row, ParameterExpression passedRows, PhysType physType) { this.row = row; this.passedRows = passedRows; generator = e -> new RexToLixTranslator.InputGetterImpl(e, physType); this.physType = physType; } void setIndex(@Nullable Expression index) { this.index = index; } @Override public Expression field(BlockBuilder list, int index, @Nullable Type storageType) { if (this.index == null) { return generator.apply(this.row).field(list, index, storageType); } return Expressions.condition( Expressions.greaterThanOrEqual(this.index, Expressions.constant(0)), generator.apply( EnumUtils.convert( Expressions.call(this.passedRows, BuiltInMethod.LIST_GET.method, this.index), physType.getJavaRowType())) .field(list, index, storageType), Expressions.constant(null)); } } /** * A special Getter that "interchanges" the PREV and the field call. */ static class PrevInputGetter implements RexToLixTranslator.InputGetter { private @Nullable Expression offset; private final ParameterExpression row; private final Function generator; private final PhysType physType; PrevInputGetter(ParameterExpression row, PhysType physType) { this.row = row; generator = e -> new RexToLixTranslator.InputGetterImpl(e, physType); this.physType = physType; } void setOffset(@Nullable Expression offset) { this.offset = offset; } @Override public Expression field(BlockBuilder list, int index, @Nullable Type storageType) { final ParameterExpression row = Expressions.parameter(physType.getJavaRowType()); final ParameterExpression tmp = Expressions.parameter(Object.class); list.add( Expressions.declare(0, tmp, Expressions.call(this.row, BuiltInMethod.MEMORY_GET1.method, requireNonNull(offset, "offset")))); list.add( Expressions.declare(0, row, Expressions.convert_(tmp, physType.getJavaRowType()))); // Add return statement if here is a null! list.add( Expressions.ifThen( Expressions.equal(tmp, Expressions.constant(null)), Expressions.return_(null, Expressions.constant(false)))); return generator.apply(row).field(list, index, storageType); } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy