io.trino.plugin.hive.util.HiveBucketing Maven / Gradle / Ivy
/*
* Licensed 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 io.trino.plugin.hive.util;
import com.fasterxml.jackson.annotation.JsonCreator;
import com.fasterxml.jackson.annotation.JsonProperty;
import com.google.common.annotations.VisibleForTesting;
import com.google.common.collect.ImmutableList;
import com.google.common.collect.ImmutableSet;
import io.trino.plugin.hive.HiveBucketHandle;
import io.trino.plugin.hive.HiveBucketProperty;
import io.trino.plugin.hive.HiveColumnHandle;
import io.trino.plugin.hive.HiveTableHandle;
import io.trino.plugin.hive.HiveTimestampPrecision;
import io.trino.plugin.hive.HiveType;
import io.trino.plugin.hive.metastore.Column;
import io.trino.plugin.hive.metastore.SortingColumn;
import io.trino.plugin.hive.metastore.Table;
import io.trino.plugin.hive.type.ListTypeInfo;
import io.trino.plugin.hive.type.MapTypeInfo;
import io.trino.plugin.hive.type.PrimitiveCategory;
import io.trino.plugin.hive.type.PrimitiveTypeInfo;
import io.trino.plugin.hive.type.TypeInfo;
import io.trino.spi.Page;
import io.trino.spi.StandardErrorCode;
import io.trino.spi.TrinoException;
import io.trino.spi.connector.ColumnHandle;
import io.trino.spi.connector.ConnectorSession;
import io.trino.spi.predicate.Domain;
import io.trino.spi.predicate.NullableValue;
import io.trino.spi.predicate.TupleDomain;
import io.trino.spi.predicate.ValueSet;
import io.trino.spi.type.TypeManager;
import java.util.HashMap;
import java.util.HashSet;
import java.util.List;
import java.util.Map;
import java.util.Objects;
import java.util.Optional;
import java.util.Set;
import java.util.stream.Collectors;
import static com.google.common.collect.ImmutableList.toImmutableList;
import static com.google.common.collect.Lists.cartesianProduct;
import static io.trino.hive.thrift.metastore.hive_metastoreConstants.TABLE_BUCKETING_VERSION;
import static io.trino.plugin.hive.HiveColumnHandle.BUCKET_COLUMN_NAME;
import static io.trino.plugin.hive.HiveErrorCode.HIVE_INVALID_METADATA;
import static io.trino.plugin.hive.HiveSessionProperties.getTimestampPrecision;
import static io.trino.plugin.hive.util.HiveBucketing.BucketingVersion.BUCKETING_V1;
import static io.trino.plugin.hive.util.HiveBucketing.BucketingVersion.BUCKETING_V2;
import static io.trino.plugin.hive.util.HiveUtil.SPARK_TABLE_PROVIDER_KEY;
import static io.trino.plugin.hive.util.HiveUtil.getRegularColumnHandles;
import static java.lang.String.format;
import static java.util.Map.Entry;
import static java.util.function.Function.identity;
public final class HiveBucketing
{
public enum BucketingVersion
{
BUCKETING_V1(1) {
@Override
int getBucketHashCode(List types, Object[] values)
{
return HiveBucketingV1.getBucketHashCode(types, values);
}
@Override
int getBucketHashCode(List types, Page page, int position)
{
return HiveBucketingV1.getBucketHashCode(types, page, position);
}
},
BUCKETING_V2(2) {
@Override
int getBucketHashCode(List types, Object[] values)
{
return HiveBucketingV2.getBucketHashCode(types, values);
}
@Override
int getBucketHashCode(List types, Page page, int position)
{
return HiveBucketingV2.getBucketHashCode(types, page, position);
}
},
/**/;
private final int version;
BucketingVersion(int version)
{
this.version = version;
}
public int getVersion()
{
return version;
}
abstract int getBucketHashCode(List types, Object[] values);
abstract int getBucketHashCode(List types, Page page, int position);
}
private static final long BUCKETS_EXPLORATION_LIMIT_FACTOR = 4;
private static final long BUCKETS_EXPLORATION_GUARANTEED_LIMIT = 1000;
private HiveBucketing() {}
public static int getHiveBucket(BucketingVersion bucketingVersion, int bucketCount, List types, Page page, int position)
{
return getBucketNumber(bucketingVersion.getBucketHashCode(types, page, position), bucketCount);
}
@VisibleForTesting
static Optional> getHiveBuckets(BucketingVersion bucketingVersion, int bucketCount, List types, List> values)
{
long explorationCount;
try {
// explorationCount is the number of combinations of discrete values allowed for bucketing columns.
// After computing the bucket for every combination, we get a complete set of buckets that need to be read.
explorationCount = values.stream()
.mapToLong(List::size)
.reduce(1, Math::multiplyExact);
}
catch (ArithmeticException e) {
return Optional.empty();
}
// explorationLimit is the maximum number of combinations for which the bucket numbers will be computed.
// If the number of combinations highly exceeds the bucket count, then probably all buckets would be hit.
// In such case, the bucket filter isn't created and all buckets will be read.
// The threshold is set to bucketCount * BUCKETS_EXPLORATION_LIMIT_FACTOR.
// The threshold doesn't apply if the number of combinations is low, that is
// within BUCKETS_EXPLORATION_GUARANTEED_LIMIT.
long explorationLimit = Math.max(bucketCount * BUCKETS_EXPLORATION_LIMIT_FACTOR, BUCKETS_EXPLORATION_GUARANTEED_LIMIT);
if (explorationCount > explorationLimit) {
return Optional.empty();
}
Set buckets = new HashSet<>();
for (List combination : cartesianProduct(values)) {
buckets.add(getBucketNumber(bucketingVersion.getBucketHashCode(types, combination.stream().map(NullableValue::getValue).toArray()), bucketCount));
if (buckets.size() >= bucketCount) {
return Optional.empty();
}
}
return Optional.of(ImmutableSet.copyOf(buckets));
}
@VisibleForTesting
static int getBucketNumber(int hashCode, int bucketCount)
{
return (hashCode & Integer.MAX_VALUE) % bucketCount;
}
public static Optional getHiveBucketHandle(ConnectorSession session, Table table, TypeManager typeManager)
{
if (table.getParameters().containsKey(SPARK_TABLE_PROVIDER_KEY)) {
return Optional.empty();
}
Optional hiveBucketProperty = table.getStorage().getBucketProperty();
if (hiveBucketProperty.isEmpty()) {
return Optional.empty();
}
if (!isSupportedBucketing(table)) {
return Optional.empty();
}
HiveTimestampPrecision timestampPrecision = getTimestampPrecision(session);
Map map = getRegularColumnHandles(table, typeManager, timestampPrecision).stream()
.collect(Collectors.toMap(HiveColumnHandle::getName, identity()));
ImmutableList.Builder bucketColumns = ImmutableList.builder();
for (String bucketColumnName : hiveBucketProperty.get().bucketedBy()) {
HiveColumnHandle bucketColumnHandle = map.get(bucketColumnName);
if (bucketColumnHandle == null) {
throw new TrinoException(
HIVE_INVALID_METADATA,
format("Table '%s.%s' is bucketed on non-existent column '%s'", table.getDatabaseName(), table.getTableName(), bucketColumnName));
}
bucketColumns.add(bucketColumnHandle);
}
BucketingVersion bucketingVersion = getBucketingVersion(table.getParameters());
int bucketCount = hiveBucketProperty.get().bucketCount();
List sortedBy = hiveBucketProperty.get().sortedBy();
return Optional.of(new HiveBucketHandle(bucketColumns.build(), bucketingVersion, bucketCount, bucketCount, sortedBy));
}
public static Optional getHiveBucketFilter(HiveTableHandle hiveTable, TupleDomain effectivePredicate)
{
if (hiveTable.getBucketHandle().isEmpty()) {
return Optional.empty();
}
HiveBucketProperty hiveBucketProperty = hiveTable.getBucketHandle().get().toTableBucketProperty();
List dataColumns = hiveTable.getDataColumns().stream()
.collect(toImmutableList());
Optional
© 2015 - 2025 Weber Informatics LLC | Privacy Policy