public void reduce()

in exec/java-exec/src/main/java/org/apache/drill/exec/planner/logical/DrillConstExecutor.java [136:393]


  public void reduce(RexBuilder rexBuilder, List<RexNode> constExps, List<RexNode> reducedValues) {
    for (RexNode newCall : constExps) {
      LogicalExpression logEx = DrillOptiq.toDrill(new DrillParseContext(plannerSettings), (RelNode) null /* input rel */, newCall);

      ErrorCollectorImpl errors = new ErrorCollectorImpl();
      LogicalExpression materializedExpr = ExpressionTreeMaterializer.materialize(logEx, null, errors, funcImplReg);
      if (errors.getErrorCount() != 0) {
        String message = String.format(
            "Failure while materializing expression in constant expression evaluator [%s].  Errors: %s",
            newCall.toString(), errors.toString());
        throw UserException.planError()
          .message(message)
          .build(logger);
      }

      if (NON_REDUCIBLE_TYPES.contains(materializedExpr.getMajorType().getMinorType())) {
        logger.debug("Constant expression not folded due to return type {}, complete expression: {}",
            materializedExpr.getMajorType(),
            ExpressionStringBuilder.toString(materializedExpr));
        reducedValues.add(newCall);
        continue;
      }

      ValueHolder output = InterpreterEvaluator.evaluateConstantExpr(udfUtilities, materializedExpr);
      RelDataTypeFactory typeFactory = rexBuilder.getTypeFactory();

      if (materializedExpr.getMajorType().getMode() == TypeProtos.DataMode.OPTIONAL && TypeHelper.isNull(output)) {
        SqlTypeName sqlTypeName = TypeInferenceUtils.getCalciteTypeFromDrillType(materializedExpr.getMajorType().getMinorType());
        if (sqlTypeName == null) {
          String message = String.format("Error reducing constant expression, unsupported type: %s.",
              materializedExpr.getMajorType().getMinorType());
          throw UserException.unsupportedError()
            .message(message)
            .build(logger);
        }

        RelDataType type = TypeInferenceUtils.createCalciteTypeWithNullability(typeFactory, sqlTypeName, true);
        reducedValues.add(rexBuilder.makeNullLiteral(type));
        continue;
      }

      Function<ValueHolder, RexNode> literator = valueHolder -> {
        switch (materializedExpr.getMajorType().getMinorType()) {
          case INT: {
            int value = (materializedExpr.getMajorType().getMode() == TypeProtos.DataMode.OPTIONAL) ?
                ((NullableIntHolder) valueHolder).value : ((IntHolder) valueHolder).value;
            return rexBuilder.makeLiteral(BigDecimal.valueOf(value),
                TypeInferenceUtils.createCalciteTypeWithNullability(typeFactory, SqlTypeName.INTEGER, newCall.getType().isNullable()), false);
          }
          case BIGINT: {
            long value = (materializedExpr.getMajorType().getMode() == TypeProtos.DataMode.OPTIONAL) ?
                ((NullableBigIntHolder) valueHolder).value : ((BigIntHolder) valueHolder).value;
            return rexBuilder.makeLiteral(BigDecimal.valueOf(value),
                TypeInferenceUtils.createCalciteTypeWithNullability(typeFactory, SqlTypeName.BIGINT, newCall.getType().isNullable()), false);
          }
          case FLOAT4: {
            float value = (materializedExpr.getMajorType().getMode() == TypeProtos.DataMode.OPTIONAL) ?
                ((NullableFloat4Holder) valueHolder).value : ((Float4Holder) valueHolder).value;

            // +Infinity, -Infinity and NaN must be represented as strings since
            // BigDecimal cannot represent them.
            if (!Float.isFinite(value)) {
              return rexBuilder.makeLiteral(Float.toString(value));
            }

            return rexBuilder.makeLiteral(BigDecimal.valueOf(value),
                TypeInferenceUtils.createCalciteTypeWithNullability(typeFactory, SqlTypeName.FLOAT, newCall.getType().isNullable()), false);
          }
          case FLOAT8: {
            double value = (materializedExpr.getMajorType().getMode() == TypeProtos.DataMode.OPTIONAL) ?
                ((NullableFloat8Holder) valueHolder).value : ((Float8Holder) valueHolder).value;

            // +Infinity, -Infinity and NaN must be represented as strings since
            // BigDecimal cannot represent them.
            if (!Double.isFinite(value)) {
              return rexBuilder.makeLiteral(Double.toString(value));
            }

            return rexBuilder.makeLiteral(BigDecimal.valueOf(value),
                TypeInferenceUtils.createCalciteTypeWithNullability(typeFactory, SqlTypeName.DOUBLE, newCall.getType().isNullable()), false);
          }
          case VARCHAR: {
            String value = (materializedExpr.getMajorType().getMode() == TypeProtos.DataMode.OPTIONAL) ?
                StringFunctionHelpers.getStringFromVarCharHolder((NullableVarCharHolder) valueHolder) :
                StringFunctionHelpers.getStringFromVarCharHolder((VarCharHolder) valueHolder);
            RelDataType type = typeFactory.createSqlType(SqlTypeName.VARCHAR, newCall.getType().getPrecision());
            RelDataType typeWithNullability = typeFactory.createTypeWithNullability(type, newCall.getType().isNullable());
            return rexBuilder.makeLiteral(value, typeWithNullability, false);
          }
          case BIT: {
            boolean value = (materializedExpr.getMajorType().getMode() == TypeProtos.DataMode.OPTIONAL) ?
                ((NullableBitHolder) valueHolder).value == 1 : ((BitHolder) valueHolder).value == 1;
            return rexBuilder.makeLiteral(value,
                TypeInferenceUtils.createCalciteTypeWithNullability(typeFactory, SqlTypeName.BOOLEAN, newCall.getType().isNullable()), false);
          }
          case DATE: {
            long millis = (materializedExpr.getMajorType().getMode() == TypeProtos.DataMode.OPTIONAL) ?
                ((NullableDateHolder) valueHolder).value :
                ((DateHolder) valueHolder).value;
            long days = millis / TimeUnit.DAYS.toMillis(1);
            LocalDate localDate = LocalDate.ofEpochDay(days);
            return rexBuilder.makeLiteral(new DateString(localDate.getYear(), localDate.getMonthValue(), localDate.getDayOfMonth()),
                TypeInferenceUtils.createCalciteTypeWithNullability(typeFactory, SqlTypeName.DATE, newCall.getType().isNullable()), false);
          }
          case DECIMAL9: {
            long value;
            int scale;
            if (materializedExpr.getMajorType().getMode() == TypeProtos.DataMode.OPTIONAL) {
              NullableDecimal9Holder decimal9Out = (NullableDecimal9Holder) valueHolder;
              value = decimal9Out.value;
              scale = decimal9Out.scale;
            } else {
              Decimal9Holder decimal9Out = (Decimal9Holder) valueHolder;
              value = decimal9Out.value;
              scale = decimal9Out.scale;
            }
            return rexBuilder.makeLiteral(
                new BigDecimal(BigInteger.valueOf(value), scale),
                TypeInferenceUtils.createCalciteTypeWithNullability(typeFactory, SqlTypeName.DECIMAL, newCall.getType().isNullable()),
                false);
          }
          case DECIMAL18: {
            long value;
            int scale;
            if (materializedExpr.getMajorType().getMode() == TypeProtos.DataMode.OPTIONAL) {
              NullableDecimal18Holder decimal18Out = (NullableDecimal18Holder) valueHolder;
              value = decimal18Out.value;
              scale = decimal18Out.scale;
            } else {
              Decimal18Holder decimal18Out = (Decimal18Holder) valueHolder;
              value = decimal18Out.value;
              scale = decimal18Out.scale;
            }
            return rexBuilder.makeLiteral(
                new BigDecimal(BigInteger.valueOf(value), scale),
                TypeInferenceUtils.createCalciteTypeWithNullability(typeFactory, SqlTypeName.DECIMAL, newCall.getType().isNullable()),
                false);
          }
          case VARDECIMAL: {
            DrillBuf buffer;
            int start;
            int end;
            int scale;
            int precision;
            if (materializedExpr.getMajorType().getMode() == TypeProtos.DataMode.OPTIONAL) {
              NullableVarDecimalHolder varDecimalHolder = (NullableVarDecimalHolder) valueHolder;
              buffer = varDecimalHolder.buffer;
              start = varDecimalHolder.start;
              end = varDecimalHolder.end;
              scale = varDecimalHolder.scale;
              precision = varDecimalHolder.precision;
            } else {
              VarDecimalHolder varDecimalHolder = (VarDecimalHolder) valueHolder;
              buffer = varDecimalHolder.buffer;
              start = varDecimalHolder.start;
              end = varDecimalHolder.end;
              scale = varDecimalHolder.scale;
              precision = varDecimalHolder.precision;
            }
            return rexBuilder.makeLiteral(
                org.apache.drill.exec.util.DecimalUtility.getBigDecimalFromDrillBuf(buffer, start, end - start, scale),
                typeFactory.createSqlType(SqlTypeName.DECIMAL, precision, scale),
                false);
          }
          case DECIMAL28SPARSE: {
            DrillBuf buffer;
            int start;
            int scale;
            if (materializedExpr.getMajorType().getMode() == TypeProtos.DataMode.OPTIONAL) {
              NullableDecimal28SparseHolder decimal28Out = (NullableDecimal28SparseHolder) valueHolder;
              buffer = decimal28Out.buffer;
              start = decimal28Out.start;
              scale = decimal28Out.scale;
            } else {
              Decimal28SparseHolder decimal28Out = (Decimal28SparseHolder) valueHolder;
              buffer = decimal28Out.buffer;
              start = decimal28Out.start;
              scale = decimal28Out.scale;
            }
            return rexBuilder.makeLiteral(
                org.apache.drill.exec.util.DecimalUtility.getBigDecimalFromSparse(buffer, start * 20, 5, scale),
                TypeInferenceUtils.createCalciteTypeWithNullability(typeFactory, SqlTypeName.DECIMAL, newCall.getType().isNullable()), false);
          }
          case DECIMAL38SPARSE: {
            DrillBuf buffer;
            int start;
            int scale;
            if (materializedExpr.getMajorType().getMode() == TypeProtos.DataMode.OPTIONAL) {
              NullableDecimal38SparseHolder decimal38Out = (NullableDecimal38SparseHolder) valueHolder;
              buffer = decimal38Out.buffer;
              start = decimal38Out.start;
              scale = decimal38Out.scale;
            } else {
              Decimal38SparseHolder decimal38Out = (Decimal38SparseHolder) valueHolder;
              buffer = decimal38Out.buffer;
              start = decimal38Out.start;
              scale = decimal38Out.scale;
            }
            return rexBuilder.makeLiteral(org.apache.drill.exec.util.DecimalUtility.getBigDecimalFromSparse(buffer, start * 24, 6, scale),
                TypeInferenceUtils.createCalciteTypeWithNullability(typeFactory, SqlTypeName.DECIMAL, newCall.getType().isNullable()),
                false);
          }
          case TIME: {
            int value = (materializedExpr.getMajorType().getMode() == TypeProtos.DataMode.OPTIONAL) ?
                ((NullableTimeHolder) valueHolder).value :
                ((TimeHolder) valueHolder).value;
            LocalTime time = LocalDateTime.ofInstant(Instant.ofEpochMilli(value), ZoneOffset.UTC).toLocalTime();
            RelDataType type = typeFactory.createSqlType(SqlTypeName.TIME, newCall.getType().getPrecision());
            RelDataType typeWithNullability = typeFactory.createTypeWithNullability(type, newCall.getType().isNullable());
            TimeString timeString = new TimeString(time.getHour(), time.getMinute(), time.getSecond()).withNanos(time.getNano());
            return rexBuilder.makeLiteral(timeString, typeWithNullability, false);
          }
          case TIMESTAMP: {
            long value = (materializedExpr.getMajorType().getMode() == TypeProtos.DataMode.OPTIONAL) ?
                ((NullableTimeStampHolder) valueHolder).value :
                ((TimeStampHolder) valueHolder).value;
            RelDataType type = typeFactory.createSqlType(SqlTypeName.TIMESTAMP, newCall.getType().getPrecision());
            RelDataType typeWithNullability = typeFactory.createTypeWithNullability(type, newCall.getType().isNullable());
            return rexBuilder.makeLiteral(TimestampString.fromMillisSinceEpoch(value), typeWithNullability, false);
          }
          case INTERVALYEAR: {
            BigDecimal value = (materializedExpr.getMajorType().getMode() == TypeProtos.DataMode.OPTIONAL) ?
                BigDecimal.valueOf(((NullableIntervalYearHolder) valueHolder).value) :
                BigDecimal.valueOf(((IntervalYearHolder) valueHolder).value);
            return rexBuilder.makeLiteral(value,
                TypeInferenceUtils.createCalciteTypeWithNullability(typeFactory, SqlTypeName.INTERVAL_YEAR_MONTH, newCall.getType().isNullable()), false);
          }
          case INTERVALDAY: {
            int days;
            int milliseconds;
            if (materializedExpr.getMajorType().getMode() == TypeProtos.DataMode.OPTIONAL) {
              NullableIntervalDayHolder intervalDayOut = (NullableIntervalDayHolder) valueHolder;
              days = intervalDayOut.days;
              milliseconds = intervalDayOut.milliseconds;
            } else {
              IntervalDayHolder intervalDayOut = (IntervalDayHolder) valueHolder;
              days = intervalDayOut.days;
              milliseconds = intervalDayOut.milliseconds;
            }
            return rexBuilder.makeLiteral(
                BigDecimal.valueOf(days * (long) DateUtilities.daysToStandardMillis + milliseconds),
                TypeInferenceUtils.createCalciteTypeWithNullability(typeFactory, SqlTypeName.INTERVAL_DAY,
                    newCall.getType().isNullable()), false);
          }
          // The list of known unsupported types is used to trigger this behavior of re-using the input expression
          // before the expression is even attempted to be evaluated, this is just here as a last precaution a
          // as new types may be added in the future.
          default:
            logger.debug("Constant expression not folded due to return type {}, complete expression: {}",
                materializedExpr.getMajorType(),
                ExpressionStringBuilder.toString(materializedExpr));
            return newCall;
        }
      };

      reducedValues.add(literator.apply(output));
    }
  }