io.trino.plugin.hive.InternalHiveSplit 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;
import com.google.common.collect.ImmutableList;
import io.trino.plugin.hive.HiveSplit.BucketConversion;
import io.trino.plugin.hive.HiveSplit.BucketValidation;
import io.trino.spi.HostAddress;
import org.openjdk.jol.info.ClassLayout;
import javax.annotation.concurrent.NotThreadSafe;
import java.util.List;
import java.util.Optional;
import java.util.OptionalInt;
import java.util.Properties;
import java.util.function.BooleanSupplier;
import java.util.function.Supplier;
import static com.google.common.base.MoreObjects.toStringHelper;
import static com.google.common.base.Preconditions.checkArgument;
import static com.google.common.base.Preconditions.checkState;
import static com.google.common.base.Verify.verify;
import static io.airlift.slice.SizeOf.estimatedSizeOf;
import static java.lang.Math.toIntExact;
import static java.util.Objects.requireNonNull;
@NotThreadSafe
public class InternalHiveSplit
{
private static final int INSTANCE_SIZE = ClassLayout.parseClass(InternalHiveSplit.class).instanceSize() +
ClassLayout.parseClass(String.class).instanceSize() +
ClassLayout.parseClass(Properties.class).instanceSize() +
ClassLayout.parseClass(String.class).instanceSize() +
ClassLayout.parseClass(OptionalInt.class).instanceSize();
private final String path;
private final long end;
private final long estimatedFileSize;
private final long fileModifiedTime;
private final Properties schema;
private final List partitionKeys;
private final List blocks;
private final String partitionName;
private final OptionalInt readBucketNumber;
private final OptionalInt tableBucketNumber;
// This supplier returns an unused statementId, to guarantee that created split
// files do not collide. Successive calls return the the next sequential integer,
// starting with zero.
private final Supplier statementIdSupplier;
private final boolean splittable;
private final boolean forceLocalScheduling;
private final TableToPartitionMapping tableToPartitionMapping;
private final Optional bucketConversion;
private final Optional bucketValidation;
private final boolean s3SelectPushdownEnabled;
private final Optional acidInfo;
private final BooleanSupplier partitionMatchSupplier;
private long start;
private int currentBlockIndex;
private int statementId;
public InternalHiveSplit(
String partitionName,
String path,
long start,
long end,
long estimatedFileSize,
long fileModifiedTime,
Properties schema,
List partitionKeys,
List blocks,
OptionalInt readBucketNumber,
OptionalInt tableBucketNumber,
Supplier statementIdSupplier,
boolean splittable,
boolean forceLocalScheduling,
TableToPartitionMapping tableToPartitionMapping,
Optional bucketConversion,
Optional bucketValidation,
boolean s3SelectPushdownEnabled,
Optional acidInfo,
BooleanSupplier partitionMatchSupplier)
{
checkArgument(start >= 0, "start must be positive");
checkArgument(end >= 0, "length must be positive");
checkArgument(estimatedFileSize >= 0, "fileSize must be positive");
requireNonNull(partitionName, "partitionName is null");
requireNonNull(path, "path is null");
requireNonNull(schema, "schema is null");
requireNonNull(partitionKeys, "partitionKeys is null");
requireNonNull(blocks, "blocks is null");
requireNonNull(readBucketNumber, "readBucketNumber is null");
requireNonNull(tableBucketNumber, "tableBucketNumber is null");
requireNonNull(statementIdSupplier, "statementIdSupplier is null");
requireNonNull(tableToPartitionMapping, "tableToPartitionMapping is null");
requireNonNull(bucketConversion, "bucketConversion is null");
requireNonNull(bucketValidation, "bucketValidation is null");
requireNonNull(acidInfo, "acidInfo is null");
requireNonNull(partitionMatchSupplier, "partitionMatchSupplier is null");
this.partitionName = partitionName;
this.path = path;
this.start = start;
this.end = end;
this.estimatedFileSize = estimatedFileSize;
this.fileModifiedTime = fileModifiedTime;
this.schema = schema;
this.partitionKeys = ImmutableList.copyOf(partitionKeys);
this.blocks = ImmutableList.copyOf(blocks);
this.readBucketNumber = readBucketNumber;
this.tableBucketNumber = tableBucketNumber;
this.statementIdSupplier = statementIdSupplier;
this.statementId = statementIdSupplier.get();
this.splittable = splittable;
this.forceLocalScheduling = forceLocalScheduling;
this.tableToPartitionMapping = tableToPartitionMapping;
this.bucketConversion = bucketConversion;
this.bucketValidation = bucketValidation;
this.s3SelectPushdownEnabled = s3SelectPushdownEnabled;
this.acidInfo = acidInfo;
this.partitionMatchSupplier = partitionMatchSupplier;
}
public String getPath()
{
return path;
}
public long getStart()
{
return start;
}
public long getEnd()
{
return end;
}
public long getEstimatedFileSize()
{
return estimatedFileSize;
}
public long getFileModifiedTime()
{
return fileModifiedTime;
}
public boolean isS3SelectPushdownEnabled()
{
return s3SelectPushdownEnabled;
}
public Properties getSchema()
{
return schema;
}
public List getPartitionKeys()
{
return partitionKeys;
}
public String getPartitionName()
{
return partitionName;
}
public OptionalInt getReadBucketNumber()
{
return readBucketNumber;
}
public OptionalInt getTableBucketNumber()
{
return tableBucketNumber;
}
public int getStatementId()
{
return statementId;
}
public boolean isSplittable()
{
return splittable;
}
public boolean isForceLocalScheduling()
{
return forceLocalScheduling;
}
public TableToPartitionMapping getTableToPartitionMapping()
{
return tableToPartitionMapping;
}
public Optional getBucketConversion()
{
return bucketConversion;
}
public Optional getBucketValidation()
{
return bucketValidation;
}
public InternalHiveBlock currentBlock()
{
checkState(!isDone(), "All blocks have been consumed");
return blocks.get(currentBlockIndex);
}
public boolean isDone()
{
return currentBlockIndex == blocks.size();
}
public void increaseStart(long value)
{
statementId = statementIdSupplier.get();
start += value;
if (start == currentBlock().getEnd()) {
currentBlockIndex++;
if (isDone()) {
return;
}
verify(start == currentBlock().getStart());
}
}
public int getEstimatedSizeInBytes()
{
long result = INSTANCE_SIZE +
estimatedSizeOf(path) +
estimatedSizeOf(partitionKeys, HivePartitionKey::getEstimatedSizeInBytes) +
estimatedSizeOf(blocks, InternalHiveBlock::getEstimatedSizeInBytes) +
estimatedSizeOf(partitionName) +
tableToPartitionMapping.getEstimatedSizeInBytes();
return toIntExact(result);
}
public Optional getAcidInfo()
{
return acidInfo;
}
public BooleanSupplier getPartitionMatchSupplier()
{
return partitionMatchSupplier;
}
@Override
public String toString()
{
return toStringHelper(this)
.add("path", path)
.add("start", start)
.add("end", end)
.add("estimatedFileSize", estimatedFileSize)
.toString();
}
public static class InternalHiveBlock
{
private static final int INSTANCE_SIZE = ClassLayout.parseClass(InternalHiveBlock.class).instanceSize();
private static final int HOST_ADDRESS_INSTANCE_SIZE = ClassLayout.parseClass(HostAddress.class).instanceSize() +
ClassLayout.parseClass(String.class).instanceSize();
private final long start;
private final long end;
private final List addresses;
public InternalHiveBlock(long start, long end, List addresses)
{
checkArgument(start <= end, "block end cannot be before block start");
this.start = start;
this.end = end;
this.addresses = ImmutableList.copyOf(addresses);
}
public long getStart()
{
return start;
}
public long getEnd()
{
return end;
}
public List getAddresses()
{
return addresses;
}
public long getEstimatedSizeInBytes()
{
return INSTANCE_SIZE + estimatedSizeOf(addresses, address -> HOST_ADDRESS_INSTANCE_SIZE + estimatedSizeOf(address.getHostText()));
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy