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

com.hazelcast.org.apache.calcite.tools.RelBuilder 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.tools;

import com.hazelcast.org.apache.calcite.linq4j.Ord;
import com.hazelcast.org.apache.calcite.linq4j.function.Experimental;
import com.hazelcast.org.apache.calcite.plan.Context;
import com.hazelcast.org.apache.calcite.plan.Contexts;
import com.hazelcast.org.apache.calcite.plan.Convention;
import com.hazelcast.org.apache.calcite.plan.RelOptCluster;
import com.hazelcast.org.apache.calcite.plan.RelOptPredicateList;
import com.hazelcast.org.apache.calcite.plan.RelOptSchema;
import com.hazelcast.org.apache.calcite.plan.RelOptTable;
import com.hazelcast.org.apache.calcite.plan.RelOptUtil;
import com.hazelcast.org.apache.calcite.plan.ViewExpanders;
import com.hazelcast.org.apache.calcite.prepare.RelOptTableImpl;
import com.hazelcast.org.apache.calcite.rel.RelCollation;
import com.hazelcast.org.apache.calcite.rel.RelCollations;
import com.hazelcast.org.apache.calcite.rel.RelDistribution;
import com.hazelcast.org.apache.calcite.rel.RelFieldCollation;
import com.hazelcast.org.apache.calcite.rel.RelHomogeneousShuttle;
import com.hazelcast.org.apache.calcite.rel.RelNode;
import com.hazelcast.org.apache.calcite.rel.core.Aggregate;
import com.hazelcast.org.apache.calcite.rel.core.AggregateCall;
import com.hazelcast.org.apache.calcite.rel.core.Correlate;
import com.hazelcast.org.apache.calcite.rel.core.CorrelationId;
import com.hazelcast.org.apache.calcite.rel.core.Filter;
import com.hazelcast.org.apache.calcite.rel.core.Intersect;
import com.hazelcast.org.apache.calcite.rel.core.Join;
import com.hazelcast.org.apache.calcite.rel.core.JoinRelType;
import com.hazelcast.org.apache.calcite.rel.core.Match;
import com.hazelcast.org.apache.calcite.rel.core.Minus;
import com.hazelcast.org.apache.calcite.rel.core.Project;
import com.hazelcast.org.apache.calcite.rel.core.RelFactories;
import com.hazelcast.org.apache.calcite.rel.core.RepeatUnion;
import com.hazelcast.org.apache.calcite.rel.core.Snapshot;
import com.hazelcast.org.apache.calcite.rel.core.Sort;
import com.hazelcast.org.apache.calcite.rel.core.Spool;
import com.hazelcast.org.apache.calcite.rel.core.TableFunctionScan;
import com.hazelcast.org.apache.calcite.rel.core.TableScan;
import com.hazelcast.org.apache.calcite.rel.core.TableSpool;
import com.hazelcast.org.apache.calcite.rel.core.Uncollect;
import com.hazelcast.org.apache.calcite.rel.core.Union;
import com.hazelcast.org.apache.calcite.rel.core.Values;
import com.hazelcast.org.apache.calcite.rel.hint.Hintable;
import com.hazelcast.org.apache.calcite.rel.hint.RelHint;
import com.hazelcast.org.apache.calcite.rel.logical.LogicalFilter;
import com.hazelcast.org.apache.calcite.rel.logical.LogicalProject;
import com.hazelcast.org.apache.calcite.rel.metadata.RelColumnMapping;
import com.hazelcast.org.apache.calcite.rel.metadata.RelMetadataQuery;
import com.hazelcast.org.apache.calcite.rel.type.RelDataType;
import com.hazelcast.org.apache.calcite.rel.type.RelDataTypeFactory;
import com.hazelcast.org.apache.calcite.rel.type.RelDataTypeField;
import com.hazelcast.org.apache.calcite.rel.type.RelDataTypeFieldImpl;
import com.hazelcast.org.apache.calcite.rex.RexBuilder;
import com.hazelcast.org.apache.calcite.rex.RexCall;
import com.hazelcast.org.apache.calcite.rex.RexCallBinding;
import com.hazelcast.org.apache.calcite.rex.RexCorrelVariable;
import com.hazelcast.org.apache.calcite.rex.RexDynamicParam;
import com.hazelcast.org.apache.calcite.rex.RexExecutor;
import com.hazelcast.org.apache.calcite.rex.RexFieldCollation;
import com.hazelcast.org.apache.calcite.rex.RexInputRef;
import com.hazelcast.org.apache.calcite.rex.RexLiteral;
import com.hazelcast.org.apache.calcite.rex.RexNode;
import com.hazelcast.org.apache.calcite.rex.RexShuttle;
import com.hazelcast.org.apache.calcite.rex.RexSimplify;
import com.hazelcast.org.apache.calcite.rex.RexSubQuery;
import com.hazelcast.org.apache.calcite.rex.RexUtil;
import com.hazelcast.org.apache.calcite.rex.RexWindowBound;
import com.hazelcast.org.apache.calcite.rex.RexWindowBounds;
import com.hazelcast.org.apache.calcite.runtime.Hook;
import com.hazelcast.org.apache.calcite.schema.TransientTable;
import com.hazelcast.org.apache.calcite.schema.impl.ListTransientTable;
import com.hazelcast.org.apache.calcite.sql.SqlAggFunction;
import com.hazelcast.org.apache.calcite.sql.SqlKind;
import com.hazelcast.org.apache.calcite.sql.SqlOperator;
import com.hazelcast.org.apache.calcite.sql.SqlUtil;
import com.hazelcast.org.apache.calcite.sql.SqlWindow;
import com.hazelcast.org.apache.calcite.sql.fun.SqlCountAggFunction;
import com.hazelcast.org.apache.calcite.sql.fun.SqlLikeOperator;
import com.hazelcast.org.apache.calcite.sql.fun.SqlQuantifyOperator;
import com.hazelcast.org.apache.calcite.sql.fun.SqlStdOperatorTable;
import com.hazelcast.org.apache.calcite.sql.type.SqlReturnTypeInference;
import com.hazelcast.org.apache.calcite.sql.type.SqlTypeName;
import com.hazelcast.org.apache.calcite.sql.type.TableFunctionReturnTypeInference;
import com.hazelcast.org.apache.calcite.sql.validate.SqlValidatorUtil;
import com.hazelcast.org.apache.calcite.sql2rel.SqlToRelConverter;
import com.hazelcast.org.apache.calcite.util.Holder;
import com.hazelcast.org.apache.calcite.util.ImmutableBitSet;
import com.hazelcast.org.apache.calcite.util.ImmutableIntList;
import com.hazelcast.org.apache.calcite.util.ImmutableNullableList;
import com.hazelcast.org.apache.calcite.util.Litmus;
import com.hazelcast.org.apache.calcite.util.NlsString;
import com.hazelcast.org.apache.calcite.util.Optionality;
import com.hazelcast.org.apache.calcite.util.Pair;
import com.hazelcast.org.apache.calcite.util.Util;
import com.hazelcast.org.apache.calcite.util.mapping.Mapping;
import com.hazelcast.org.apache.calcite.util.mapping.Mappings;

import com.hazelcast.com.google.common.base.Preconditions;
import com.hazelcast.com.google.common.collect.ImmutableList;
import com.hazelcast.com.google.common.collect.ImmutableMap;
import com.hazelcast.com.google.common.collect.ImmutableSet;
import com.hazelcast.com.google.common.collect.ImmutableSortedMultiset;
import com.hazelcast.com.google.common.collect.Iterables;
import com.hazelcast.com.google.common.collect.Lists;
import com.hazelcast.com.google.common.collect.Multiset;
import com.hazelcast.com.google.common.collect.Sets;

import com.hazelcast.org.checkerframework.checker.nullness.qual.MonotonicNonNull;
import com.hazelcast.org.checkerframework.checker.nullness.qual.Nullable;
import org.immutables.value.Value;

import java.math.BigDecimal;
import java.util.AbstractList;
import java.util.ArrayDeque;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.BitSet;
import java.util.Collections;
import java.util.Deque;
import java.util.EnumSet;
import java.util.HashMap;
import java.util.HashSet;
import java.util.List;
import java.util.Locale;
import java.util.Map;
import java.util.Objects;
import java.util.Set;
import java.util.SortedSet;
import java.util.TreeSet;
import java.util.function.BiFunction;
import java.util.function.Function;
import java.util.function.UnaryOperator;
import java.util.stream.Collectors;
import java.util.stream.StreamSupport;

import static com.hazelcast.org.apache.calcite.linq4j.Nullness.castNonNull;
import static com.hazelcast.org.apache.calcite.sql.SqlKind.UNION;
import static com.hazelcast.org.apache.calcite.util.Static.RESOURCE;

import static java.util.Objects.requireNonNull;

/**
 * Builder for relational expressions.
 *
 * 

{@code RelBuilder} does not make possible anything that you could not * also accomplish by calling the factory methods of the particular relational * expression. But it makes common tasks more straightforward and concise. * *

{@code RelBuilder} uses factories to create relational expressions. * By default, it uses the default factories, which create logical relational * expressions ({@link LogicalFilter}, * {@link LogicalProject} and so forth). * But you could override those factories so that, say, {@code filter} creates * instead a {@code HiveFilter}. * *

It is not thread-safe. */ @Value.Enclosing public class RelBuilder { protected final RelOptCluster cluster; protected final @Nullable RelOptSchema relOptSchema; private final Deque stack = new ArrayDeque<>(); private RexSimplify simplifier; private final Config config; private final RelOptTable.ViewExpander viewExpander; private RelFactories.Struct struct; protected RelBuilder(@Nullable Context context, RelOptCluster cluster, @Nullable RelOptSchema relOptSchema) { this.cluster = cluster; this.relOptSchema = relOptSchema; if (context == null) { context = Contexts.EMPTY_CONTEXT; } this.config = getConfig(context); this.viewExpander = getViewExpander(cluster, context); this.struct = requireNonNull(RelFactories.Struct.fromContext(context)); final RexExecutor executor = context.maybeUnwrap(RexExecutor.class) .orElse( Util.first(cluster.getPlanner().getExecutor(), RexUtil.EXECUTOR)); final RelOptPredicateList predicates = RelOptPredicateList.EMPTY; this.simplifier = new RexSimplify(cluster.getRexBuilder(), predicates, executor); } /** * Derives the view expander * {@link com.hazelcast.org.apache.calcite.plan.RelOptTable.ViewExpander} * to be used for this RelBuilder. * *

The ViewExpander instance is used for expanding views in the default * table scan factory {@code RelFactories.TableScanFactoryImpl}. * You can also define a new table scan factory in the {@code struct} * to override the whole table scan creation. * *

The default view expander does not support expanding views. */ private static RelOptTable.ViewExpander getViewExpander(RelOptCluster cluster, Context context) { return context.maybeUnwrap(RelOptTable.ViewExpander.class) .orElseGet(() -> ViewExpanders.simpleContext(cluster)); } /** Derives the Config to be used for this RelBuilder. * *

Overrides {@link RelBuilder.Config#simplify} if * {@link Hook#REL_BUILDER_SIMPLIFY} is set. */ private static Config getConfig(Context context) { final Config config = context.maybeUnwrap(Config.class).orElse(Config.DEFAULT); boolean simplify = Hook.REL_BUILDER_SIMPLIFY.get(config.simplify()); return config.withSimplify(simplify); } /** Creates a RelBuilder. */ public static RelBuilder create(FrameworkConfig config) { return Frameworks.withPrepare(config, (cluster, relOptSchema, rootSchema, statement) -> new RelBuilder(config.getContext(), cluster, relOptSchema)); } /** Creates a copy of this RelBuilder, with the same state as this, applying * a transform to the config. */ public RelBuilder transform(UnaryOperator transform) { final Context context = Contexts.of(struct, transform.apply(config)); return new RelBuilder(context, cluster, relOptSchema); } /** Performs an action on this RelBuilder. * *

For example, consider the following code: * *

   *   RelNode filterAndRename(RelBuilder relBuilder, RelNode rel,
   *       RexNode condition, List<String> fieldNames) {
   *     relBuilder.push(rel)
   *         .filter(condition);
   *     if (fieldNames != null) {
   *       relBuilder.rename(fieldNames);
   *     }
   *     return relBuilder
   *         .build();
*
* *

The pipeline is disrupted by the 'if'. The {@code let} method * allows you to perform the flow as a single pipeline: * *

   *   RelNode filterAndRename(RelBuilder relBuilder, RelNode rel,
   *       RexNode condition, List<String> fieldNames) {
   *     return relBuilder.push(rel)
   *         .filter(condition)
   *         .let(r -> fieldNames == null ? r : r.rename(fieldNames))
   *         .build();
*
* *

In pipelined cases such as this one, the lambda must return this * RelBuilder. But {@code let} return values of other types. */ public R let(Function consumer) { return consumer.apply(this); } /** Converts this RelBuilder to a string. * The string is the string representation of all of the RelNodes on the stack. */ @Override public String toString() { return stack.stream() .map(frame -> RelOptUtil.toString(frame.rel)) .collect(Collectors.joining("")); } /** Returns the type factory. */ public RelDataTypeFactory getTypeFactory() { return cluster.getTypeFactory(); } /** Returns new RelBuilder that adopts the convention provided. * RelNode will be created with such convention if corresponding factory is provided. */ public RelBuilder adoptConvention(Convention convention) { this.struct = convention.getRelFactories(); return this; } /** Returns the builder for {@link RexNode} expressions. */ public RexBuilder getRexBuilder() { return cluster.getRexBuilder(); } /** Creates a {@link RelBuilderFactory}, a partially-created RelBuilder. * Just add a {@link RelOptCluster} and a {@link RelOptSchema} */ public static RelBuilderFactory proto(final Context context) { return (cluster, schema) -> new RelBuilder(context, cluster, schema); } /** Creates a {@link RelBuilderFactory} that uses a given set of factories. */ public static RelBuilderFactory proto(Object... factories) { return proto(Contexts.of(factories)); } public RelOptCluster getCluster() { return cluster; } public @Nullable RelOptSchema getRelOptSchema() { return relOptSchema; } public RelFactories.TableScanFactory getScanFactory() { return struct.scanFactory; } // Methods for manipulating the stack /** Adds a relational expression to be the input to the next relational * expression constructed. * *

This method is usual when you want to weave in relational expressions * that are not supported by the builder. If, while creating such expressions, * you need to use previously built expressions as inputs, call * {@link #build()} to pop those inputs. */ public RelBuilder push(RelNode node) { stack.push(new Frame(node)); return this; } /** Adds a rel node to the top of the stack while preserving the field names * and aliases. */ private void replaceTop(RelNode node) { final Frame frame = stack.pop(); stack.push(new Frame(node, frame.fields)); } /** Pushes a collection of relational expressions. */ public RelBuilder pushAll(Iterable nodes) { for (RelNode node : nodes) { push(node); } return this; } /** Returns the size of the stack. */ public int size() { return stack.size(); } /** Returns the final relational expression. * *

Throws if the stack is empty. */ public RelNode build() { return stack.pop().rel; } /** Returns the relational expression at the top of the stack, but does not * remove it. */ public RelNode peek() { return castNonNull(peek_()).rel; } private @Nullable Frame peek_() { return stack.peek(); } /** Returns the relational expression {@code n} positions from the top of the * stack, but does not remove it. */ public RelNode peek(int n) { return peek_(n).rel; } private Frame peek_(int n) { if (n == 0) { // more efficient than starting an iterator return Objects.requireNonNull(stack.peek(), "stack.peek"); } return Iterables.get(stack, n); } /** Returns the relational expression {@code n} positions from the top of the * stack, but does not remove it. */ public RelNode peek(int inputCount, int inputOrdinal) { return peek_(inputCount, inputOrdinal).rel; } private Frame peek_(int inputCount, int inputOrdinal) { return peek_(inputCount - 1 - inputOrdinal); } /** Returns the number of fields in all inputs before (to the left of) * the given input. * * @param inputCount Number of inputs * @param inputOrdinal Input ordinal */ private int inputOffset(int inputCount, int inputOrdinal) { int offset = 0; for (int i = 0; i < inputOrdinal; i++) { offset += peek(inputCount, i).getRowType().getFieldCount(); } return offset; } /** Evaluates an expression with a relational expression temporarily on the * stack. */ public E with(RelNode r, Function fn) { try { push(r); return fn.apply(this); } finally { stack.pop(); } } /** Performs an action with a temporary simplifier. */ public E withSimplifier( BiFunction simplifierTransform, Function fn) { final RexSimplify previousSimplifier = this.simplifier; try { this.simplifier = simplifierTransform.apply(this, previousSimplifier); return fn.apply(this); } finally { this.simplifier = previousSimplifier; } } /** Performs an action using predicates of * the {@link #peek() current node} to simplify. */ public E withPredicates(RelMetadataQuery mq, Function fn) { final RelOptPredicateList predicates = mq.getPulledUpPredicates(peek()); return withSimplifier((r, s) -> s.withPredicates(predicates), fn); } // Methods that return scalar expressions /** Creates a literal (constant expression). */ public RexLiteral literal(@Nullable Object value) { final RexBuilder rexBuilder = cluster.getRexBuilder(); if (value == null) { final RelDataType type = getTypeFactory().createSqlType(SqlTypeName.NULL); return rexBuilder.makeNullLiteral(type); } else if (value instanceof Boolean) { return rexBuilder.makeLiteral((Boolean) value); } else if (value instanceof BigDecimal) { return rexBuilder.makeExactLiteral((BigDecimal) value); } else if (value instanceof Float || value instanceof Double) { return rexBuilder.makeApproxLiteral( BigDecimal.valueOf(((Number) value).doubleValue())); } else if (value instanceof Number) { return rexBuilder.makeExactLiteral( BigDecimal.valueOf(((Number) value).longValue())); } else if (value instanceof String) { return rexBuilder.makeLiteral((String) value); } else if (value instanceof Enum) { return rexBuilder.makeLiteral(value, getTypeFactory().createSqlType(SqlTypeName.SYMBOL)); } else { throw new IllegalArgumentException("cannot convert " + value + " (" + value.getClass() + ") to a constant"); } } /** Creates a correlation variable for the current input, and writes it into * a Holder. */ public RelBuilder variable(Holder v) { v.set((RexCorrelVariable) getRexBuilder().makeCorrel(peek().getRowType(), cluster.createCorrel())); return this; } /** Creates a reference to a field by name. * *

Equivalent to {@code field(1, 0, fieldName)}. * * @param fieldName Field name */ public RexInputRef field(String fieldName) { return field(1, 0, fieldName); } /** Creates a reference to a field of given input relational expression * by name. * * @param inputCount Number of inputs * @param inputOrdinal Input ordinal * @param fieldName Field name */ public RexInputRef field(int inputCount, int inputOrdinal, String fieldName) { final Frame frame = peek_(inputCount, inputOrdinal); final List fieldNames = Pair.left(frame.fields()); int i = fieldNames.indexOf(fieldName); if (i >= 0) { return field(inputCount, inputOrdinal, i); } else { throw new IllegalArgumentException("field [" + fieldName + "] not found; input fields are: " + fieldNames); } } /** Creates a reference to an input field by ordinal. * *

Equivalent to {@code field(1, 0, ordinal)}. * * @param fieldOrdinal Field ordinal */ public RexInputRef field(int fieldOrdinal) { return (RexInputRef) field(1, 0, fieldOrdinal, false); } /** Creates a reference to a field of a given input relational expression * by ordinal. * * @param inputCount Number of inputs * @param inputOrdinal Input ordinal * @param fieldOrdinal Field ordinal within input */ public RexInputRef field(int inputCount, int inputOrdinal, int fieldOrdinal) { return (RexInputRef) field(inputCount, inputOrdinal, fieldOrdinal, false); } /** As {@link #field(int, int, int)}, but if {@code alias} is true, the method * may apply an alias to make sure that the field has the same name as in the * input frame. If no alias is applied the expression is definitely a * {@link RexInputRef}. */ private RexNode field(int inputCount, int inputOrdinal, int fieldOrdinal, boolean alias) { final Frame frame = peek_(inputCount, inputOrdinal); final RelNode input = frame.rel; final RelDataType rowType = input.getRowType(); if (fieldOrdinal < 0 || fieldOrdinal > rowType.getFieldCount()) { throw new IllegalArgumentException("field ordinal [" + fieldOrdinal + "] out of range; input fields are: " + rowType.getFieldNames()); } final RelDataTypeField field = rowType.getFieldList().get(fieldOrdinal); final int offset = inputOffset(inputCount, inputOrdinal); final RexInputRef ref = cluster.getRexBuilder() .makeInputRef(field.getType(), offset + fieldOrdinal); final RelDataTypeField aliasField = frame.fields().get(fieldOrdinal); if (!alias || field.getName().equals(aliasField.getName())) { return ref; } else { return alias(ref, aliasField.getName()); } } /** Creates a reference to a field of the current record which originated * in a relation with a given alias. */ public RexNode field(String alias, String fieldName) { return field(1, alias, fieldName); } /** Creates a reference to a field which originated in a relation with the * given alias. Searches for the relation starting at the top of the * stack. */ public RexNode field(int inputCount, String alias, String fieldName) { requireNonNull(alias, "alias"); requireNonNull(fieldName, "fieldName"); final List fields = new ArrayList<>(); for (int inputOrdinal = 0; inputOrdinal < inputCount; ++inputOrdinal) { final Frame frame = peek_(inputOrdinal); for (Ord p : Ord.zip(frame.fields)) { // If alias and field name match, reference that field. if (p.e.left.contains(alias) && p.e.right.getName().equals(fieldName)) { return field(inputCount, inputCount - 1 - inputOrdinal, p.i); } fields.add( String.format(Locale.ROOT, "{aliases=%s,fieldName=%s}", p.e.left, p.e.right.getName())); } } throw new IllegalArgumentException("{alias=" + alias + ",fieldName=" + fieldName + "} " + "field not found; fields are: " + fields); } /** Returns a reference to a given field of a record-valued expression. */ public RexNode field(RexNode e, String name) { return getRexBuilder().makeFieldAccess(e, name, false); } /** Returns references to the fields of the top input. */ public ImmutableList fields() { return fields(1, 0); } /** Returns references to the fields of a given input. */ public ImmutableList fields(int inputCount, int inputOrdinal) { final RelNode input = peek(inputCount, inputOrdinal); final RelDataType rowType = input.getRowType(); final ImmutableList.Builder nodes = ImmutableList.builder(); for (int fieldOrdinal : Util.range(rowType.getFieldCount())) { nodes.add(field(inputCount, inputOrdinal, fieldOrdinal)); } return nodes.build(); } /** Returns references to fields for a given collation. */ public ImmutableList fields(RelCollation collation) { final ImmutableList.Builder nodes = ImmutableList.builder(); for (RelFieldCollation fieldCollation : collation.getFieldCollations()) { RexNode node = field(fieldCollation.getFieldIndex()); switch (fieldCollation.direction) { case DESCENDING: node = desc(node); break; default: break; } switch (fieldCollation.nullDirection) { case FIRST: node = nullsFirst(node); break; case LAST: node = nullsLast(node); break; default: break; } nodes.add(node); } return nodes.build(); } /** Returns references to fields for a given list of input ordinals. */ public ImmutableList fields(List ordinals) { final ImmutableList.Builder nodes = ImmutableList.builder(); for (Number ordinal : ordinals) { RexNode node = field(1, 0, ordinal.intValue(), false); nodes.add(node); } return nodes.build(); } /** Returns references to fields for a given bit set of input ordinals. */ public ImmutableList fields(ImmutableBitSet ordinals) { return fields(ordinals.asList()); } /** Returns references to fields identified by name. */ public ImmutableList fields(Iterable fieldNames) { final ImmutableList.Builder builder = ImmutableList.builder(); for (String fieldName : fieldNames) { builder.add(field(fieldName)); } return builder.build(); } /** Returns references to fields identified by a mapping. */ public ImmutableList fields(Mappings.TargetMapping mapping) { return fields(Mappings.asListNonNull(mapping)); } /** Creates an access to a field by name. */ public RexNode dot(RexNode node, String fieldName) { final RexBuilder builder = cluster.getRexBuilder(); return builder.makeFieldAccess(node, fieldName, true); } /** Creates an access to a field by ordinal. */ public RexNode dot(RexNode node, int fieldOrdinal) { final RexBuilder builder = cluster.getRexBuilder(); return builder.makeFieldAccess(node, fieldOrdinal); } /** Creates a call to a scalar operator. */ public RexNode call(SqlOperator operator, RexNode... operands) { return call(operator, ImmutableList.copyOf(operands)); } /** Creates a call to a scalar operator. */ private RexCall call(SqlOperator operator, List operandList) { switch (operator.getKind()) { case LIKE: case SIMILAR: final SqlLikeOperator likeOperator = (SqlLikeOperator) operator; if (likeOperator.isNegated()) { final SqlOperator notLikeOperator = likeOperator.not(); return (RexCall) not(call(notLikeOperator, operandList)); } break; case BETWEEN: assert operandList.size() == 3; return (RexCall) between(operandList.get(0), operandList.get(1), operandList.get(2)); default: break; } final RexBuilder builder = cluster.getRexBuilder(); final RelDataType type = builder.deriveReturnType(operator, operandList); return (RexCall) builder.makeCall(type, operator, operandList); } /** Creates a call to a scalar operator. */ public RexNode call(SqlOperator operator, Iterable operands) { return call(operator, ImmutableList.copyOf(operands)); } /** Creates an IN predicate with a list of values. * *

For example, *

{@code
   * b.scan("Emp")
   *     .filter(b.in(b.field("deptno"), b.literal(10), b.literal(20)))
   * }
* is equivalent to SQL *
{@code
   * SELECT *
   * FROM Emp
   * WHERE deptno IN (10, 20)
   * }
*/ public RexNode in(RexNode arg, RexNode... ranges) { return in(arg, ImmutableList.copyOf(ranges)); } /** Creates an IN predicate with a list of values. * *

For example, * *

{@code
   * b.scan("Emps")
   *     .filter(
   *         b.in(b.field("deptno"),
   *             Arrays.asList(b.literal(10), b.literal(20))))
   * }
* *

is equivalent to the SQL * *

{@code
   * SELECT *
   * FROM Emps
   * WHERE deptno IN (10, 20)
   * }
*/ public RexNode in(RexNode arg, Iterable ranges) { return getRexBuilder().makeIn(arg, ImmutableList.copyOf(ranges)); } /** Creates an IN predicate with a sub-query. */ @Experimental public RexSubQuery in(RelNode rel, Iterable nodes) { return RexSubQuery.in(rel, ImmutableList.copyOf(nodes)); } /** Creates an IN predicate with a sub-query. * *

For example, * *

{@code
   * b.scan("Emps")
   *     .filter(
   *         b.in(b.field("deptno"),
   *             b2 -> b2.scan("Depts")
   *                 .filter(
   *                     b2.eq(b2.field("location"), b2.literal("Boston")))
   *                 .project(b.field("deptno"))
   *                 .build()))
   * }
* *

is equivalent to the SQL * *

{@code
   * SELECT *
   * FROM Emps
   * WHERE deptno IN (SELECT deptno FROM Dept WHERE location = 'Boston')
   * }
*/ @Experimental public RexNode in(RexNode arg, Function f) { final RelNode rel = f.apply(this); return RexSubQuery.in(rel, ImmutableList.of(arg)); } /** Creates a SOME (or ANY) predicate. * *

For example, * *

{@code
   * b.scan("Emps")
   *     .filter(
   *         b.some(b.field("commission"),
   *             SqlStdOperatorTable.GREATER_THAN,
   *             b2 -> b2.scan("Emps")
   *                 .filter(
   *                     b2.eq(b2.field("job"), b2.literal("Manager")))
   *                 .project(b2.field("sal"))
   *                 .build()))
   * }
* *

is equivalent to the SQL * *

{@code
   * SELECT *
   * FROM Emps
   * WHERE commission > SOME (SELECT sal FROM Emps WHERE job = 'Manager')
   * }
* *

or (since {@code SOME} and {@code ANY} are synonyms) the SQL * *

{@code
   * SELECT *
   * FROM Emps
   * WHERE commission > ANY (SELECT sal FROM Emps WHERE job = 'Manager')
   * }
*/ @Experimental public RexSubQuery some(RexNode node, SqlOperator op, Function f) { return some_(node, op.kind, f); } private RexSubQuery some_(RexNode node, SqlKind kind, Function f) { final RelNode rel = f.apply(this); final SqlQuantifyOperator quantifyOperator = SqlStdOperatorTable.some(kind); return RexSubQuery.some(rel, ImmutableList.of(node), quantifyOperator); } /** Creates an ALL predicate. * *

For example, * *

{@code
   * b.scan("Emps")
   *     .filter(
   *         b.all(b.field("commission"),
   *             SqlStdOperatorTable.GREATER_THAN,
   *             b2 -> b2.scan("Emps")
   *                 .filter(
   *                     b2.eq(b2.field("job"), b2.literal("Manager")))
   *                 .project(b2.field("sal"))
   *                 .build()))
   * }
* *

is equivalent to the SQL * *

{@code
   * SELECT *
   * FROM Emps
   * WHERE commission > ALL (SELECT sal FROM Emps WHERE job = 'Manager')
   * }
* *

Calcite translates {@code ALL} predicates to {@code NOT SOME}. The * following SQL is equivalent to the previous: * *

{@code
   * SELECT *
   * FROM Emps
   * WHERE NOT (commission <= SOME (SELECT sal FROM Emps WHERE job = 'Manager'))
   * }
*/ @Experimental public RexNode all(RexNode node, SqlOperator op, Function f) { return not(some_(node, op.kind.negateNullSafe(), f)); } /** Creates an EXISTS predicate. * *

For example, * *

{@code
   * b.scan("Depts")
   *     .filter(
   *         b.exists(b2 ->
   *             b2.scan("Emps")
   *                 .filter(
   *                     b2.eq(b2.field("job"), b2.literal("Manager")))
   *                 .build()))
   * }
* *

is equivalent to the SQL * *

{@code
   * SELECT *
   * FROM Depts
   * WHERE EXISTS (SELECT 1 FROM Emps WHERE job = 'Manager')
   * }
*/ @Experimental public RexSubQuery exists(Function f) { final RelNode rel = f.apply(this); return RexSubQuery.exists(rel); } /** Creates a UNIQUE predicate. * *

For example, * *

{@code
   * b.scan("Depts")
   *     .filter(
   *         b.exists(b2 ->
   *             b2.scan("Emps")
   *                 .filter(
   *                     b2.eq(b2.field("job"), b2.literal("Manager")))
   *                 .project(b2.field("deptno")
   *                 .build()))
   * }
* *

is equivalent to the SQL * *

{@code
   * SELECT *
   * FROM Depts
   * WHERE UNIQUE (SELECT deptno FROM Emps WHERE job = 'Manager')
   * }
*/ @Experimental public RexSubQuery unique(Function f) { final RelNode rel = f.apply(this); return RexSubQuery.unique(rel); } /** Creates a scalar sub-query. * *

For example, * *

{@code
   * b.scan("Depts")
   *     .project(
   *         b.field("deptno")
   *         b.scalarQuery(b2 ->
   *             b2.scan("Emps")
   *                 .aggregate(
   *                     b2.eq(b2.field("job"), b2.literal("Manager")))
   *                 .build()))
   * }
* *

is equivalent to the SQL * *

{@code
   * SELECT deptno, (SELECT MAX(sal) FROM Emps)
   * FROM Depts
   * }
*/ @Experimental public RexSubQuery scalarQuery(Function f) { return RexSubQuery.scalar(f.apply(this)); } /** Creates an ARRAY sub-query. * *

For example, * *

{@code
   * b.scan("Depts")
   *     .project(
   *         b.field("deptno")
   *         b.arrayQuery(b2 ->
   *             b2.scan("Emps")
   *                 .build()))
   * }
* *

is equivalent to the SQL * *

{@code
   * SELECT deptno, ARRAY (SELECT * FROM Emps)
   * FROM Depts
   * }
*/ @Experimental public RexSubQuery arrayQuery(Function f) { return RexSubQuery.array(f.apply(this)); } /** Creates a MULTISET sub-query. * *

For example, * *

{@code
   * b.scan("Depts")
   *     .project(
   *         b.field("deptno")
   *         b.multisetQuery(b2 ->
   *             b2.scan("Emps")
   *                 .build()))
   * }
* *

is equivalent to the SQL * *

{@code
   * SELECT deptno, MULTISET (SELECT * FROM Emps)
   * FROM Depts
   * }
*/ @Experimental public RexSubQuery multisetQuery(Function f) { return RexSubQuery.multiset(f.apply(this)); } /** Creates a MAP sub-query. * *

For example, * *

{@code
   * b.scan("Depts")
   *     .project(
   *         b.field("deptno")
   *         b.multisetQuery(b2 ->
   *             b2.scan("Emps")
   *                 .project(b2.field("empno"), b2.field("job"))
   *                 .build()))
   * }
* *

is equivalent to the SQL * *

{@code
   * SELECT deptno, MAP (SELECT empno, job FROM Emps)
   * FROM Depts
   * }
*/ @Experimental public RexSubQuery mapQuery(Function f) { return RexSubQuery.map(f.apply(this)); } /** Creates an AND. */ public RexNode and(RexNode... operands) { return and(ImmutableList.copyOf(operands)); } /** Creates an AND. * *

Simplifies the expression a little: * {@code e AND TRUE} becomes {@code e}; * {@code e AND e2 AND NOT e} becomes {@code e2}. */ public RexNode and(Iterable operands) { return RexUtil.composeConjunction(getRexBuilder(), operands); } /** Creates an OR. */ public RexNode or(RexNode... operands) { return or(ImmutableList.copyOf(operands)); } /** Creates an OR. */ public RexNode or(Iterable operands) { return RexUtil.composeDisjunction(cluster.getRexBuilder(), operands); } /** Creates a NOT. */ public RexNode not(RexNode operand) { return call(SqlStdOperatorTable.NOT, operand); } /** Creates an {@code =}. */ public RexNode equals(RexNode operand0, RexNode operand1) { return call(SqlStdOperatorTable.EQUALS, operand0, operand1); } /** Creates a {@code >}. */ public RexNode greaterThan(RexNode operand0, RexNode operand1) { return call(SqlStdOperatorTable.GREATER_THAN, operand0, operand1); } /** Creates a {@code >=}. */ public RexNode greaterThanOrEqual(RexNode operand0, RexNode operand1) { return call(SqlStdOperatorTable.GREATER_THAN_OR_EQUAL, operand0, operand1); } /** Creates a {@code <}. */ public RexNode lessThan(RexNode operand0, RexNode operand1) { return call(SqlStdOperatorTable.LESS_THAN, operand0, operand1); } /** Creates a {@code <=}. */ public RexNode lessThanOrEqual(RexNode operand0, RexNode operand1) { return call(SqlStdOperatorTable.LESS_THAN_OR_EQUAL, operand0, operand1); } /** Creates a {@code <>}. */ public RexNode notEquals(RexNode operand0, RexNode operand1) { return call(SqlStdOperatorTable.NOT_EQUALS, operand0, operand1); } /** Creates an expression equivalent to "{@code o0 IS NOT DISTINCT FROM o1}". * It is also equivalent to * "{@code o0 = o1 OR (o0 IS NULL AND o1 IS NULL)}". */ public RexNode isNotDistinctFrom(RexNode operand0, RexNode operand1) { return RelOptUtil.isDistinctFrom(getRexBuilder(), operand0, operand1, true); } /** Creates an expression equivalent to {@code o0 IS DISTINCT FROM o1}. * It is also equivalent to * "{@code NOT (o0 = o1 OR (o0 IS NULL AND o1 IS NULL))}. */ public RexNode isDistinctFrom(RexNode operand0, RexNode operand1) { return RelOptUtil.isDistinctFrom(getRexBuilder(), operand0, operand1, false); } /** Creates a {@code BETWEEN}. */ public RexNode between(RexNode arg, RexNode lower, RexNode upper) { return getRexBuilder().makeBetween(arg, lower, upper); } /** Creates ab {@code IS NULL}. */ public RexNode isNull(RexNode operand) { return call(SqlStdOperatorTable.IS_NULL, operand); } /** Creates an {@code IS NOT NULL}. */ public RexNode isNotNull(RexNode operand) { return call(SqlStdOperatorTable.IS_NOT_NULL, operand); } /** Creates an expression that casts an expression to a given type. */ public RexNode cast(RexNode expr, SqlTypeName typeName) { final RelDataType type = cluster.getTypeFactory().createSqlType(typeName); return cluster.getRexBuilder().makeCast(type, expr); } /** Creates an expression that casts an expression to a type with a given name * and precision or length. */ public RexNode cast(RexNode expr, SqlTypeName typeName, int precision) { final RelDataType type = cluster.getTypeFactory().createSqlType(typeName, precision); return cluster.getRexBuilder().makeCast(type, expr); } /** Creates an expression that casts an expression to a type with a given * name, precision and scale. */ public RexNode cast(RexNode expr, SqlTypeName typeName, int precision, int scale) { final RelDataType type = cluster.getTypeFactory().createSqlType(typeName, precision, scale); return cluster.getRexBuilder().makeCast(type, expr); } /** * Returns an expression wrapped in an alias. * *

This method is idempotent: If the expression is already wrapped in the * correct alias, does nothing; if wrapped in an incorrect alias, removes * the incorrect alias and applies the correct alias. * * @see #project */ public RexNode alias(RexNode expr, @Nullable String alias) { final RexNode aliasLiteral = literal(alias); switch (expr.getKind()) { case AS: final RexCall call = (RexCall) expr; if (call.operands.get(1).equals(aliasLiteral)) { // current alias is correct return expr; } expr = call.operands.get(0); // strip current (incorrect) alias, and fall through default: return call(SqlStdOperatorTable.AS, expr, aliasLiteral); } } /** Converts a sort expression to descending. */ public RexNode desc(RexNode node) { return call(SqlStdOperatorTable.DESC, node); } /** Converts a sort expression to nulls last. */ public RexNode nullsLast(RexNode node) { return call(SqlStdOperatorTable.NULLS_LAST, node); } /** Converts a sort expression to nulls first. */ public RexNode nullsFirst(RexNode node) { return call(SqlStdOperatorTable.NULLS_FIRST, node); } // Methods that create window bounds /** Creates an {@code UNBOUNDED PRECEDING} window bound, * for use in methods such as {@link OverCall#rowsFrom(RexWindowBound)} * and {@link OverCall#rangeBetween(RexWindowBound, RexWindowBound)}. */ public RexWindowBound unboundedPreceding() { return RexWindowBounds.UNBOUNDED_PRECEDING; } /** Creates a {@code bound PRECEDING} window bound, * for use in methods such as {@link OverCall#rowsFrom(RexWindowBound)} * and {@link OverCall#rangeBetween(RexWindowBound, RexWindowBound)}. */ public RexWindowBound preceding(RexNode bound) { return RexWindowBounds.preceding(bound); } /** Creates a {@code CURRENT ROW} window bound, * for use in methods such as {@link OverCall#rowsFrom(RexWindowBound)} * and {@link OverCall#rangeBetween(RexWindowBound, RexWindowBound)}. */ public RexWindowBound currentRow() { return RexWindowBounds.CURRENT_ROW; } /** Creates a {@code bound FOLLOWING} window bound, * for use in methods such as {@link OverCall#rowsFrom(RexWindowBound)} * and {@link OverCall#rangeBetween(RexWindowBound, RexWindowBound)}. */ public RexWindowBound following(RexNode bound) { return RexWindowBounds.following(bound); } /** Creates an {@code UNBOUNDED FOLLOWING} window bound, * for use in methods such as {@link OverCall#rowsFrom(RexWindowBound)} * and {@link OverCall#rangeBetween(RexWindowBound, RexWindowBound)}. */ public RexWindowBound unboundedFollowing() { return RexWindowBounds.UNBOUNDED_FOLLOWING; } // Methods that create group keys and aggregate calls /** Creates an empty group key. */ public GroupKey groupKey() { return groupKey(ImmutableList.of()); } /** Creates a group key. */ public GroupKey groupKey(RexNode... nodes) { return groupKey(ImmutableList.copyOf(nodes)); } /** Creates a group key. */ public GroupKey groupKey(Iterable nodes) { return new GroupKeyImpl(ImmutableList.copyOf(nodes), null, null); } /** Creates a group key with grouping sets. */ public GroupKey groupKey(Iterable nodes, Iterable> nodeLists) { return groupKey_(nodes, nodeLists); } // CHECKSTYLE: IGNORE 1 /** @deprecated Now that indicator is deprecated, use * {@link #groupKey(Iterable, Iterable)}, which has the same behavior as * calling this method with {@code indicator = false}. */ @Deprecated // to be removed before 2.0 public GroupKey groupKey(Iterable nodes, boolean indicator, Iterable> nodeLists) { Aggregate.checkIndicator(indicator); return groupKey_(nodes, nodeLists); } private static GroupKey groupKey_(Iterable nodes, Iterable> nodeLists) { final ImmutableList.Builder> builder = ImmutableList.builder(); for (Iterable nodeList : nodeLists) { builder.add(ImmutableList.copyOf(nodeList)); } return new GroupKeyImpl(ImmutableList.copyOf(nodes), builder.build(), null); } /** Creates a group key of fields identified by ordinal. */ public GroupKey groupKey(int... fieldOrdinals) { return groupKey(fields(ImmutableIntList.of(fieldOrdinals))); } /** Creates a group key of fields identified by name. */ public GroupKey groupKey(String... fieldNames) { return groupKey(fields(ImmutableList.copyOf(fieldNames))); } /** Creates a group key, identified by field positions * in the underlying relational expression. * *

This method of creating a group key does not allow you to group on new * expressions, only column projections, but is efficient, especially when you * are coming from an existing {@link Aggregate}. */ public GroupKey groupKey(ImmutableBitSet groupSet) { return groupKey_(groupSet, ImmutableList.of(groupSet)); } /** Creates a group key with grouping sets, both identified by field positions * in the underlying relational expression. * *

This method of creating a group key does not allow you to group on new * expressions, only column projections, but is efficient, especially when you * are coming from an existing {@link Aggregate}. * *

It is possible for {@code groupSet} to be strict superset of all * {@code groupSets}. For example, in the pseudo SQL * *

{@code
   * GROUP BY 0, 1, 2
   * GROUPING SETS ((0, 1), 0)
   * }
* *

column 2 does not appear in either grouping set. This is not valid SQL. * We can approximate in actual SQL by adding an extra grouping set and * filtering out using {@code HAVING}, as follows: * *

{@code
   * GROUP BY GROUPING SETS ((0, 1, 2), (0, 1), 0)
   * HAVING GROUPING_ID(0, 1, 2) <> 0
   * }
*/ public GroupKey groupKey(ImmutableBitSet groupSet, Iterable groupSets) { return groupKey_(groupSet, ImmutableList.copyOf(groupSets)); } // CHECKSTYLE: IGNORE 1 /** @deprecated Use {@link #groupKey(ImmutableBitSet, Iterable)}. */ @Deprecated // to be removed before 2.0 public GroupKey groupKey(ImmutableBitSet groupSet, boolean indicator, @Nullable ImmutableList groupSets) { Aggregate.checkIndicator(indicator); return groupKey_(groupSet, groupSets == null ? ImmutableList.of(groupSet) : ImmutableList.copyOf(groupSets)); } private GroupKey groupKey_(ImmutableBitSet groupSet, ImmutableList groupSets) { if (groupSet.length() > peek().getRowType().getFieldCount()) { throw new IllegalArgumentException("out of bounds: " + groupSet); } requireNonNull(groupSets, "groupSets"); final ImmutableList nodes = fields(groupSet); return groupKey_(nodes, Util.transform(groupSets, this::fields)); } @Deprecated // to be removed before 2.0 public AggCall aggregateCall(SqlAggFunction aggFunction, boolean distinct, RexNode filter, @Nullable String alias, RexNode... operands) { return aggregateCall(aggFunction, distinct, false, false, filter, null, ImmutableList.of(), alias, ImmutableList.copyOf(operands)); } @Deprecated // to be removed before 2.0 public AggCall aggregateCall(SqlAggFunction aggFunction, boolean distinct, boolean approximate, RexNode filter, @Nullable String alias, RexNode... operands) { return aggregateCall(aggFunction, distinct, approximate, false, filter, null, ImmutableList.of(), alias, ImmutableList.copyOf(operands)); } @Deprecated // to be removed before 2.0 public AggCall aggregateCall(SqlAggFunction aggFunction, boolean distinct, RexNode filter, @Nullable String alias, Iterable operands) { return aggregateCall(aggFunction, distinct, false, false, filter, null, ImmutableList.of(), alias, ImmutableList.copyOf(operands)); } @Deprecated // to be removed before 2.0 public AggCall aggregateCall(SqlAggFunction aggFunction, boolean distinct, boolean approximate, RexNode filter, @Nullable String alias, Iterable operands) { return aggregateCall(aggFunction, distinct, approximate, false, filter, null, ImmutableList.of(), alias, ImmutableList.copyOf(operands)); } /** Creates a call to an aggregate function. * *

To add other operands, apply * {@link AggCall#distinct()}, * {@link AggCall#approximate(boolean)}, * {@link AggCall#filter(RexNode...)}, * {@link AggCall#sort}, * {@link AggCall#as} to the result. */ public AggCall aggregateCall(SqlAggFunction aggFunction, Iterable operands) { return aggregateCall(aggFunction, false, false, false, null, null, ImmutableList.of(), null, ImmutableList.copyOf(operands)); } /** Creates a call to an aggregate function. * *

To add other operands, apply * {@link AggCall#distinct()}, * {@link AggCall#approximate(boolean)}, * {@link AggCall#filter(RexNode...)}, * {@link AggCall#sort}, * {@link AggCall#as} to the result. */ public AggCall aggregateCall(SqlAggFunction aggFunction, RexNode... operands) { return aggregateCall(aggFunction, false, false, false, null, null, ImmutableList.of(), null, ImmutableList.copyOf(operands)); } /** Creates a call to an aggregate function as a copy of an * {@link AggregateCall}. */ public AggCall aggregateCall(AggregateCall a) { return aggregateCall(a.getAggregation(), a.isDistinct(), a.isApproximate(), a.ignoreNulls(), a.filterArg < 0 ? null : field(a.filterArg), a.distinctKeys == null ? null : fields(a.distinctKeys), fields(a.collation), a.name, fields(a.getArgList())); } /** Creates a call to an aggregate function as a copy of an * {@link AggregateCall}, applying a mapping. */ public AggCall aggregateCall(AggregateCall a, Mapping mapping) { return aggregateCall(a.getAggregation(), a.isDistinct(), a.isApproximate(), a.ignoreNulls(), a.filterArg < 0 ? null : field(Mappings.apply(mapping, a.filterArg)), a.distinctKeys == null ? null : fields(Mappings.apply(mapping, a.distinctKeys)), fields(RexUtil.apply(mapping, a.collation)), a.name, fields(Mappings.apply2(mapping, a.getArgList()))); } /** Creates a call to an aggregate function with all applicable operands. */ protected AggCall aggregateCall(SqlAggFunction aggFunction, boolean distinct, boolean approximate, boolean ignoreNulls, @Nullable RexNode filter, @Nullable ImmutableList distinctKeys, ImmutableList orderKeys, @Nullable String alias, ImmutableList operands) { return new AggCallImpl(aggFunction, distinct, approximate, ignoreNulls, filter, alias, operands, distinctKeys, orderKeys); } /** Creates a call to the {@code COUNT} aggregate function. */ public AggCall count(RexNode... operands) { return count(false, null, operands); } /** Creates a call to the {@code COUNT} aggregate function. */ public AggCall count(Iterable operands) { return count(false, null, operands); } /** Creates a call to the {@code COUNT} aggregate function, * optionally distinct and with an alias. */ public AggCall count(boolean distinct, @Nullable String alias, RexNode... operands) { return aggregateCall(SqlStdOperatorTable.COUNT, distinct, false, false, null, null, ImmutableList.of(), alias, ImmutableList.copyOf(operands)); } /** Creates a call to the {@code COUNT} aggregate function, * optionally distinct and with an alias. */ public AggCall count(boolean distinct, @Nullable String alias, Iterable operands) { return aggregateCall(SqlStdOperatorTable.COUNT, distinct, false, false, null, null, ImmutableList.of(), alias, ImmutableList.copyOf(operands)); } /** Creates a call to the {@code COUNT(*)} aggregate function. */ public AggCall countStar(@Nullable String alias) { return count(false, alias); } /** Creates a call to the {@code SUM} aggregate function. */ public AggCall sum(RexNode operand) { return sum(false, null, operand); } /** Creates a call to the {@code SUM} aggregate function, * optionally distinct and with an alias. */ public AggCall sum(boolean distinct, @Nullable String alias, RexNode operand) { return aggregateCall(SqlStdOperatorTable.SUM, distinct, false, false, null, null, ImmutableList.of(), alias, ImmutableList.of(operand)); } /** Creates a call to the {@code AVG} aggregate function. */ public AggCall avg(RexNode operand) { return avg(false, null, operand); } /** Creates a call to the {@code AVG} aggregate function, * optionally distinct and with an alias. */ public AggCall avg(boolean distinct, @Nullable String alias, RexNode operand) { return aggregateCall(SqlStdOperatorTable.AVG, distinct, false, false, null, null, ImmutableList.of(), alias, ImmutableList.of(operand)); } /** Creates a call to the {@code MIN} aggregate function. */ public AggCall min(RexNode operand) { return min(null, operand); } /** Creates a call to the {@code MIN} aggregate function, * optionally with an alias. */ public AggCall min(@Nullable String alias, RexNode operand) { return aggregateCall(SqlStdOperatorTable.MIN, false, false, false, null, null, ImmutableList.of(), alias, ImmutableList.of(operand)); } /** Creates a call to the {@code MAX} aggregate function, * optionally with an alias. */ public AggCall max(RexNode operand) { return max(null, operand); } /** Creates a call to the {@code MAX} aggregate function. */ public AggCall max(@Nullable String alias, RexNode operand) { return aggregateCall(SqlStdOperatorTable.MAX, false, false, false, null, null, ImmutableList.of(), alias, ImmutableList.of(operand)); } // Methods for patterns /** * Creates a reference to a given field of the pattern. * * @param alpha the pattern name * @param type Type of field * @param i Ordinal of field * @return Reference to field of pattern */ public RexNode patternField(String alpha, RelDataType type, int i) { return getRexBuilder().makePatternFieldRef(alpha, type, i); } /** Creates a call that concatenates patterns; * for use in {@link #match}. */ public RexNode patternConcat(Iterable nodes) { final ImmutableList list = ImmutableList.copyOf(nodes); if (list.size() > 2) { // Convert into binary calls return patternConcat(patternConcat(Util.skipLast(list)), Util.last(list)); } final RelDataType t = getTypeFactory().createSqlType(SqlTypeName.NULL); return getRexBuilder().makeCall(t, SqlStdOperatorTable.PATTERN_CONCAT, list); } /** Creates a call that concatenates patterns; * for use in {@link #match}. */ public RexNode patternConcat(RexNode... nodes) { return patternConcat(ImmutableList.copyOf(nodes)); } /** Creates a call that creates alternate patterns; * for use in {@link #match}. */ public RexNode patternAlter(Iterable nodes) { final RelDataType t = getTypeFactory().createSqlType(SqlTypeName.NULL); return getRexBuilder().makeCall(t, SqlStdOperatorTable.PATTERN_ALTER, ImmutableList.copyOf(nodes)); } /** Creates a call that creates alternate patterns; * for use in {@link #match}. */ public RexNode patternAlter(RexNode... nodes) { return patternAlter(ImmutableList.copyOf(nodes)); } /** Creates a call that creates quantify patterns; * for use in {@link #match}. */ public RexNode patternQuantify(Iterable nodes) { final RelDataType t = getTypeFactory().createSqlType(SqlTypeName.NULL); return getRexBuilder().makeCall(t, SqlStdOperatorTable.PATTERN_QUANTIFIER, ImmutableList.copyOf(nodes)); } /** Creates a call that creates quantify patterns; * for use in {@link #match}. */ public RexNode patternQuantify(RexNode... nodes) { return patternQuantify(ImmutableList.copyOf(nodes)); } /** Creates a call that creates permute patterns; * for use in {@link #match}. */ public RexNode patternPermute(Iterable nodes) { final RelDataType t = getTypeFactory().createSqlType(SqlTypeName.NULL); return getRexBuilder().makeCall(t, SqlStdOperatorTable.PATTERN_PERMUTE, ImmutableList.copyOf(nodes)); } /** Creates a call that creates permute patterns; * for use in {@link #match}. */ public RexNode patternPermute(RexNode... nodes) { return patternPermute(ImmutableList.copyOf(nodes)); } /** Creates a call that creates an exclude pattern; * for use in {@link #match}. */ public RexNode patternExclude(RexNode node) { final RelDataType t = getTypeFactory().createSqlType(SqlTypeName.NULL); return getRexBuilder().makeCall(t, SqlStdOperatorTable.PATTERN_EXCLUDE, ImmutableList.of(node)); } // Methods that create relational expressions /** Creates a {@link TableScan} of the table * with a given name. * *

Throws if the table does not exist. * *

Returns this builder. * * @param tableNames Name of table (can optionally be qualified) */ public RelBuilder scan(Iterable tableNames) { final List names = ImmutableList.copyOf(tableNames); requireNonNull(relOptSchema, "relOptSchema"); final RelOptTable relOptTable = relOptSchema.getTableForMember(names); if (relOptTable == null) { throw RESOURCE.tableNotFound(String.join(".", names)).ex(); } final RelNode scan = struct.scanFactory.createScan( ViewExpanders.toRelContext(viewExpander, cluster), relOptTable); push(scan); rename(relOptTable.getRowType().getFieldNames()); // When the node is not a TableScan but from expansion, // we need to explicitly add the alias. if (!(scan instanceof TableScan)) { as(Util.last(ImmutableList.copyOf(tableNames))); } return this; } /** Creates a {@link TableScan} of the table * with a given name. * *

Throws if the table does not exist. * *

Returns this builder. * * @param tableNames Name of table (can optionally be qualified) */ public RelBuilder scan(String... tableNames) { return scan(ImmutableList.copyOf(tableNames)); } /** Creates a {@link Snapshot} of a given snapshot period. * *

Returns this builder. * * @param period Name of table (can optionally be qualified) */ public RelBuilder snapshot(RexNode period) { final Frame frame = stack.pop(); final RelNode snapshot = struct.snapshotFactory.createSnapshot(frame.rel, period); stack.push(new Frame(snapshot, frame.fields)); return this; } /** * Gets column mappings of the operator. * * @param op operator instance * @return column mappings associated with this function */ private static @Nullable Set getColumnMappings(SqlOperator op) { SqlReturnTypeInference inference = op.getReturnTypeInference(); if (inference instanceof TableFunctionReturnTypeInference) { return ((TableFunctionReturnTypeInference) inference).getColumnMappings(); } else { return null; } } /** * Creates a RexCall to the {@code CURSOR} function by ordinal. * * @param inputCount Number of inputs * @param ordinal The reference to the relational input * @return RexCall to CURSOR function */ public RexNode cursor(int inputCount, int ordinal) { if (inputCount <= ordinal || ordinal < 0) { throw new IllegalArgumentException("bad input count or ordinal"); } // Refer to the "ordinal"th input as if it were a field // (because that's how things are laid out inside a TableFunctionScan) final RelNode input = peek(inputCount, ordinal); return call(SqlStdOperatorTable.CURSOR, getRexBuilder().makeInputRef(input.getRowType(), ordinal)); } /** Creates a {@link TableFunctionScan}. */ public RelBuilder functionScan(SqlOperator operator, int inputCount, RexNode... operands) { return functionScan(operator, inputCount, ImmutableList.copyOf(operands)); } /** Creates a {@link TableFunctionScan}. */ public RelBuilder functionScan(SqlOperator operator, int inputCount, Iterable operands) { if (inputCount < 0 || inputCount > stack.size()) { throw new IllegalArgumentException("bad input count"); } // Gets inputs. final List inputs = new ArrayList<>(); for (int i = 0; i < inputCount; i++) { inputs.add(0, build()); } final RexCall call = call(operator, ImmutableList.copyOf(operands)); final RelNode functionScan = struct.tableFunctionScanFactory.createTableFunctionScan(cluster, inputs, call, null, getColumnMappings(operator)); push(functionScan); return this; } /** Creates a {@link Filter} of an array of * predicates. * *

The predicates are combined using AND, * and optimized in a similar way to the {@link #and} method. * If the result is TRUE no filter is created. */ public RelBuilder filter(RexNode... predicates) { return filter(ImmutableSet.of(), ImmutableList.copyOf(predicates)); } /** Creates a {@link Filter} of a list of * predicates. * *

The predicates are combined using AND, * and optimized in a similar way to the {@link #and} method. * If the result is TRUE no filter is created. */ public RelBuilder filter(Iterable predicates) { return filter(ImmutableSet.of(), predicates); } /** Creates a {@link Filter} of a list of correlation variables * and an array of predicates. * *

The predicates are combined using AND, * and optimized in a similar way to the {@link #and} method. * If the result is TRUE no filter is created. */ public RelBuilder filter(Iterable variablesSet, RexNode... predicates) { return filter(variablesSet, ImmutableList.copyOf(predicates)); } /** * Creates a {@link Filter} of a list of correlation variables * and a list of predicates. * *

The predicates are combined using AND, * and optimized in a similar way to the {@link #and} method. * If simplification is on and the result is TRUE, no filter is created. */ public RelBuilder filter(Iterable variablesSet, Iterable predicates) { final RexNode conjunctionPredicates; if (config.simplify()) { conjunctionPredicates = simplifier.simplifyFilterPredicates(predicates); } else { conjunctionPredicates = RexUtil.composeConjunction(simplifier.rexBuilder, predicates); } if (conjunctionPredicates == null || conjunctionPredicates.isAlwaysFalse()) { return empty(); } if (conjunctionPredicates.isAlwaysTrue()) { return this; } final Frame frame = stack.pop(); final RelNode filter = struct.filterFactory.createFilter(frame.rel, conjunctionPredicates, ImmutableSet.copyOf(variablesSet)); stack.push(new Frame(filter, frame.fields)); return this; } /** Creates a {@link Project} of the given * expressions. */ public RelBuilder project(RexNode... nodes) { return project(ImmutableList.copyOf(nodes)); } /** Creates a {@link Project} of the given list * of expressions. * *

Infers names as would {@link #project(Iterable, Iterable)} if all * suggested names were null. * * @param nodes Expressions */ public RelBuilder project(Iterable nodes) { return project(nodes, ImmutableList.of()); } /** Creates a {@link Project} of the given list * of expressions and field names. * * @param nodes Expressions * @param fieldNames field names for expressions */ public RelBuilder project(Iterable nodes, Iterable fieldNames) { return project(nodes, fieldNames, false); } /** Creates a {@link Project} of the given list * of expressions, using the given names. * *

Names are deduced as follows: *

    *
  • If the length of {@code fieldNames} is greater than the index of * the current entry in {@code nodes}, and the entry in * {@code fieldNames} is not null, uses it; otherwise *
  • If an expression projects an input field, * or is a cast an input field, * uses the input field name; otherwise *
  • If an expression is a call to * {@link SqlStdOperatorTable#AS} * (see {@link #alias}), removes the call but uses the intended alias. *
* *

After the field names have been inferred, makes the * field names unique by appending numeric suffixes. * * @param nodes Expressions * @param fieldNames Suggested field names * @param force create project even if it is identity */ public RelBuilder project(Iterable nodes, Iterable fieldNames, boolean force) { return project_(nodes, fieldNames, ImmutableList.of(), force); } /** Creates a {@link Project} of all original fields, plus the given * expressions. */ public RelBuilder projectPlus(RexNode... nodes) { return projectPlus(ImmutableList.copyOf(nodes)); } /** Creates a {@link Project} of all original fields, plus the given list of * expressions. */ public RelBuilder projectPlus(Iterable nodes) { return project(Iterables.concat(fields(), nodes)); } /** Creates a {@link Project} of all original fields, except the given * expressions. * * @throws IllegalArgumentException if the given expressions contain duplicates * or there is an expression that does not match an existing field */ public RelBuilder projectExcept(RexNode... expressions) { return projectExcept(ImmutableList.copyOf(expressions)); } /** Creates a {@link Project} of all original fields, except the given list of * expressions. * * @throws IllegalArgumentException if the given expressions contain duplicates * or there is an expression that does not match an existing field */ public RelBuilder projectExcept(Iterable expressions) { List allExpressions = new ArrayList<>(fields()); Set excludeExpressions = new HashSet<>(); for (RexNode excludeExp : expressions) { if (!excludeExpressions.add(excludeExp)) { throw new IllegalArgumentException( "Input list contains duplicates. Expression " + excludeExp + " exists multiple times."); } if (!allExpressions.remove(excludeExp)) { throw new IllegalArgumentException("Expression " + excludeExp.toString() + " not found."); } } return this.project(allExpressions); } /** Creates a {@link Project} of the given list * of expressions, using the given names. * *

Names are deduced as follows: *

    *
  • If the length of {@code fieldNames} is greater than the index of * the current entry in {@code nodes}, and the entry in * {@code fieldNames} is not null, uses it; otherwise *
  • If an expression projects an input field, * or is a cast an input field, * uses the input field name; otherwise *
  • If an expression is a call to * {@link SqlStdOperatorTable#AS} * (see {@link #alias}), removes the call but uses the intended alias. *
* *

After the field names have been inferred, makes the * field names unique by appending numeric suffixes. * * @param nodes Expressions * @param fieldNames Suggested field names * @param hints Hints * @param force create project even if it is identity */ private RelBuilder project_( Iterable nodes, Iterable fieldNames, Iterable hints, boolean force) { final Frame frame = requireNonNull(peek_(), "frame stack is empty"); final RelDataType inputRowType = frame.rel.getRowType(); final List nodeList = Lists.newArrayList(nodes); // Perform a quick check for identity. We'll do a deeper check // later when we've derived column names. if (!force && Iterables.isEmpty(fieldNames) && RexUtil.isIdentity(nodeList, inputRowType)) { return this; } final List<@Nullable String> fieldNameList = Lists.newArrayList(fieldNames); while (fieldNameList.size() < nodeList.size()) { fieldNameList.add(null); } bloat: if (frame.rel instanceof Project && config.bloat() >= 0) { final Project project = (Project) frame.rel; // Populate field names. If the upper expression is an input ref and does // not have a recommended name, use the name of the underlying field. for (int i = 0; i < fieldNameList.size(); i++) { if (fieldNameList.get(i) == null) { final RexNode node = nodeList.get(i); if (node instanceof RexInputRef) { final RexInputRef ref = (RexInputRef) node; fieldNameList.set(i, project.getRowType().getFieldNames().get(ref.getIndex())); } } } final List newNodes = RelOptUtil.pushPastProjectUnlessBloat(nodeList, project, config.bloat()); if (newNodes == null) { // The merged expression is more complex than the input expressions. // Do not merge. break bloat; } // Carefully build a list of fields, so that table aliases from the input // can be seen for fields that are based on a RexInputRef. final Frame frame1 = stack.pop(); final List fields = new ArrayList<>(); for (RelDataTypeField f : project.getInput().getRowType().getFieldList()) { fields.add(new Field(ImmutableSet.of(), f)); } for (Pair pair : Pair.zip(project.getProjects(), frame1.fields)) { switch (pair.left.getKind()) { case INPUT_REF: final int i = ((RexInputRef) pair.left).getIndex(); final Field field = fields.get(i); final ImmutableSet aliases = pair.right.left; fields.set(i, new Field(aliases, field.right)); break; default: break; } } stack.push(new Frame(project.getInput(), ImmutableList.copyOf(fields))); final ImmutableSet.Builder mergedHints = ImmutableSet.builder(); mergedHints.addAll(project.getHints()); mergedHints.addAll(hints); return project_(newNodes, fieldNameList, mergedHints.build(), force); } // Simplify expressions. if (config.simplify()) { for (int i = 0; i < nodeList.size(); i++) { nodeList.set(i, simplifier.simplifyPreservingType(nodeList.get(i))); } } // Replace null names with generated aliases. for (int i = 0; i < fieldNameList.size(); i++) { if (fieldNameList.get(i) == null) { fieldNameList.set(i, inferAlias(nodeList, nodeList.get(i), i)); } } final ImmutableList.Builder fields = ImmutableList.builder(); final Set uniqueNameList = getTypeFactory().getTypeSystem().isSchemaCaseSensitive() ? new HashSet<>() : new TreeSet<>(String.CASE_INSENSITIVE_ORDER); // calculate final names and build field list for (int i = 0; i < fieldNameList.size(); ++i) { final RexNode node = nodeList.get(i); String name = fieldNameList.get(i); String originalName = name; Field field; if (name == null || uniqueNameList.contains(name)) { int j = 0; if (name == null) { j = i; } do { name = SqlValidatorUtil.F_SUGGESTER.apply(originalName, j, j++); } while (uniqueNameList.contains(name)); fieldNameList.set(i, name); } RelDataTypeField fieldType = new RelDataTypeFieldImpl(name, i, node.getType()); switch (node.getKind()) { case INPUT_REF: // preserve rel aliases for INPUT_REF fields final int index = ((RexInputRef) node).getIndex(); field = new Field(frame.fields.get(index).left, fieldType); break; default: field = new Field(ImmutableSet.of(), fieldType); break; } uniqueNameList.add(name); fields.add(field); } if (!force && RexUtil.isIdentity(nodeList, inputRowType)) { if (fieldNameList.equals(inputRowType.getFieldNames())) { // Do not create an identity project if it does not rename any fields return this; } else { // create "virtual" row type for project only rename fields stack.pop(); // Ignore the hints. stack.push(new Frame(frame.rel, fields.build())); } return this; } // If the expressions are all literals, and the input is a Values with N // rows, replace with a Values with same tuple N times. if (config.simplifyValues() && frame.rel instanceof Values && nodeList.stream().allMatch(e -> e instanceof RexLiteral)) { final Values values = (Values) build(); final RelDataTypeFactory.Builder typeBuilder = getTypeFactory().builder(); Pair.forEach(fieldNameList, nodeList, (name, expr) -> typeBuilder.add(requireNonNull(name, "name"), expr.getType())); @SuppressWarnings({"unchecked", "rawtypes"}) final List tuple = (List) (List) nodeList; return values(Collections.nCopies(values.tuples.size(), tuple), typeBuilder.build()); } final RelNode project = struct.projectFactory.createProject(frame.rel, ImmutableList.copyOf(hints), ImmutableList.copyOf(nodeList), fieldNameList); stack.pop(); stack.push(new Frame(project, fields.build())); return this; } /** Creates a {@link Project} of the given * expressions and field names, and optionally optimizing. * *

If {@code fieldNames} is null, or if a particular entry in * {@code fieldNames} is null, derives field names from the input * expressions. * *

If {@code force} is false, * and the input is a {@code Project}, * and the expressions make the trivial projection ($0, $1, ...), * modifies the input. * * @param nodes Expressions * @param fieldNames Suggested field names, or null to generate * @param force Whether to create a renaming Project if the * projections are trivial */ public RelBuilder projectNamed(Iterable nodes, @Nullable Iterable fieldNames, boolean force) { @SuppressWarnings("unchecked") final List nodeList = nodes instanceof List ? (List) nodes : ImmutableList.copyOf(nodes); final List<@Nullable String> fieldNameList = fieldNames == null ? null : fieldNames instanceof List ? (List<@Nullable String>) fieldNames : ImmutableNullableList.copyOf(fieldNames); final RelNode input = peek(); final RelDataType rowType = RexUtil.createStructType(cluster.getTypeFactory(), nodeList, fieldNameList, SqlValidatorUtil.F_SUGGESTER); if (!force && RexUtil.isIdentity(nodeList, input.getRowType())) { if (input instanceof Project && fieldNames != null) { // Rename columns of child projection if desired field names are given. final Frame frame = stack.pop(); final Project childProject = (Project) frame.rel; final Project newInput = childProject.copy(childProject.getTraitSet(), childProject.getInput(), childProject.getProjects(), rowType); stack.push(new Frame(newInput.attachHints(childProject.getHints()), frame.fields)); } if (input instanceof Values && fieldNameList != null) { // Rename columns of child values if desired field names are given. final Frame frame = stack.pop(); final Values values = (Values) frame.rel; final RelDataTypeFactory.Builder typeBuilder = getTypeFactory().builder(); Pair.forEach(fieldNameList, rowType.getFieldList(), (name, field) -> typeBuilder.add(requireNonNull(name, "name"), field.getType())); final RelDataType newRowType = typeBuilder.build(); final RelNode newValues = struct.valuesFactory.createValues(cluster, newRowType, values.tuples); stack.push(new Frame(newValues, frame.fields)); } } else { project(nodeList, rowType.getFieldNames(), force); } return this; } /** * Creates an {@link Uncollect} with given item aliases. * * @param itemAliases Operand item aliases, never null * @param withOrdinality If {@code withOrdinality}, the output contains an extra * {@code ORDINALITY} column */ public RelBuilder uncollect(List itemAliases, boolean withOrdinality) { Frame frame = stack.pop(); stack.push( new Frame( new Uncollect( cluster, cluster.traitSetOf(Convention.NONE), frame.rel, withOrdinality, requireNonNull(itemAliases, "itemAliases")))); return this; } /** Ensures that the field names match those given. * *

If all fields have the same name, adds nothing; * if any fields do not have the same name, adds a {@link Project}. * *

Note that the names can be short-lived. Other {@code RelBuilder} * operations make no guarantees about the field names of the rows they * produce. * * @param fieldNames List of desired field names; may contain null values or * have fewer fields than the current row type */ public RelBuilder rename(List fieldNames) { final List oldFieldNames = peek().getRowType().getFieldNames(); Preconditions.checkArgument(fieldNames.size() <= oldFieldNames.size(), "More names than fields"); final List newFieldNames = new ArrayList<>(oldFieldNames); for (int i = 0; i < fieldNames.size(); i++) { final String s = fieldNames.get(i); if (s != null) { newFieldNames.set(i, s); } } if (oldFieldNames.equals(newFieldNames)) { return this; } if (peek() instanceof Values) { // Special treatment for VALUES. Re-build it rather than add a project. final Values v = (Values) build(); final RelDataTypeFactory.Builder b = getTypeFactory().builder(); for (Pair p : Pair.zip(newFieldNames, v.getRowType().getFieldList())) { b.add(p.left, p.right.getType()); } return values(v.tuples, b.build()); } return project(fields(), newFieldNames, true); } /** Infers the alias of an expression. * *

If the expression was created by {@link #alias}, replaces the expression * in the project list. */ private @Nullable String inferAlias(List exprList, RexNode expr, int i) { switch (expr.getKind()) { case INPUT_REF: final RexInputRef ref = (RexInputRef) expr; return requireNonNull(stack.peek(), "empty frame stack") .fields.get(ref.getIndex()).getValue().getName(); case CAST: return inferAlias(exprList, ((RexCall) expr).getOperands().get(0), -1); case AS: final RexCall call = (RexCall) expr; if (i >= 0) { exprList.set(i, call.getOperands().get(0)); } NlsString value = (NlsString) ((RexLiteral) call.getOperands().get(1)).getValue(); return castNonNull(value) .getValue(); default: return null; } } /** Creates an {@link Aggregate} that makes the * relational expression distinct on all fields. */ public RelBuilder distinct() { return aggregate(groupKey(fields())); } /** Creates an {@link Aggregate} with an array of * calls. */ public RelBuilder aggregate(GroupKey groupKey, AggCall... aggCalls) { return aggregate(groupKey, ImmutableList.copyOf(aggCalls)); } /** Creates an {@link Aggregate} with an array of * {@link AggregateCall}s. */ public RelBuilder aggregate(GroupKey groupKey, List aggregateCalls) { return aggregate(groupKey, aggregateCalls.stream() .map(aggregateCall -> new AggCallImpl2(aggregateCall, aggregateCall.getArgList().stream() .map(this::field) .collect(Util.toImmutableList()))) .collect(Collectors.toList())); } /** Creates an {@link Aggregate} with multiple calls. */ public RelBuilder aggregate(GroupKey groupKey, Iterable aggCalls) { final Registrar registrar = new Registrar(fields(), peek().getRowType().getFieldNames()); final GroupKeyImpl groupKey_ = (GroupKeyImpl) groupKey; ImmutableBitSet groupSet = ImmutableBitSet.of(registrar.registerExpressions(groupKey_.nodes)); label: if (Iterables.isEmpty(aggCalls)) { final RelMetadataQuery mq = peek().getCluster().getMetadataQuery(); if (groupSet.isEmpty()) { final Double minRowCount = mq.getMinRowCount(peek()); if (minRowCount == null || minRowCount < 1D) { // We can't remove "GROUP BY ()" if there's a chance the rel could be // empty. break label; } } if (registrar.extraNodes.size() == fields().size()) { final Boolean unique = mq.areColumnsUnique(peek(), groupSet); if (unique != null && unique && !config.aggregateUnique() && groupKey_.isSimple()) { // Rel is already unique. return project(fields(groupSet)); } } final Double maxRowCount = mq.getMaxRowCount(peek()); if (maxRowCount != null && maxRowCount <= 1D && !config.aggregateUnique() && groupKey_.isSimple()) { // If there is at most one row, rel is already unique. return project(fields(groupSet)); } } ImmutableList groupSets; if (groupKey_.nodeLists != null) { final int sizeBefore = registrar.extraNodes.size(); final List groupSetList = new ArrayList<>(); for (ImmutableList nodeList : groupKey_.nodeLists) { final ImmutableBitSet groupSet2 = ImmutableBitSet.of(registrar.registerExpressions(nodeList)); if (!groupSet.contains(groupSet2)) { throw new IllegalArgumentException("group set element " + nodeList + " must be a subset of group key"); } groupSetList.add(groupSet2); } final ImmutableSortedMultiset groupSetMultiset = ImmutableSortedMultiset.copyOf(ImmutableBitSet.COMPARATOR, groupSetList); if (Iterables.any(aggCalls, RelBuilder::isGroupId) || !ImmutableBitSet.ORDERING.isStrictlyOrdered(groupSetMultiset)) { return rewriteAggregateWithDuplicateGroupSets(groupSet, groupSetMultiset, ImmutableList.copyOf(aggCalls)); } groupSets = ImmutableList.copyOf(groupSetMultiset.elementSet()); if (registrar.extraNodes.size() > sizeBefore) { throw new IllegalArgumentException("group sets contained expressions " + "not in group key: " + Util.skip(registrar.extraNodes, sizeBefore)); } } else { groupSets = ImmutableList.of(groupSet); } for (AggCall aggCall : aggCalls) { ((AggCallPlus) aggCall).register(registrar); } project(registrar.extraNodes); rename(registrar.names); final Frame frame = stack.pop(); RelNode r = frame.rel; final List aggregateCalls = new ArrayList<>(); for (AggCall aggCall : aggCalls) { aggregateCalls.add( ((AggCallPlus) aggCall).aggregateCall(registrar, groupSet, r)); } assert ImmutableBitSet.ORDERING.isStrictlyOrdered(groupSets) : groupSets; for (ImmutableBitSet set : groupSets) { assert groupSet.contains(set); } List inFields = frame.fields; if (config.pruneInputOfAggregate() && r instanceof Project) { final Set fieldsUsed = RelOptUtil.getAllFields2(groupSet, aggregateCalls); // Some parts of the system can't handle rows with zero fields, so // pretend that one field is used. if (fieldsUsed.isEmpty()) { r = ((Project) r).getInput(); } else if (fieldsUsed.size() < r.getRowType().getFieldCount()) { // Some fields are computed but not used. Prune them. final Map map = new HashMap<>(); for (int source : fieldsUsed) { map.put(source, map.size()); } groupSet = groupSet.permute(map); groupSets = ImmutableBitSet.ORDERING.immutableSortedCopy( ImmutableBitSet.permute(groupSets, map)); final Mappings.TargetMapping targetMapping = Mappings.target(map, r.getRowType().getFieldCount(), fieldsUsed.size()); final List oldAggregateCalls = new ArrayList<>(aggregateCalls); aggregateCalls.clear(); for (AggregateCall aggregateCall : oldAggregateCalls) { aggregateCalls.add(aggregateCall.transform(targetMapping)); } inFields = Mappings.permute(inFields, targetMapping.inverse()); final Project project = (Project) r; final List newProjects = new ArrayList<>(); final RelDataTypeFactory.Builder builder = cluster.getTypeFactory().builder(); for (int i : fieldsUsed) { newProjects.add(project.getProjects().get(i)); builder.add(project.getRowType().getFieldList().get(i)); } r = project.copy(cluster.traitSet(), project.getInput(), newProjects, builder.build()); } } if (!config.dedupAggregateCalls() || Util.isDistinct(aggregateCalls)) { return aggregate_(groupSet, groupSets, r, aggregateCalls, registrar.extraNodes, inFields); } // There are duplicate aggregate calls. Rebuild the list to eliminate // duplicates, then add a Project. final Set callSet = new HashSet<>(); final List> projects = new ArrayList<>(); Util.range(groupSet.cardinality()) .forEach(i -> projects.add(Pair.of(i, null))); final List distinctAggregateCalls = new ArrayList<>(); for (AggregateCall aggregateCall : aggregateCalls) { final int i; if (callSet.add(aggregateCall)) { i = distinctAggregateCalls.size(); distinctAggregateCalls.add(aggregateCall); } else { i = distinctAggregateCalls.indexOf(aggregateCall); assert i >= 0; } projects.add(Pair.of(groupSet.cardinality() + i, aggregateCall.name)); } aggregate_(groupSet, groupSets, r, distinctAggregateCalls, registrar.extraNodes, inFields); final List fields = projects.stream() .map(p -> p.right == null ? field(p.left) : alias(field(p.left), p.right)) .collect(Collectors.toList()); return project(fields); } /** Finishes the implementation of {@link #aggregate} by creating an * {@link Aggregate} and pushing it onto the stack. */ private RelBuilder aggregate_(ImmutableBitSet groupSet, ImmutableList groupSets, RelNode input, List aggregateCalls, List extraNodes, List inFields) { final RelNode aggregate = struct.aggregateFactory.createAggregate(input, ImmutableList.of(), groupSet, groupSets, aggregateCalls); // build field list final ImmutableList.Builder fields = ImmutableList.builder(); final List aggregateFields = aggregate.getRowType().getFieldList(); int i = 0; // first, group fields for (Integer groupField : groupSet.asList()) { RexNode node = extraNodes.get(groupField); final SqlKind kind = node.getKind(); switch (kind) { case INPUT_REF: fields.add(inFields.get(((RexInputRef) node).getIndex())); break; default: String name = aggregateFields.get(i).getName(); RelDataTypeField fieldType = new RelDataTypeFieldImpl(name, i, node.getType()); fields.add(new Field(ImmutableSet.of(), fieldType)); break; } i++; } // second, aggregate fields. retain `i' as field index for (int j = 0; j < aggregateCalls.size(); ++j) { final AggregateCall call = aggregateCalls.get(j); final RelDataTypeField fieldType = new RelDataTypeFieldImpl(aggregateFields.get(i + j).getName(), i + j, call.getType()); fields.add(new Field(ImmutableSet.of(), fieldType)); } stack.push(new Frame(aggregate, fields.build())); return this; } /** * The {@code GROUP_ID()} function is used to distinguish duplicate groups. * However, as Aggregate normalizes group sets to canonical form (i.e., * flatten, sorting, redundancy removal), this information is lost in RelNode. * Therefore, it is impossible to implement the function in runtime. * *

To fill this gap, an aggregation query that contains duplicate group * sets is rewritten into a Union of Aggregate operators whose group sets are * distinct. The number of inputs to the Union is equal to the maximum number * of duplicates. In the {@code N}th input to the Union, calls to the * {@code GROUP_ID} aggregate function are replaced by the integer literal * {@code N}. * *

This method also handles the case where group sets are distinct but * there is a call to {@code GROUP_ID}. That call is replaced by the integer * literal {@code 0}. * *

Also see the discussion in * [CALCITE-1824] * GROUP_ID returns wrong result and * [CALCITE-4748] * If there are duplicate GROUPING SETS, Calcite should return duplicate * rows. */ private RelBuilder rewriteAggregateWithDuplicateGroupSets( ImmutableBitSet groupSet, ImmutableSortedMultiset groupSets, List aggregateCalls) { final List fieldNamesIfNoRewrite = Aggregate.deriveRowType(getTypeFactory(), peek().getRowType(), false, groupSet, groupSets.asList(), aggregateCalls.stream().map(c -> ((AggCallPlus) c).aggregateCall()) .collect(Util.toImmutableList())).getFieldNames(); // If n duplicates exist for a particular grouping, the {@code GROUP_ID()} // function produces values in the range 0 to n-1. For each value, // we need to figure out the corresponding group sets. // // For example, "... GROUPING SETS (a, a, b, c, c, c, c)" // (i) The max value of the GROUP_ID() function returns is 3 // (ii) GROUPING SETS (a, b, c) produces value 0, // GROUPING SETS (a, c) produces value 1, // GROUPING SETS (c) produces value 2 // GROUPING SETS (c) produces value 3 final Map> groupIdToGroupSets = new HashMap<>(); int maxGroupId = 0; for (Multiset.Entry entry: groupSets.entrySet()) { int groupId = entry.getCount() - 1; if (groupId > maxGroupId) { maxGroupId = groupId; } for (int i = 0; i <= groupId; i++) { groupIdToGroupSets.computeIfAbsent(i, k -> Sets.newTreeSet(ImmutableBitSet.COMPARATOR)) .add(entry.getElement()); } } // AggregateCall list without GROUP_ID function final List aggregateCallsWithoutGroupId = new ArrayList<>(aggregateCalls); aggregateCallsWithoutGroupId.removeIf(RelBuilder::isGroupId); // For each group id value, we first construct an Aggregate without // GROUP_ID() function call, and then create a Project node on top of it. // The Project adds literal value for group id in right position. final Frame frame = stack.pop(); for (int groupId = 0; groupId <= maxGroupId; groupId++) { // Create the Aggregate node without GROUP_ID() call stack.push(frame); aggregate(groupKey(groupSet, castNonNull(groupIdToGroupSets.get(groupId))), aggregateCallsWithoutGroupId); final List selectList = new ArrayList<>(); final int groupExprLength = groupSet.cardinality(); // Project fields in group by expressions for (int i = 0; i < groupExprLength; i++) { selectList.add(field(i)); } // Project fields in aggregate calls int groupIdCount = 0; for (int i = 0; i < aggregateCalls.size(); i++) { if (isGroupId(aggregateCalls.get(i))) { selectList.add( getRexBuilder().makeExactLiteral(BigDecimal.valueOf(groupId), getTypeFactory().createSqlType(SqlTypeName.BIGINT))); groupIdCount++; } else { selectList.add(field(groupExprLength + i - groupIdCount)); } } project(selectList, fieldNamesIfNoRewrite); } return union(true, maxGroupId + 1); } private static boolean isGroupId(AggCall c) { return ((AggCallPlus) c).op().kind == SqlKind.GROUP_ID; } private RelBuilder setOp(boolean all, SqlKind kind, int n) { List inputs = new ArrayList<>(); for (int i = 0; i < n; i++) { inputs.add(0, build()); } switch (kind) { case UNION: case INTERSECT: case EXCEPT: if (n < 1) { throw new IllegalArgumentException( "bad INTERSECT/UNION/EXCEPT input count"); } break; default: throw new AssertionError("bad setOp " + kind); } if (n == 1) { return push(inputs.get(0)); } if (config.simplifyValues() && kind == UNION && inputs.stream().allMatch(r -> r instanceof Values)) { List inputTypes = Util.transform(inputs, RelNode::getRowType); RelDataType rowType = getTypeFactory() .leastRestrictive(inputTypes); requireNonNull(rowType, () -> "leastRestrictive(" + inputTypes + ")"); final List> tuples = new ArrayList<>(); for (RelNode input : inputs) { tuples.addAll(((Values) input).tuples); } final List> tuples2 = all ? tuples : Util.distinctList(tuples); return values(tuples2, rowType); } return push(struct.setOpFactory.createSetOp(kind, inputs, all)); } /** Creates a {@link Union} of the two most recent * relational expressions on the stack. * * @param all Whether to create UNION ALL */ public RelBuilder union(boolean all) { return union(all, 2); } /** Creates a {@link Union} of the {@code n} * most recent relational expressions on the stack. * * @param all Whether to create UNION ALL * @param n Number of inputs to the UNION operator */ public RelBuilder union(boolean all, int n) { return setOp(all, UNION, n); } /** Creates an {@link Intersect} of the two most * recent relational expressions on the stack. * * @param all Whether to create INTERSECT ALL */ public RelBuilder intersect(boolean all) { return intersect(all, 2); } /** Creates an {@link Intersect} of the {@code n} * most recent relational expressions on the stack. * * @param all Whether to create INTERSECT ALL * @param n Number of inputs to the INTERSECT operator */ public RelBuilder intersect(boolean all, int n) { return setOp(all, SqlKind.INTERSECT, n); } /** Creates a {@link Minus} of the two most recent * relational expressions on the stack. * * @param all Whether to create EXCEPT ALL */ public RelBuilder minus(boolean all) { return minus(all, 2); } /** Creates a {@link Minus} of the {@code n} * most recent relational expressions on the stack. * * @param all Whether to create EXCEPT ALL */ public RelBuilder minus(boolean all, int n) { return setOp(all, SqlKind.EXCEPT, n); } /** * Creates a {@link TableScan} on a {@link TransientTable} with the given name, using as type * the top of the stack's type. * * @param tableName table name */ @Experimental public RelBuilder transientScan(String tableName) { return this.transientScan(tableName, this.peek().getRowType()); } /** * Creates a {@link TableScan} on a {@link TransientTable} with the given name and type. * * @param tableName table name * @param rowType row type of the table */ @Experimental public RelBuilder transientScan(String tableName, RelDataType rowType) { TransientTable transientTable = new ListTransientTable(tableName, rowType); requireNonNull(relOptSchema, "relOptSchema"); RelOptTable relOptTable = RelOptTableImpl.create( relOptSchema, rowType, transientTable, ImmutableList.of(tableName)); RelNode scan = struct.scanFactory.createScan( ViewExpanders.toRelContext(viewExpander, cluster), relOptTable); push(scan); rename(rowType.getFieldNames()); return this; } /** * Creates a {@link TableSpool} for the most recent relational expression. * * @param readType Spool's read type (as described in {@link Spool.Type}) * @param writeType Spool's write type (as described in {@link Spool.Type}) * @param table Table to write into */ private RelBuilder tableSpool(Spool.Type readType, Spool.Type writeType, RelOptTable table) { RelNode spool = struct.spoolFactory.createTableSpool(peek(), readType, writeType, table); replaceTop(spool); return this; } /** * Creates a {@link RepeatUnion} associated to a {@link TransientTable} without a maximum number * of iterations, i.e. repeatUnion(tableName, all, -1). * * @param tableName name of the {@link TransientTable} associated to the {@link RepeatUnion} * @param all whether duplicates will be considered or not */ @Experimental public RelBuilder repeatUnion(String tableName, boolean all) { return repeatUnion(tableName, all, -1); } /** * Creates a {@link RepeatUnion} associated to a {@link TransientTable} of the * two most recent relational expressions on the stack. * *

Warning: if these relational expressions are not * correctly defined, this operation might lead to an infinite loop. * *

The generated {@link RepeatUnion} operates as follows: * *

    *
  • Evaluate its left term once, propagating the results into the * {@link TransientTable}; *
  • Evaluate its right term (which may contain a {@link TableScan} on the * {@link TransientTable}) over and over until it produces no more results * (or until an optional maximum number of iterations is reached). On each * iteration, the results are propagated into the {@link TransientTable}, * overwriting the results from the previous one. *
* * @param tableName Name of the {@link TransientTable} associated to the * {@link RepeatUnion} * @param all Whether duplicates are considered * @param iterationLimit Maximum number of iterations; negative value means no limit */ @Experimental public RelBuilder repeatUnion(String tableName, boolean all, int iterationLimit) { RelOptTableFinder finder = new RelOptTableFinder(tableName); for (int i = 0; i < stack.size(); i++) { // search scan(tableName) in the stack peek(i).accept(finder); if (finder.relOptTable != null) { // found break; } } if (finder.relOptTable == null) { throw RESOURCE.tableNotFound(tableName).ex(); } RelNode iterative = tableSpool(Spool.Type.LAZY, Spool.Type.LAZY, finder.relOptTable).build(); RelNode seed = tableSpool(Spool.Type.LAZY, Spool.Type.LAZY, finder.relOptTable).build(); RelNode repeatUnion = struct.repeatUnionFactory.createRepeatUnion(seed, iterative, all, iterationLimit, finder.relOptTable); return push(repeatUnion); } /** * Auxiliary class to find a certain RelOptTable based on its name. */ private static final class RelOptTableFinder extends RelHomogeneousShuttle { private @MonotonicNonNull RelOptTable relOptTable = null; private final String tableName; private RelOptTableFinder(String tableName) { this.tableName = tableName; } @Override public RelNode visit(TableScan scan) { final RelOptTable scanTable = scan.getTable(); final List qualifiedName = scanTable.getQualifiedName(); if (qualifiedName.get(qualifiedName.size() - 1).equals(tableName)) { relOptTable = scanTable; } return super.visit(scan); } } /** Creates a {@link Join} with an array of conditions. */ public RelBuilder join(JoinRelType joinType, RexNode condition0, RexNode... conditions) { return join(joinType, Lists.asList(condition0, conditions)); } /** Creates a {@link Join} with multiple * conditions. */ public RelBuilder join(JoinRelType joinType, Iterable conditions) { return join(joinType, and(conditions), ImmutableSet.of()); } /** Creates a {@link Join} with one condition. */ public RelBuilder join(JoinRelType joinType, RexNode condition) { return join(joinType, condition, ImmutableSet.of()); } /** Creates a {@link Join} with correlating variables. */ public RelBuilder join(JoinRelType joinType, RexNode condition, Set variablesSet) { Frame right = stack.pop(); final Frame left = stack.pop(); final RelNode join; final boolean correlate = checkIfCorrelated(variablesSet, joinType, left.rel, right.rel); RexNode postCondition = literal(true); if (config.simplify()) { // Normalize expanded versions IS NOT DISTINCT FROM so that simplifier does not // transform the expression to something unrecognizable if (condition instanceof RexCall) { condition = RelOptUtil.collapseExpandedIsNotDistinctFromExpr((RexCall) condition, getRexBuilder()); } condition = simplifier.simplifyUnknownAsFalse(condition); } if (correlate) { final CorrelationId id = Iterables.getOnlyElement(variablesSet); // Correlate does not have an ON clause. switch (joinType) { case LEFT: case SEMI: case ANTI: // For a LEFT/SEMI/ANTI, predicate must be evaluated first. stack.push(right); filter(condition.accept(new Shifter(left.rel, id, right.rel))); right = stack.pop(); break; case INNER: // For INNER, we can defer. postCondition = condition; break; default: throw new IllegalArgumentException("Correlated " + joinType + " join is not supported"); } final ImmutableBitSet requiredColumns = RelOptUtil.correlationColumns(id, right.rel); join = struct.correlateFactory.createCorrelate(left.rel, right.rel, ImmutableList.of(), id, requiredColumns, joinType); } else { RelNode join0 = struct.joinFactory.createJoin(left.rel, right.rel, ImmutableList.of(), condition, variablesSet, joinType, false); if (join0 instanceof Join && config.pushJoinCondition()) { join = RelOptUtil.pushDownJoinConditions((Join) join0, this); } else { join = join0; } } final ImmutableList.Builder fields = ImmutableList.builder(); fields.addAll(left.fields); fields.addAll(right.fields); stack.push(new Frame(join, fields.build())); filter(postCondition); return this; } /** Creates a {@link Correlate} * with a {@link CorrelationId} and an array of fields that are used by correlation. */ public RelBuilder correlate(JoinRelType joinType, CorrelationId correlationId, RexNode... requiredFields) { return correlate(joinType, correlationId, ImmutableList.copyOf(requiredFields)); } /** Creates a {@link Correlate} * with a {@link CorrelationId} and a list of fields that are used by correlation. */ public RelBuilder correlate(JoinRelType joinType, CorrelationId correlationId, Iterable requiredFields) { Frame right = stack.pop(); final Registrar registrar = new Registrar(fields(), peek().getRowType().getFieldNames()); List requiredOrdinals = registrar.registerExpressions(ImmutableList.copyOf(requiredFields)); project(registrar.extraNodes); rename(registrar.names); Frame left = stack.pop(); final RelNode correlate = struct.correlateFactory.createCorrelate(left.rel, right.rel, ImmutableList.of(), correlationId, ImmutableBitSet.of(requiredOrdinals), joinType); final ImmutableList.Builder fields = ImmutableList.builder(); fields.addAll(left.fields); fields.addAll(right.fields); stack.push(new Frame(correlate, fields.build())); return this; } /** Creates a {@link Join} using USING syntax. * *

For each of the field names, both left and right inputs must have a * field of that name. Constructs a join condition that the left and right * fields are equal. * * @param joinType Join type * @param fieldNames Field names */ public RelBuilder join(JoinRelType joinType, String... fieldNames) { final List conditions = new ArrayList<>(); for (String fieldName : fieldNames) { conditions.add( equals(field(2, 0, fieldName), field(2, 1, fieldName))); } return join(joinType, conditions); } /** Creates a {@link Join} with {@link JoinRelType#SEMI}. * *

A semi-join is a form of join that combines two relational expressions * according to some condition, and outputs only rows from the left input for * which at least one row from the right input matches. It only outputs * columns from the left input, and ignores duplicates on the right. * *

For example, {@code EMP semi-join DEPT} finds all {@code EMP} records * that do not have a corresponding {@code DEPT} record, similar to the * following SQL: * *

   * SELECT * FROM EMP
   * WHERE EXISTS (SELECT 1 FROM DEPT
   *     WHERE DEPT.DEPTNO = EMP.DEPTNO)
*
*/ public RelBuilder semiJoin(Iterable conditions) { final Frame right = stack.pop(); final RelNode semiJoin = struct.joinFactory.createJoin(peek(), right.rel, ImmutableList.of(), and(conditions), ImmutableSet.of(), JoinRelType.SEMI, false); replaceTop(semiJoin); return this; } /** Creates a {@link Join} with {@link JoinRelType#SEMI}. * * @see #semiJoin(Iterable) */ public RelBuilder semiJoin(RexNode... conditions) { return semiJoin(ImmutableList.copyOf(conditions)); } /** Creates an anti-join. * *

An anti-join is a form of join that combines two relational expressions * according to some condition, but outputs only rows from the left input * for which no rows from the right input match. * *

For example, {@code EMP anti-join DEPT} finds all {@code EMP} records * that do not have a corresponding {@code DEPT} record, similar to the * following SQL: * *

   * SELECT * FROM EMP
   * WHERE NOT EXISTS (SELECT 1 FROM DEPT
   *     WHERE DEPT.DEPTNO = EMP.DEPTNO)
*
*/ public RelBuilder antiJoin(Iterable conditions) { final Frame right = stack.pop(); final RelNode antiJoin = struct.joinFactory.createJoin(peek(), right.rel, ImmutableList.of(), and(conditions), ImmutableSet.of(), JoinRelType.ANTI, false); replaceTop(antiJoin); return this; } /** Creates an anti-join. * * @see #antiJoin(Iterable) */ public RelBuilder antiJoin(RexNode... conditions) { return antiJoin(ImmutableList.copyOf(conditions)); } /** Assigns a table alias to the top entry on the stack. */ public RelBuilder as(final String alias) { final Frame pair = stack.pop(); List newFields = Util.transform(pair.fields, field -> field.addAlias(alias)); stack.push(new Frame(pair.rel, ImmutableList.copyOf(newFields))); return this; } /** Creates a {@link Values}. * *

The {@code values} array must have the same number of entries as * {@code fieldNames}, or an integer multiple if you wish to create multiple * rows. * *

If there are zero rows, or if all values of a any column are * null, this method cannot deduce the type of columns. For these cases, * call {@link #values(Iterable, RelDataType)}. * * @param fieldNames Field names * @param values Values */ public RelBuilder values(@Nullable String[] fieldNames, @Nullable Object... values) { if (fieldNames == null || fieldNames.length == 0 || values.length % fieldNames.length != 0 || values.length < fieldNames.length) { throw new IllegalArgumentException( "Value count must be a positive multiple of field count"); } final int rowCount = values.length / fieldNames.length; for (Ord<@Nullable String> fieldName : Ord.zip(fieldNames)) { if (allNull(values, fieldName.i, fieldNames.length)) { throw new IllegalArgumentException("All values of field '" + fieldName.e + "' (field index " + fieldName.i + ")" + " are null; cannot deduce type"); } } final ImmutableList> tupleList = tupleList(fieldNames.length, values); assert tupleList.size() == rowCount; final List fieldNameList = Util.transformIndexed(Arrays.asList(fieldNames), (name, i) -> name != null ? name : SqlUtil.deriveAliasFromOrdinal(i)); return values(tupleList, fieldNameList); } private RelBuilder values(List> tupleList, List fieldNames) { final RelDataTypeFactory typeFactory = cluster.getTypeFactory(); final RelDataTypeFactory.Builder builder = typeFactory.builder(); Ord.forEach(fieldNames, (fieldName, i) -> { final RelDataType type = typeFactory.leastRestrictive( new AbstractList() { @Override public RelDataType get(int index) { return tupleList.get(index).get(i).getType(); } @Override public int size() { return tupleList.size(); } }); assert type != null : "can't infer type for field " + i + ", " + fieldName; builder.add(fieldName, type); }); final RelDataType rowType = builder.build(); return values(tupleList, rowType); } private ImmutableList> tupleList(int columnCount, @Nullable Object[] values) { final ImmutableList.Builder> listBuilder = ImmutableList.builder(); final List valueList = new ArrayList<>(); for (int i = 0; i < values.length; i++) { Object value = values[i]; valueList.add(literal(value)); if ((i + 1) % columnCount == 0) { listBuilder.add(ImmutableList.copyOf(valueList)); valueList.clear(); } } return listBuilder.build(); } /** Returns whether all values for a given column are null. */ private static boolean allNull(@Nullable Object[] values, int column, int columnCount) { for (int i = column; i < values.length; i += columnCount) { if (values[i] != null) { return false; } } return true; } /** Creates a relational expression that reads from an input and throws * all of the rows away. * *

Note that this method always pops one relational expression from the * stack. {@code values}, in contrast, does not pop any relational * expressions, and always produces a leaf. * *

The default implementation creates a {@link Values} with the same * specified row type and aliases as the input, and ignores the input entirely. * But schema-on-query systems such as Drill might override this method to * create a relation expression that retains the input, just to read its * schema. */ public RelBuilder empty() { final Frame frame = stack.pop(); final RelNode values = struct.valuesFactory.createValues(cluster, frame.rel.getRowType(), ImmutableList.of()); stack.push(new Frame(values, frame.fields)); return this; } /** Creates a {@link Values} with a specified row type. * *

This method can handle cases that {@link #values(String[], Object...)} * cannot, such as all values of a column being null, or there being zero * rows. * * @param rowType Row type * @param columnValues Values */ public RelBuilder values(RelDataType rowType, Object... columnValues) { final ImmutableList> tupleList = tupleList(rowType.getFieldCount(), columnValues); RelNode values = struct.valuesFactory.createValues(cluster, rowType, ImmutableList.copyOf(tupleList)); push(values); return this; } /** Creates a {@link Values} with a specified row type. * *

This method can handle cases that {@link #values(String[], Object...)} * cannot, such as all values of a column being null, or there being zero * rows. * * @param tupleList Tuple list * @param rowType Row type */ public RelBuilder values(Iterable> tupleList, RelDataType rowType) { RelNode values = struct.valuesFactory.createValues(cluster, rowType, copy(tupleList)); push(values); return this; } /** Creates a {@link Values} with a specified row type and * zero rows. * * @param rowType Row type */ public RelBuilder values(RelDataType rowType) { return values(ImmutableList.>of(), rowType); } /** Converts an iterable of lists into an immutable list of immutable lists * with the same contents. Returns the same object if possible. */ private static ImmutableList> copy( Iterable> tupleList) { final ImmutableList.Builder> builder = ImmutableList.builder(); int changeCount = 0; for (List literals : tupleList) { final ImmutableList literals2 = ImmutableList.copyOf(literals); builder.add(literals2); if (literals != literals2) { ++changeCount; } } if (changeCount == 0 && tupleList instanceof ImmutableList) { // don't make a copy if we don't have to //noinspection unchecked return (ImmutableList>) tupleList; } return builder.build(); } /** Creates a limit without a sort. */ public RelBuilder limit(int offset, int fetch) { return sortLimit(offset, fetch, ImmutableList.of()); } /** Creates an Exchange by distribution. */ public RelBuilder exchange(RelDistribution distribution) { RelNode exchange = struct.exchangeFactory.createExchange(peek(), distribution); replaceTop(exchange); return this; } /** Creates a SortExchange by distribution and collation. */ public RelBuilder sortExchange(RelDistribution distribution, RelCollation collation) { RelNode exchange = struct.sortExchangeFactory.createSortExchange(peek(), distribution, collation); replaceTop(exchange); return this; } /** Creates a {@link Sort} by field ordinals. * *

Negative fields mean descending: -1 means field(0) descending, * -2 means field(1) descending, etc. */ public RelBuilder sort(int... fields) { final ImmutableList.Builder builder = ImmutableList.builder(); for (int field : fields) { builder.add(field < 0 ? desc(field(-field - 1)) : field(field)); } return sortLimit(-1, -1, builder.build()); } /** Creates a {@link Sort} by expressions. */ public RelBuilder sort(RexNode... nodes) { return sortLimit(-1, -1, ImmutableList.copyOf(nodes)); } /** Creates a {@link Sort} by expressions. */ public RelBuilder sort(Iterable nodes) { return sortLimit(-1, -1, nodes); } /** Creates a {@link Sort} by expressions, with limit and offset. */ public RelBuilder sortLimit(int offset, int fetch, RexNode... nodes) { return sortLimit(offset, fetch, ImmutableList.copyOf(nodes)); } /** Creates a {@link Sort} by specifying collations. */ public RelBuilder sort(RelCollation collation) { final RelNode sort = struct.sortFactory.createSort(peek(), collation, null, null); replaceTop(sort); return this; } /** Creates a {@link Sort} by a list of expressions, with limit and offset. * * @param offset Number of rows to skip; non-positive means don't skip any * @param fetch Maximum number of rows to fetch; negative means no limit * @param nodes Sort expressions */ public RelBuilder sortLimit(int offset, int fetch, Iterable nodes) { final @Nullable RexNode offsetNode = offset <= 0 ? null : literal(offset); final @Nullable RexNode fetchNode = fetch < 0 ? null : literal(fetch); return sortLimit(offsetNode, fetchNode, nodes); } /** Creates a {@link Sort} by a list of expressions, with limitNode and offsetNode. * * @param offsetNode RexLiteral means number of rows to skip is deterministic, * RexDynamicParam means number of rows to skip is dynamic. * @param fetchNode RexLiteral means maximum number of rows to fetch is deterministic, * RexDynamicParam mean maximum number is dynamic. * @param nodes Sort expressions */ public RelBuilder sortLimit(@Nullable RexNode offsetNode, @Nullable RexNode fetchNode, Iterable nodes) { if (offsetNode != null) { if (!(offsetNode instanceof RexLiteral || offsetNode instanceof RexDynamicParam)) { throw new IllegalArgumentException("OFFSET node must be RexLiteral or RexDynamicParam"); } } if (fetchNode != null) { if (!(fetchNode instanceof RexLiteral || fetchNode instanceof RexDynamicParam)) { throw new IllegalArgumentException("FETCH node must be RexLiteral or RexDynamicParam"); } } final Registrar registrar = new Registrar(fields(), ImmutableList.of()); final List fieldCollations = registrar.registerFieldCollations(nodes); final int fetch = fetchNode instanceof RexLiteral ? RexLiteral.intValue(fetchNode) : -1; if (offsetNode == null && fetch == 0 && config.simplifyLimit()) { return empty(); } if (offsetNode == null && fetchNode == null && fieldCollations.isEmpty()) { return this; // sort is trivial } if (fieldCollations.isEmpty()) { assert registrar.addedFieldCount() == 0; RelNode top = peek(); if (top instanceof Sort) { final Sort sort2 = (Sort) top; if (sort2.offset == null && sort2.fetch == null) { replaceTop(sort2.getInput()); final RelNode sort = struct.sortFactory.createSort(peek(), sort2.collation, offsetNode, fetchNode); replaceTop(sort); return this; } } if (top instanceof Project) { final Project project = (Project) top; if (project.getInput() instanceof Sort) { final Sort sort2 = (Sort) project.getInput(); if (sort2.offset == null && sort2.fetch == null) { final RelNode sort = struct.sortFactory.createSort(sort2.getInput(), sort2.collation, offsetNode, fetchNode); replaceTop( struct.projectFactory.createProject(sort, project.getHints(), project.getProjects(), Pair.right(project.getNamedProjects()))); return this; } } } } if (registrar.addedFieldCount() > 0) { project(registrar.extraNodes); } final RelNode sort = struct.sortFactory.createSort(peek(), RelCollations.of(fieldCollations), offsetNode, fetchNode); replaceTop(sort); if (registrar.addedFieldCount() > 0) { project(registrar.originalExtraNodes); } return this; } private static RelFieldCollation collation(RexNode node, RelFieldCollation.Direction direction, RelFieldCollation.@Nullable NullDirection nullDirection, List extraNodes) { switch (node.getKind()) { case INPUT_REF: return new RelFieldCollation(((RexInputRef) node).getIndex(), direction, Util.first(nullDirection, direction.defaultNullDirection())); case DESCENDING: return collation(((RexCall) node).getOperands().get(0), RelFieldCollation.Direction.DESCENDING, nullDirection, extraNodes); case NULLS_FIRST: return collation(((RexCall) node).getOperands().get(0), direction, RelFieldCollation.NullDirection.FIRST, extraNodes); case NULLS_LAST: return collation(((RexCall) node).getOperands().get(0), direction, RelFieldCollation.NullDirection.LAST, extraNodes); default: final int fieldIndex = extraNodes.size(); extraNodes.add(node); return new RelFieldCollation(fieldIndex, direction, Util.first(nullDirection, direction.defaultNullDirection())); } } private static RexFieldCollation rexCollation(RexNode node, RelFieldCollation.Direction direction, RelFieldCollation.@Nullable NullDirection nullDirection) { switch (node.getKind()) { case DESCENDING: return rexCollation(((RexCall) node).operands.get(0), RelFieldCollation.Direction.DESCENDING, nullDirection); case NULLS_LAST: return rexCollation(((RexCall) node).operands.get(0), direction, RelFieldCollation.NullDirection.LAST); case NULLS_FIRST: return rexCollation(((RexCall) node).operands.get(0), direction, RelFieldCollation.NullDirection.FIRST); default: final Set flags = EnumSet.noneOf(SqlKind.class); if (direction == RelFieldCollation.Direction.DESCENDING) { flags.add(SqlKind.DESCENDING); } if (nullDirection == RelFieldCollation.NullDirection.FIRST) { flags.add(SqlKind.NULLS_FIRST); } if (nullDirection == RelFieldCollation.NullDirection.LAST) { flags.add(SqlKind.NULLS_LAST); } return new RexFieldCollation(node, flags); } } /** * Creates a projection that converts the current relational expression's * output to a desired row type. * *

The desired row type and the row type to be converted must have the * same number of fields. * * @param castRowType row type after cast * @param rename if true, use field names from castRowType; if false, * preserve field names from rel */ public RelBuilder convert(RelDataType castRowType, boolean rename) { final RelNode r = build(); final RelNode r2 = RelOptUtil.createCastRel(r, castRowType, rename, struct.projectFactory); push(r2); return this; } public RelBuilder permute(Mapping mapping) { assert mapping.getMappingType().isSingleSource(); assert mapping.getMappingType().isMandatorySource(); if (mapping.isIdentity()) { return this; } final List exprList = new ArrayList<>(); for (int i = 0; i < mapping.getTargetCount(); i++) { exprList.add(field(mapping.getSource(i))); } return project(exprList); } /** Creates a {@link Match}. */ public RelBuilder match(RexNode pattern, boolean strictStart, boolean strictEnd, Map patternDefinitions, Iterable measureList, RexNode after, Map> subsets, boolean allRows, Iterable partitionKeys, Iterable orderKeys, RexNode interval) { final Registrar registrar = new Registrar(fields(), peek().getRowType().getFieldNames()); final List fieldCollations = registrar.registerFieldCollations(orderKeys); final ImmutableBitSet partitionBitSet = ImmutableBitSet.of(registrar.registerExpressions(partitionKeys)); final RelDataTypeFactory.Builder typeBuilder = cluster.getTypeFactory().builder(); for (RexNode partitionKey : partitionKeys) { typeBuilder.add(partitionKey.toString(), partitionKey.getType()); } if (allRows) { for (RexNode orderKey : orderKeys) { if (!typeBuilder.nameExists(orderKey.toString())) { typeBuilder.add(orderKey.toString(), orderKey.getType()); } } final RelDataType inputRowType = peek().getRowType(); for (RelDataTypeField fs : inputRowType.getFieldList()) { if (!typeBuilder.nameExists(fs.getName())) { typeBuilder.add(fs); } } } final ImmutableMap.Builder measures = ImmutableMap.builder(); for (RexNode measure : measureList) { List operands = ((RexCall) measure).getOperands(); String alias = operands.get(1).toString(); typeBuilder.add(alias, operands.get(0).getType()); measures.put(alias, operands.get(0)); } final RelNode match = struct.matchFactory.createMatch(peek(), pattern, typeBuilder.build(), strictStart, strictEnd, patternDefinitions, measures.build(), after, subsets, allRows, partitionBitSet, RelCollations.of(fieldCollations), interval); stack.push(new Frame(match)); return this; } /** Creates a Pivot. * *

To achieve the same effect as the SQL * *

{@code
   * SELECT *
   * FROM (SELECT mgr, deptno, job, sal FROM emp)
   * PIVOT (SUM(sal) AS ss, COUNT(*) AS c
   *     FOR (job, deptno)
   *     IN (('CLERK', 10) AS c10, ('MANAGER', 20) AS m20))
   * }
* *

use the builder as follows: * *

{@code
   * RelBuilder b;
   * b.scan("EMP");
   * final RelBuilder.GroupKey groupKey = b.groupKey("MGR");
   * final List aggCalls =
   *     Arrays.asList(b.sum(b.field("SAL")).as("SS"),
   *         b.count().as("C"));
   * final List axes =
   *     Arrays.asList(b.field("JOB"),
   *         b.field("DEPTNO"));
   * final ImmutableMap.Builder> valueMap =
   *     ImmutableMap.builder();
   * valueMap.put("C10",
   *     Arrays.asList(b.literal("CLERK"), b.literal(10)));
   * valueMap.put("M20",
   *     Arrays.asList(b.literal("MANAGER"), b.literal(20)));
   * b.pivot(groupKey, aggCalls, axes, valueMap.build().entrySet());
   * }
* *

Note that the SQL uses a sub-query to project away columns (e.g. * {@code HIREDATE}) that it does not reference, so that they do not appear in * the {@code GROUP BY}. You do not need to do that in this API, because the * {@code groupKey} parameter specifies the keys. * *

Pivot is implemented by desugaring. The above example becomes the * following: * *

{@code
   * SELECT mgr,
   *     SUM(sal) FILTER (WHERE job = 'CLERK' AND deptno = 10) AS c10_ss,
   *     COUNT(*) FILTER (WHERE job = 'CLERK' AND deptno = 10) AS c10_c,
   *     SUM(sal) FILTER (WHERE job = 'MANAGER' AND deptno = 20) AS m20_ss,
   *      COUNT(*) FILTER (WHERE job = 'MANAGER' AND deptno = 20) AS m20_c
   * FROM emp
   * GROUP BY mgr
   * }
* * @param groupKey Key columns * @param aggCalls Aggregate expressions to compute for each value * @param axes Columns to pivot * @param values Values to pivot, and the alias for each column group * * @return this RelBuilder */ public RelBuilder pivot(GroupKey groupKey, Iterable aggCalls, Iterable axes, Iterable>> values) { final List axisList = ImmutableList.copyOf(axes); final List multipliedAggCalls = new ArrayList<>(); Pair.forEach(values, (alias, expressions) -> { final List expressionList = ImmutableList.copyOf(expressions); if (expressionList.size() != axisList.size()) { throw new IllegalArgumentException("value count must match axis count [" + expressionList + "], [" + axisList + "]"); } aggCalls.forEach(aggCall -> { final String alias2 = alias + "_" + ((AggCallPlus) aggCall).alias(); final List filters = new ArrayList<>(); Pair.forEach(axisList, expressionList, (axis, expression) -> filters.add(equals(axis, expression))); multipliedAggCalls.add(aggCall.filter(and(filters)).as(alias2)); }); }); return aggregate(groupKey, multipliedAggCalls); } /** * Creates an Unpivot. * *

To achieve the same effect as the SQL * *

{@code
   * SELECT *
   * FROM (SELECT deptno, job, sal, comm FROM emp)
   *   UNPIVOT INCLUDE NULLS (remuneration
   *     FOR remuneration_type IN (comm AS 'commission',
   *                               sal AS 'salary'))
   * }
* *

use the builder as follows: * *

{@code
   * RelBuilder b;
   * b.scan("EMP");
   * final List measureNames = Arrays.asList("REMUNERATION");
   * final List axisNames = Arrays.asList("REMUNERATION_TYPE");
   * final Map, List> axisMap =
   *     ImmutableMap., List>builder()
   *         .put(Arrays.asList(b.literal("commission")),
   *             Arrays.asList(b.field("COMM")))
   *         .put(Arrays.asList(b.literal("salary")),
   *             Arrays.asList(b.field("SAL")))
   *         .build();
   * b.unpivot(false, measureNames, axisNames, axisMap);
   * }
* *

The query generates two columns: {@code remuneration_type} (an axis * column) and {@code remuneration} (a measure column). Axis columns contain * values to indicate the source of the row (in this case, {@code 'salary'} * if the row came from the {@code sal} column, and {@code 'commission'} * if the row came from the {@code comm} column). * * @param includeNulls Whether to include NULL values in the output * @param measureNames Names of columns to be generated to hold pivoted * measures * @param axisNames Names of columns to be generated to hold qualifying values * @param axisMap Mapping from the columns that hold measures to the values * that the axis columns will hold in the generated rows * @return This RelBuilder */ public RelBuilder unpivot(boolean includeNulls, Iterable measureNames, Iterable axisNames, Iterable, ? extends List>> axisMap) { // Make immutable copies of all arguments. final List measureNameList = ImmutableList.copyOf(measureNames); final List axisNameList = ImmutableList.copyOf(axisNames); final List, List>> map = StreamSupport.stream(axisMap.spliterator(), false) .map(pair -> Pair., List>of( ImmutableList.copyOf(pair.getKey()), ImmutableList.copyOf(pair.getValue()))) .collect(Util.toImmutableList()); // Check that counts match. Pair.forEach(map, (valueList, inputMeasureList) -> { if (inputMeasureList.size() != measureNameList.size()) { throw new IllegalArgumentException("Number of measures (" + inputMeasureList.size() + ") must match number of measure names (" + measureNameList.size() + ")"); } if (valueList.size() != axisNameList.size()) { throw new IllegalArgumentException("Number of axis values (" + valueList.size() + ") match match number of axis names (" + axisNameList.size() + ")"); } }); final RelDataType leftRowType = peek().getRowType(); final BitSet usedFields = new BitSet(); Pair.forEach(map, (aliases, nodes) -> nodes.forEach(node -> { if (node instanceof RexInputRef) { usedFields.set(((RexInputRef) node).getIndex()); } })); // Create "VALUES (('commission'), ('salary')) AS t (remuneration_type)" values(ImmutableList.copyOf(Pair.left(map)), axisNameList); join(JoinRelType.INNER); final ImmutableBitSet unusedFields = ImmutableBitSet.range(leftRowType.getFieldCount()) .except(ImmutableBitSet.fromBitSet(usedFields)); final List projects = new ArrayList<>(fields(unusedFields)); Ord.forEach(axisNameList, (dimensionName, d) -> projects.add( alias(field(leftRowType.getFieldCount() + d), dimensionName))); final List conditions = new ArrayList<>(); Ord.forEach(measureNameList, (measureName, m) -> { final List caseOperands = new ArrayList<>(); Pair.forEach(map, (literals, nodes) -> { Ord.forEach(literals, (literal, d) -> conditions.add( equals(field(leftRowType.getFieldCount() + d), literal))); caseOperands.add(and(conditions)); conditions.clear(); caseOperands.add(nodes.get(m)); }); caseOperands.add(literal(null)); projects.add( alias(call(SqlStdOperatorTable.CASE, caseOperands), measureName)); }); project(projects); if (!includeNulls) { // Add 'WHERE m1 IS NOT NULL OR m2 IS NOT NULL' final BitSet notNullFields = new BitSet(); Ord.forEach(measureNameList, (measureName, m) -> { final int f = unusedFields.cardinality() + axisNameList.size() + m; conditions.add(isNotNull(field(f))); notNullFields.set(f); }); filter(or(conditions)); if (measureNameList.size() == 1) { // If there is one field, EXCLUDE NULLS will have converted it to NOT // NULL. final RelDataTypeFactory.Builder builder = getTypeFactory().builder(); peek().getRowType().getFieldList().forEach(field -> { final RelDataType type = field.getType(); builder.add(field.getName(), notNullFields.get(field.getIndex()) ? getTypeFactory().createTypeWithNullability(type, false) : type); }); convert(builder.build(), false); } conditions.clear(); } return this; } /** * Attaches an array of hints to the stack top relational expression. * *

The redundant hints would be eliminated. * * @param hints Hints * * @throws AssertionError if the top relational expression does not implement * {@link com.hazelcast.org.apache.calcite.rel.hint.Hintable} */ public RelBuilder hints(RelHint... hints) { return hints(ImmutableList.copyOf(hints)); } /** * Attaches multiple hints to the stack top relational expression. * *

The redundant hints would be eliminated. * * @param hints Hints * * @throws AssertionError if the top relational expression does not implement * {@link com.hazelcast.org.apache.calcite.rel.hint.Hintable} */ public RelBuilder hints(Iterable hints) { requireNonNull(hints, "hints"); final List relHintList = hints instanceof List ? (List) hints : Lists.newArrayList(hints); if (relHintList.isEmpty()) { return this; } final Frame frame = peek_(); assert frame != null : "There is no relational expression to attach the hints"; assert frame.rel instanceof Hintable : "The top relational expression is not a Hintable"; Hintable hintable = (Hintable) frame.rel; replaceTop(hintable.attachHints(relHintList)); return this; } /** Clears the stack. * *

The builder's state is now the same as when it was created. */ public void clear() { stack.clear(); } /** Information necessary to create a call to an aggregate function. * * @see RelBuilder#aggregateCall */ public interface AggCall { /** Returns a copy of this AggCall that applies a filter before aggregating * values. */ AggCall filter(@Nullable RexNode condition); /** Returns a copy of this AggCall that sorts its input values by * {@code orderKeys} before aggregating, as in SQL's {@code WITHIN GROUP} * clause. */ AggCall sort(Iterable orderKeys); /** Returns a copy of this AggCall that sorts its input values by * {@code orderKeys} before aggregating, as in SQL's {@code WITHIN GROUP} * clause. */ default AggCall sort(RexNode... orderKeys) { return sort(ImmutableList.copyOf(orderKeys)); } /** Returns a copy of this AggCall that makes its input values unique by * {@code distinctKeys} before aggregating, as in SQL's * {@code WITHIN DISTINCT} clause. */ AggCall unique(@Nullable Iterable distinctKeys); /** Returns a copy of this AggCall that makes its input values unique by * {@code distinctKeys} before aggregating, as in SQL's * {@code WITHIN DISTINCT} clause. */ default AggCall unique(RexNode... distinctKeys) { return unique(ImmutableList.copyOf(distinctKeys)); } /** Returns a copy of this AggCall that may return approximate results * if {@code approximate} is true. */ AggCall approximate(boolean approximate); /** Returns a copy of this AggCall that ignores nulls. */ AggCall ignoreNulls(boolean ignoreNulls); /** Returns a copy of this AggCall with a given alias. */ AggCall as(@Nullable String alias); /** Returns a copy of this AggCall that is optionally distinct. */ AggCall distinct(boolean distinct); /** Returns a copy of this AggCall that is distinct. */ default AggCall distinct() { return distinct(true); } /** Converts this aggregate call to a windowed aggregate call. */ OverCall over(); } /** Internal methods shared by all implementations of {@link AggCall}. */ private interface AggCallPlus extends AggCall { /** Returns the aggregate function. */ SqlAggFunction op(); /** Returns the alias. */ @Nullable String alias(); /** Returns an {@link AggregateCall} that is approximately equivalent * to this {@code AggCall} and is good for certain things, such as deriving * field names. */ AggregateCall aggregateCall(); /** Converts this {@code AggCall} to a good {@link AggregateCall}. */ AggregateCall aggregateCall(Registrar registrar, ImmutableBitSet groupSet, RelNode r); /** Registers expressions in operands and filters. */ void register(Registrar registrar); } /** Information necessary to create the GROUP BY clause of an Aggregate. * * @see RelBuilder#groupKey */ public interface GroupKey { /** Assigns an alias to this group key. * *

Used to assign field names in the {@code group} operation. */ GroupKey alias(@Nullable String alias); /** Returns the number of columns in the group key. */ int groupKeyCount(); } /** Implementation of {@link RelBuilder.GroupKey}. */ static class GroupKeyImpl implements GroupKey { final ImmutableList nodes; final @Nullable ImmutableList> nodeLists; final @Nullable String alias; GroupKeyImpl(ImmutableList nodes, @Nullable ImmutableList> nodeLists, @Nullable String alias) { this.nodes = requireNonNull(nodes, "nodes"); this.nodeLists = nodeLists; this.alias = alias; } @Override public String toString() { return alias == null ? nodes.toString() : nodes + " as " + alias; } @Override public int groupKeyCount() { return nodes.size(); } @Override public GroupKey alias(@Nullable String alias) { return Objects.equals(this.alias, alias) ? this : new GroupKeyImpl(nodes, nodeLists, alias); } boolean isSimple() { return nodeLists == null || nodeLists.size() == 1; } } /** * Checks for {@link CorrelationId}, then validates the id is not used on left, * and finally checks if id is actually used on right. * * @return true if a correlate id is present and used * * @throws IllegalArgumentException if the {@link CorrelationId} is used by left side or if the a * {@link CorrelationId} is present and the {@link JoinRelType} is FULL or RIGHT. */ private static boolean checkIfCorrelated(Set variablesSet, JoinRelType joinType, RelNode leftNode, RelNode rightRel) { if (variablesSet.size() != 1) { return false; } CorrelationId id = Iterables.getOnlyElement(variablesSet); if (!RelOptUtil.notContainsCorrelation(leftNode, id, Litmus.IGNORE)) { throw new IllegalArgumentException("variable " + id + " must not be used by left input to correlation"); } switch (joinType) { case RIGHT: case FULL: throw new IllegalArgumentException("Correlated " + joinType + " join is not supported"); default: return !RelOptUtil.correlationColumns( Iterables.getOnlyElement(variablesSet), rightRel).isEmpty(); } } /** Implementation of {@link AggCall}. */ private class AggCallImpl implements AggCallPlus { private final SqlAggFunction aggFunction; private final boolean distinct; private final boolean approximate; private final boolean ignoreNulls; private final @Nullable RexNode filter; private final @Nullable String alias; private final ImmutableList operands; // may be empty private final @Nullable ImmutableList distinctKeys; // may be empty or null private final ImmutableList orderKeys; // may be empty AggCallImpl(SqlAggFunction aggFunction, boolean distinct, boolean approximate, boolean ignoreNulls, @Nullable RexNode filter, @Nullable String alias, ImmutableList operands, @Nullable ImmutableList distinctKeys, ImmutableList orderKeys) { this.aggFunction = requireNonNull(aggFunction, "aggFunction"); // If the aggregate function ignores DISTINCT, // make the DISTINCT flag FALSE. this.distinct = distinct && aggFunction.getDistinctOptionality() != Optionality.IGNORED; this.approximate = approximate; this.ignoreNulls = ignoreNulls; this.alias = alias; this.operands = requireNonNull(operands, "operands"); this.distinctKeys = distinctKeys; this.orderKeys = requireNonNull(orderKeys, "orderKeys"); if (filter != null) { if (filter.getType().getSqlTypeName() != SqlTypeName.BOOLEAN) { throw RESOURCE.filterMustBeBoolean().ex(); } if (filter.getType().isNullable()) { filter = call(SqlStdOperatorTable.IS_TRUE, filter); } } this.filter = filter; } @Override public String toString() { final StringBuilder b = new StringBuilder(); b.append(aggFunction.getName()) .append('('); if (distinct) { b.append("DISTINCT "); } if (operands.size() > 0) { b.append(operands.get(0)); for (int i = 1; i < operands.size(); i++) { b.append(", "); b.append(operands.get(i)); } } b.append(')'); if (filter != null) { b.append(" FILTER (WHERE ").append(filter).append(')'); } if (distinctKeys != null) { b.append(" WITHIN DISTINCT (").append(distinctKeys).append(')'); } return b.toString(); } @Override public SqlAggFunction op() { return aggFunction; } @Override public @Nullable String alias() { return alias; } @Override public AggregateCall aggregateCall() { // Use dummy values for collation and type. This method only promises to // return a call that is "approximately equivalent ... and is good for // deriving field names", so dummy values are good enough. final RelCollation collation = RelCollations.EMPTY; final RelDataType type = getTypeFactory().createSqlType(SqlTypeName.BOOLEAN); return AggregateCall.create(aggFunction, distinct, approximate, ignoreNulls, ImmutableList.of(), -1, null, collation, type, alias); } @Override public AggregateCall aggregateCall(Registrar registrar, ImmutableBitSet groupSet, RelNode r) { List args = registrar.registerExpressions(this.operands); final int filterArg = this.filter == null ? -1 : registrar.registerExpression(this.filter); if (this.distinct && !this.aggFunction.isQuantifierAllowed()) { throw new IllegalArgumentException("DISTINCT not allowed"); } if (this.filter != null && !this.aggFunction.allowsFilter()) { throw new IllegalArgumentException("FILTER not allowed"); } final @Nullable ImmutableBitSet distinctKeys = this.distinctKeys == null ? null : ImmutableBitSet.of( registrar.registerExpressions(this.distinctKeys)); final RelCollation collation = RelCollations.of(this.orderKeys .stream() .map(orderKey -> collation(orderKey, RelFieldCollation.Direction.ASCENDING, null, Collections.emptyList())) .collect(Collectors.toList())); if (aggFunction instanceof SqlCountAggFunction && !distinct) { args = args.stream() .filter(r::fieldIsNullable) .collect(Util.toImmutableList()); } return AggregateCall.create(aggFunction, distinct, approximate, ignoreNulls, args, filterArg, distinctKeys, collation, groupSet.cardinality(), r, null, alias); } @Override public void register(Registrar registrar) { registrar.registerExpressions(operands); if (filter != null) { registrar.registerExpression(filter); } if (distinctKeys != null) { registrar.registerExpressions(distinctKeys); } registrar.registerExpressions(orderKeys); } @Override public OverCall over() { return new OverCallImpl(aggFunction, distinct, operands, ignoreNulls, alias); } @Override public AggCall sort(Iterable orderKeys) { final ImmutableList orderKeyList = ImmutableList.copyOf(orderKeys); return orderKeyList.equals(this.orderKeys) ? this : new AggCallImpl(aggFunction, distinct, approximate, ignoreNulls, filter, alias, operands, distinctKeys, orderKeyList); } @Override public AggCall sort(RexNode... orderKeys) { return sort(ImmutableList.copyOf(orderKeys)); } @Override public AggCall unique(@Nullable Iterable distinctKeys) { final @Nullable ImmutableList distinctKeyList = distinctKeys == null ? null : ImmutableList.copyOf(distinctKeys); return Objects.equals(distinctKeyList, this.distinctKeys) ? this : new AggCallImpl(aggFunction, distinct, approximate, ignoreNulls, filter, alias, operands, distinctKeyList, orderKeys); } @Override public AggCall approximate(boolean approximate) { return approximate == this.approximate ? this : new AggCallImpl(aggFunction, distinct, approximate, ignoreNulls, filter, alias, operands, distinctKeys, orderKeys); } @Override public AggCall filter(@Nullable RexNode condition) { return Objects.equals(condition, this.filter) ? this : new AggCallImpl(aggFunction, distinct, approximate, ignoreNulls, condition, alias, operands, distinctKeys, orderKeys); } @Override public AggCall as(@Nullable String alias) { return Objects.equals(alias, this.alias) ? this : new AggCallImpl(aggFunction, distinct, approximate, ignoreNulls, filter, alias, operands, distinctKeys, orderKeys); } @Override public AggCall distinct(boolean distinct) { return distinct == this.distinct ? this : new AggCallImpl(aggFunction, distinct, approximate, ignoreNulls, filter, alias, operands, distinctKeys, orderKeys); } @Override public AggCall ignoreNulls(boolean ignoreNulls) { return ignoreNulls == this.ignoreNulls ? this : new AggCallImpl(aggFunction, distinct, approximate, ignoreNulls, filter, alias, operands, distinctKeys, orderKeys); } } /** Implementation of {@link AggCall} that wraps an * {@link AggregateCall}. */ private class AggCallImpl2 implements AggCallPlus { private final AggregateCall aggregateCall; private final ImmutableList operands; AggCallImpl2(AggregateCall aggregateCall, ImmutableList operands) { this.aggregateCall = requireNonNull(aggregateCall, "aggregateCall"); this.operands = requireNonNull(operands, "operands"); } @Override public OverCall over() { return new OverCallImpl(aggregateCall.getAggregation(), aggregateCall.isDistinct(), operands, aggregateCall.ignoreNulls(), aggregateCall.name); } @Override public String toString() { return aggregateCall.toString(); } @Override public SqlAggFunction op() { return aggregateCall.getAggregation(); } @Override public @Nullable String alias() { return aggregateCall.name; } @Override public AggregateCall aggregateCall() { return aggregateCall; } @Override public AggregateCall aggregateCall(Registrar registrar, ImmutableBitSet groupSet, RelNode r) { return aggregateCall; } @Override public void register(Registrar registrar) { // nothing to do } @Override public AggCall sort(Iterable orderKeys) { throw new UnsupportedOperationException(); } @Override public AggCall sort(RexNode... orderKeys) { throw new UnsupportedOperationException(); } @Override public AggCall unique(@Nullable Iterable distinctKeys) { throw new UnsupportedOperationException(); } @Override public AggCall approximate(boolean approximate) { throw new UnsupportedOperationException(); } @Override public AggCall filter(@Nullable RexNode condition) { throw new UnsupportedOperationException(); } @Override public AggCall as(@Nullable String alias) { throw new UnsupportedOperationException(); } @Override public AggCall distinct(boolean distinct) { throw new UnsupportedOperationException(); } @Override public AggCall ignoreNulls(boolean ignoreNulls) { throw new UnsupportedOperationException(); } } /** Call to a windowed aggregate function. * *

To create an {@code OverCall}, start with an {@link AggCall} (created * by a method such as {@link #aggregateCall}, {@link #sum} or {@link #count}) * and call its {@link AggCall#over()} method. For example, * *

{@code
   *      b.scan("EMP")
   *         .project(b.field("DEPTNO"),
   *            b.aggregateCall(SqlStdOperatorTable.ROW_NUMBER)
   *               .over()
   *               .partitionBy()
   *               .orderBy(b.field("EMPNO"))
   *               .rowsUnbounded()
   *               .allowPartial(true)
   *               .nullWhenCountZero(false)
   *               .as("x"))
   * }
* *

Unlike an aggregate call, a windowed aggregate call is an expression * that you can use in a {@link Project} or {@link Filter}. So, to finish, * call {@link OverCall#toRex()} to convert the {@code OverCall} to a * {@link RexNode}; the {@link OverCall#as} method (used in the above example) * does the same but also assigns an column alias. */ public interface OverCall { /** Performs an action on this OverCall. */ default R let(Function consumer) { return consumer.apply(this); } /** Sets the PARTITION BY clause to an array of expressions. */ OverCall partitionBy(RexNode... expressions); /** Sets the PARTITION BY clause to a list of expressions. */ OverCall partitionBy(Iterable expressions); /** Sets the ORDER BY BY clause to an array of expressions. * *

Use {@link #desc(RexNode)}, {@link #nullsFirst(RexNode)}, * {@link #nullsLast(RexNode)} to control the sort order. */ OverCall orderBy(RexNode... expressions); /** Sets the ORDER BY BY clause to a list of expressions. * *

Use {@link #desc(RexNode)}, {@link #nullsFirst(RexNode)}, * {@link #nullsLast(RexNode)} to control the sort order. */ OverCall orderBy(Iterable expressions); /** Sets an unbounded ROWS window, * equivalent to SQL {@code ROWS BETWEEN UNBOUNDED PRECEDING AND * UNBOUNDED FOLLOWING}. */ default OverCall rowsUnbounded() { return rowsBetween(RexWindowBounds.UNBOUNDED_PRECEDING, RexWindowBounds.UNBOUNDED_FOLLOWING); } /** Sets a ROWS window with a lower bound, * equivalent to SQL {@code ROWS BETWEEN lower AND CURRENT ROW}. */ default OverCall rowsFrom(RexWindowBound lower) { return rowsBetween(lower, RexWindowBounds.UNBOUNDED_FOLLOWING); } /** Sets a ROWS window with an upper bound, * equivalent to SQL {@code ROWS BETWEEN CURRENT ROW AND upper}. */ default OverCall rowsTo(RexWindowBound upper) { return rowsBetween(RexWindowBounds.UNBOUNDED_PRECEDING, upper); } /** Sets a RANGE window with lower and upper bounds, * equivalent to SQL {@code ROWS BETWEEN lower ROW AND upper}. */ OverCall rowsBetween(RexWindowBound lower, RexWindowBound upper); /** Sets an unbounded RANGE window, * equivalent to SQL {@code RANGE BETWEEN UNBOUNDED PRECEDING AND * UNBOUNDED FOLLOWING}. */ default OverCall rangeUnbounded() { return rangeBetween(RexWindowBounds.UNBOUNDED_PRECEDING, RexWindowBounds.UNBOUNDED_FOLLOWING); } /** Sets a RANGE window with a lower bound, * equivalent to SQL {@code RANGE BETWEEN lower AND CURRENT ROW}. */ default OverCall rangeFrom(RexWindowBound lower) { return rangeBetween(lower, RexWindowBounds.CURRENT_ROW); } /** Sets a RANGE window with an upper bound, * equivalent to SQL {@code RANGE BETWEEN CURRENT ROW AND upper}. */ default OverCall rangeTo(RexWindowBound upper) { return rangeBetween(RexWindowBounds.UNBOUNDED_PRECEDING, upper); } /** Sets a RANGE window with lower and upper bounds, * equivalent to SQL {@code RANGE BETWEEN lower ROW AND upper}. */ OverCall rangeBetween(RexWindowBound lower, RexWindowBound upper); /** Sets whether to allow partial width windows; default true. */ OverCall allowPartial(boolean allowPartial); /** Sets whether the aggregate function should evaluate to null if no rows * are in the window; default false. */ OverCall nullWhenCountZero(boolean nullWhenCountZero); /** Sets the alias of this expression, and converts it to a {@link RexNode}; * default is the alias that was set via {@link AggCall#as(String)}. */ RexNode as(String alias); /** Converts this expression to a {@link RexNode}. */ RexNode toRex(); } /** Implementation of {@link OverCall}. */ private class OverCallImpl implements OverCall { private final ImmutableList operands; private final boolean ignoreNulls; private final @Nullable String alias; private final boolean nullWhenCountZero; private final boolean allowPartial; private final boolean rows; private final RexWindowBound lowerBound; private final RexWindowBound upperBound; private final ImmutableList partitionKeys; private final ImmutableList sortKeys; private final SqlAggFunction op; private final boolean distinct; private OverCallImpl(SqlAggFunction op, boolean distinct, ImmutableList operands, boolean ignoreNulls, @Nullable String alias, ImmutableList partitionKeys, ImmutableList sortKeys, boolean rows, RexWindowBound lowerBound, RexWindowBound upperBound, boolean nullWhenCountZero, boolean allowPartial) { this.op = op; this.distinct = distinct; this.operands = operands; this.ignoreNulls = ignoreNulls; this.alias = alias; this.partitionKeys = partitionKeys; this.sortKeys = sortKeys; this.nullWhenCountZero = nullWhenCountZero; this.allowPartial = allowPartial; this.rows = rows; this.lowerBound = lowerBound; this.upperBound = upperBound; } /** Creates an OverCallImpl with default settings. */ OverCallImpl(SqlAggFunction op, boolean distinct, ImmutableList operands, boolean ignoreNulls, @Nullable String alias) { this(op, distinct, operands, ignoreNulls, alias, ImmutableList.of(), ImmutableList.of(), true, RexWindowBounds.UNBOUNDED_PRECEDING, RexWindowBounds.UNBOUNDED_FOLLOWING, false, true); } @Override public OverCall partitionBy( Iterable expressions) { return partitionBy_(ImmutableList.copyOf(expressions)); } @Override public OverCall partitionBy(RexNode... expressions) { return partitionBy_(ImmutableList.copyOf(expressions)); } private OverCall partitionBy_(ImmutableList partitionKeys) { return new OverCallImpl(op, distinct, operands, ignoreNulls, alias, partitionKeys, sortKeys, rows, lowerBound, upperBound, nullWhenCountZero, allowPartial); } private OverCall orderBy_(ImmutableList sortKeys) { return new OverCallImpl(op, distinct, operands, ignoreNulls, alias, partitionKeys, sortKeys, rows, lowerBound, upperBound, nullWhenCountZero, allowPartial); } @Override public OverCall orderBy(Iterable sortKeys) { ImmutableList.Builder fieldCollations = ImmutableList.builder(); sortKeys.forEach(sortKey -> fieldCollations.add( rexCollation(sortKey, RelFieldCollation.Direction.ASCENDING, RelFieldCollation.NullDirection.UNSPECIFIED))); return orderBy_(fieldCollations.build()); } @Override public OverCall orderBy(RexNode... sortKeys) { return orderBy(Arrays.asList(sortKeys)); } @Override public OverCall rowsBetween(RexWindowBound lowerBound, RexWindowBound upperBound) { return new OverCallImpl(op, distinct, operands, ignoreNulls, alias, partitionKeys, sortKeys, true, lowerBound, upperBound, nullWhenCountZero, allowPartial); } @Override public OverCall rangeBetween(RexWindowBound lowerBound, RexWindowBound upperBound) { return new OverCallImpl(op, distinct, operands, ignoreNulls, alias, partitionKeys, sortKeys, false, lowerBound, upperBound, nullWhenCountZero, allowPartial); } @Override public OverCall allowPartial(boolean allowPartial) { return new OverCallImpl(op, distinct, operands, ignoreNulls, alias, partitionKeys, sortKeys, rows, lowerBound, upperBound, nullWhenCountZero, allowPartial); } @Override public OverCall nullWhenCountZero(boolean nullWhenCountZero) { return new OverCallImpl(op, distinct, operands, ignoreNulls, alias, partitionKeys, sortKeys, rows, lowerBound, upperBound, nullWhenCountZero, allowPartial); } @Override public RexNode as(String alias) { return new OverCallImpl(op, distinct, operands, ignoreNulls, alias, partitionKeys, sortKeys, rows, lowerBound, upperBound, nullWhenCountZero, allowPartial).toRex(); } @Override public RexNode toRex() { final RexCallBinding bind = new RexCallBinding(getTypeFactory(), op, operands, ImmutableList.of()) { @Override public int getGroupCount() { return SqlWindow.isAlwaysNonEmpty(lowerBound, upperBound) ? 1 : 0; } }; final RelDataType type = op.inferReturnType(bind); final RexNode over = getRexBuilder() .makeOver(type, op, operands, partitionKeys, sortKeys, lowerBound, upperBound, rows, allowPartial, nullWhenCountZero, distinct, ignoreNulls); return alias == null ? over : alias(over, alias); } } /** Collects the extra expressions needed for {@link #aggregate}. * *

The extra expressions come from the group key and as arguments to * aggregate calls, and later there will be a {@link #project} or a * {@link #rename(List)} if necessary. */ private static class Registrar { final List originalExtraNodes; final List extraNodes; final List<@Nullable String> names; Registrar(Iterable fields, List fieldNames) { originalExtraNodes = ImmutableList.copyOf(fields); extraNodes = new ArrayList<>(originalExtraNodes); names = new ArrayList<>(fieldNames); } int registerExpression(RexNode node) { switch (node.getKind()) { case AS: final List operands = ((RexCall) node).operands; final int i = registerExpression(operands.get(0)); names.set(i, RexLiteral.stringValue(operands.get(1))); return i; case DESCENDING: case NULLS_FIRST: case NULLS_LAST: return registerExpression(((RexCall) node).operands.get(0)); default: final int i2 = extraNodes.indexOf(node); if (i2 >= 0) { return i2; } extraNodes.add(node); names.add(null); return extraNodes.size() - 1; } } List registerExpressions(Iterable nodes) { final List builder = new ArrayList<>(); for (RexNode node : nodes) { builder.add(registerExpression(node)); } return builder; } List registerFieldCollations( Iterable orderKeys) { final List fieldCollations = new ArrayList<>(); for (RexNode orderKey : orderKeys) { final RelFieldCollation collation = collation(orderKey, RelFieldCollation.Direction.ASCENDING, null, extraNodes); if (!RelCollations.ordinals(fieldCollations) .contains(collation.getFieldIndex())) { fieldCollations.add(collation); } } return ImmutableList.copyOf(fieldCollations); } /** Returns the number of fields added. */ int addedFieldCount() { return extraNodes.size() - originalExtraNodes.size(); } } /** Builder stack frame. * *

Describes a previously created relational expression and * information about how table aliases map into its row type. */ private static class Frame { final RelNode rel; final ImmutableList fields; private Frame(RelNode rel, ImmutableList fields) { this.rel = rel; this.fields = fields; } private Frame(RelNode rel) { String tableAlias = deriveAlias(rel); ImmutableList.Builder builder = ImmutableList.builder(); ImmutableSet aliases = tableAlias == null ? ImmutableSet.of() : ImmutableSet.of(tableAlias); for (RelDataTypeField field : rel.getRowType().getFieldList()) { builder.add(new Field(aliases, field)); } this.rel = rel; this.fields = builder.build(); } @Override public String toString() { return rel + ": " + fields; } private static @Nullable String deriveAlias(RelNode rel) { if (rel instanceof TableScan) { TableScan scan = (TableScan) rel; final List names = scan.getTable().getQualifiedName(); if (!names.isEmpty()) { return Util.last(names); } } return null; } List fields() { return Pair.right(fields); } } /** A field that belongs to a stack {@link Frame}. */ private static class Field extends Pair, RelDataTypeField> { Field(ImmutableSet left, RelDataTypeField right) { super(left, right); } Field addAlias(String alias) { if (left.contains(alias)) { return this; } final ImmutableSet aliasList = ImmutableSet.builder().addAll(left).add(alias).build(); return new Field(aliasList, right); } } /** Shuttle that shifts a predicate's inputs to the left, replacing early * ones with references to a * {@link RexCorrelVariable}. */ private class Shifter extends RexShuttle { private final RelNode left; private final CorrelationId id; private final RelNode right; Shifter(RelNode left, CorrelationId id, RelNode right) { this.left = left; this.id = id; this.right = right; } @Override public RexNode visitInputRef(RexInputRef inputRef) { final RelDataType leftRowType = left.getRowType(); final RexBuilder rexBuilder = getRexBuilder(); final int leftCount = leftRowType.getFieldCount(); if (inputRef.getIndex() < leftCount) { final RexNode v = rexBuilder.makeCorrel(leftRowType, id); return rexBuilder.makeFieldAccess(v, inputRef.getIndex()); } else { return rexBuilder.makeInputRef(right, inputRef.getIndex() - leftCount); } } } /** Configuration of RelBuilder. * *

It is immutable, and all fields are public. * *

Start with the {@link #DEFAULT} instance, * and call {@code withXxx} methods to set its properties. */ @Value.Immutable public interface Config { /** Default configuration. */ Config DEFAULT = ImmutableRelBuilder.Config.of(); /** Controls whether to merge two {@link Project} operators when inlining * expressions causes complexity to increase. * *

Usually merging projects is beneficial, but occasionally the * result is more complex than the original projects. Consider: * *

     * P: Project(a+b+c AS x, d+e+f AS y, g+h+i AS z)  # complexity 15
     * Q: Project(x*y*z AS p, x-y-z AS q)              # complexity 10
     * R: Project((a+b+c)*(d+e+f)*(g+h+i) AS s,
     *            (a+b+c)-(d+e+f)-(g+h+i) AS t)        # complexity 34
     * 
* * The complexity of an expression is the number of nodes (leaves and * operators). For example, {@code a+b+c} has complexity 5 (3 field * references and 2 calls): * *
     *       +
     *      /  \
     *     +    c
     *    / \
     *   a   b
     * 
* *

A negative value never allows merges. * *

A zero or positive value, {@code bloat}, allows a merge if complexity * of the result is less than or equal to the sum of the complexity of the * originals plus {@code bloat}. * *

The default value, 100, allows a moderate increase in complexity but * prevents cases where complexity would run away into the millions and run * out of memory. Moderate complexity is OK; the implementation, say via * {@link com.hazelcast.org.apache.calcite.adapter.enumerable.EnumerableCalc}, will often * gather common sub-expressions and compute them only once. */ @Value.Default default int bloat() { return 100; } /** Sets {@link #bloat}. */ Config withBloat(int bloat); /** Whether {@link RelBuilder#aggregate} should eliminate duplicate * aggregate calls; default true. */ @Value.Default default boolean dedupAggregateCalls() { return true; } /** Sets {@link #dedupAggregateCalls}. */ Config withDedupAggregateCalls(boolean dedupAggregateCalls); /** Whether {@link RelBuilder#aggregate} should prune unused * input columns; default true. */ @Value.Default default boolean pruneInputOfAggregate() { return true; } /** Sets {@link #pruneInputOfAggregate}. */ Config withPruneInputOfAggregate(boolean pruneInputOfAggregate); /** Whether to push down join conditions; default false (but * {@link SqlToRelConverter#config()} by default sets this to true). */ @Value.Default default boolean pushJoinCondition() { return false; } /** Sets {@link #pushJoinCondition()}. */ Config withPushJoinCondition(boolean pushJoinCondition); /** Whether to simplify expressions; default true. */ @Value.Default default boolean simplify() { return true; } /** Sets {@link #simplify}. */ Config withSimplify(boolean simplify); /** Whether to simplify LIMIT 0 to an empty relation; default true. */ @Value.Default default boolean simplifyLimit() { return true; } /** Sets {@link #simplifyLimit()}. */ Config withSimplifyLimit(boolean simplifyLimit); /** Whether to simplify {@code Union(Values, Values)} or * {@code Union(Project(Values))} to {@code Values}; default true. */ @Value.Default default boolean simplifyValues() { return true; } /** Sets {@link #simplifyValues()}. */ Config withSimplifyValues(boolean simplifyValues); /** Whether to create an Aggregate even if we know that the input is * already unique; default false. */ @Value.Default default boolean aggregateUnique() { return false; } /** Sets {@link #aggregateUnique()}. */ Config withAggregateUnique(boolean aggregateUnique); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy