io.trino.parquet.writer.repdef.DefLevelWriterProviders 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.parquet.writer.repdef;
import io.trino.parquet.writer.valuewriter.ColumnDescriptorValuesWriter;
import io.trino.spi.block.ArrayBlock;
import io.trino.spi.block.Block;
import io.trino.spi.block.ColumnarArray;
import io.trino.spi.block.ColumnarMap;
import io.trino.spi.block.MapBlock;
import io.trino.spi.block.RowBlock;
import java.util.Optional;
import static com.google.common.base.Preconditions.checkArgument;
import static java.lang.String.format;
import static java.util.Objects.requireNonNull;
public class DefLevelWriterProviders
{
private DefLevelWriterProviders() {}
public static DefLevelWriterProvider of(Block block, int maxDefinitionLevel)
{
if (block.getUnderlyingValueBlock() instanceof RowBlock) {
return new RowDefLevelWriterProvider(block, maxDefinitionLevel);
}
return new PrimitiveDefLevelWriterProvider(block, maxDefinitionLevel);
}
public static DefLevelWriterProvider of(ColumnarArray columnarArray, int maxDefinitionLevel)
{
return new ColumnArrayDefLevelWriterProvider(columnarArray, maxDefinitionLevel);
}
public static DefLevelWriterProvider of(ColumnarMap columnarMap, int maxDefinitionLevel)
{
return new ColumnMapDefLevelWriterProvider(columnarMap, maxDefinitionLevel);
}
static class PrimitiveDefLevelWriterProvider
implements DefLevelWriterProvider
{
private final Block block;
private final int maxDefinitionLevel;
PrimitiveDefLevelWriterProvider(Block block, int maxDefinitionLevel)
{
this.block = requireNonNull(block, "block is null");
this.maxDefinitionLevel = maxDefinitionLevel;
checkArgument(!(block.getUnderlyingValueBlock() instanceof RowBlock), "block is a row block");
checkArgument(!(block.getUnderlyingValueBlock() instanceof ArrayBlock), "block is an array block");
checkArgument(!(block.getUnderlyingValueBlock() instanceof MapBlock), "block is a map block");
}
@Override
public DefinitionLevelWriter getDefinitionLevelWriter(Optional nestedWriter, ColumnDescriptorValuesWriter encoder)
{
checkArgument(nestedWriter.isEmpty(), "nestedWriter should be empty for primitive definition level writer");
return new DefinitionLevelWriter()
{
private int offset;
@Override
public ValuesCount writeDefinitionLevels()
{
return writeDefinitionLevels(block.getPositionCount());
}
@Override
public ValuesCount writeDefinitionLevels(int positionsCount)
{
checkValidPosition(offset, positionsCount, block.getPositionCount());
int nonNullsCount = 0;
if (!block.mayHaveNull()) {
encoder.writeRepeatInteger(maxDefinitionLevel, positionsCount);
nonNullsCount = positionsCount;
}
else {
for (int position = offset; position < offset + positionsCount; position++) {
int isNull = block.isNull(position) ? 1 : 0;
encoder.writeInteger(maxDefinitionLevel - isNull);
nonNullsCount += isNull ^ 1;
}
}
offset += positionsCount;
return new ValuesCount(positionsCount, nonNullsCount);
}
};
}
}
static class RowDefLevelWriterProvider
implements DefLevelWriterProvider
{
private final Block block;
private final int maxDefinitionLevel;
RowDefLevelWriterProvider(Block block, int maxDefinitionLevel)
{
this.block = requireNonNull(block, "block is null");
this.maxDefinitionLevel = maxDefinitionLevel;
checkArgument(block.getUnderlyingValueBlock() instanceof RowBlock, "block is not a row block");
}
@Override
public DefinitionLevelWriter getDefinitionLevelWriter(Optional nestedWriterOptional, ColumnDescriptorValuesWriter encoder)
{
checkArgument(nestedWriterOptional.isPresent(), "nestedWriter should be present for column row definition level writer");
return new DefinitionLevelWriter()
{
private final DefinitionLevelWriter nestedWriter = nestedWriterOptional.orElseThrow();
private int offset;
@Override
public ValuesCount writeDefinitionLevels()
{
return writeDefinitionLevels(block.getPositionCount());
}
@Override
public ValuesCount writeDefinitionLevels(int positionsCount)
{
checkValidPosition(offset, positionsCount, block.getPositionCount());
if (!block.mayHaveNull()) {
offset += positionsCount;
return nestedWriter.writeDefinitionLevels(positionsCount);
}
int maxDefinitionValuesCount = 0;
int totalValuesCount = 0;
for (int position = offset; position < offset + positionsCount; ) {
if (block.isNull(position)) {
encoder.writeInteger(maxDefinitionLevel - 1);
totalValuesCount++;
position++;
}
else {
int consecutiveNonNullsCount = 1;
position++;
while (position < offset + positionsCount && !block.isNull(position)) {
position++;
consecutiveNonNullsCount++;
}
ValuesCount valuesCount = nestedWriter.writeDefinitionLevels(consecutiveNonNullsCount);
maxDefinitionValuesCount += valuesCount.maxDefinitionLevelValuesCount();
totalValuesCount += valuesCount.totalValuesCount();
}
}
offset += positionsCount;
return new ValuesCount(totalValuesCount, maxDefinitionValuesCount);
}
};
}
}
static class ColumnMapDefLevelWriterProvider
implements DefLevelWriterProvider
{
private final ColumnarMap columnarMap;
private final int maxDefinitionLevel;
ColumnMapDefLevelWriterProvider(ColumnarMap columnarMap, int maxDefinitionLevel)
{
this.columnarMap = requireNonNull(columnarMap, "columnarMap is null");
this.maxDefinitionLevel = maxDefinitionLevel;
}
@Override
public DefinitionLevelWriter getDefinitionLevelWriter(Optional nestedWriterOptional, ColumnDescriptorValuesWriter encoder)
{
checkArgument(nestedWriterOptional.isPresent(), "nestedWriter should be present for column map definition level writer");
return new DefinitionLevelWriter()
{
private final DefinitionLevelWriter nestedWriter = nestedWriterOptional.orElseThrow();
private int offset;
@Override
public ValuesCount writeDefinitionLevels()
{
return writeDefinitionLevels(columnarMap.getPositionCount());
}
@Override
public ValuesCount writeDefinitionLevels(int positionsCount)
{
checkValidPosition(offset, positionsCount, columnarMap.getPositionCount());
int maxDefinitionValuesCount = 0;
int totalValuesCount = 0;
if (!columnarMap.mayHaveNull()) {
for (int position = offset; position < offset + positionsCount; ) {
int mapLength = columnarMap.getEntryCount(position);
if (mapLength == 0) {
encoder.writeInteger(maxDefinitionLevel - 1);
totalValuesCount++;
position++;
}
else {
int consecutiveNonEmptyArrayLength = mapLength;
position++;
while (position < offset + positionsCount) {
mapLength = columnarMap.getEntryCount(position);
if (mapLength == 0) {
break;
}
position++;
consecutiveNonEmptyArrayLength += mapLength;
}
ValuesCount valuesCount = nestedWriter.writeDefinitionLevels(consecutiveNonEmptyArrayLength);
maxDefinitionValuesCount += valuesCount.maxDefinitionLevelValuesCount();
totalValuesCount += valuesCount.totalValuesCount();
}
}
}
else {
for (int position = offset; position < offset + positionsCount; position++) {
if (columnarMap.isNull(position)) {
encoder.writeInteger(maxDefinitionLevel - 2);
totalValuesCount++;
continue;
}
int mapLength = columnarMap.getEntryCount(position);
if (mapLength == 0) {
encoder.writeInteger(maxDefinitionLevel - 1);
totalValuesCount++;
}
else {
ValuesCount valuesCount = nestedWriter.writeDefinitionLevels(mapLength);
maxDefinitionValuesCount += valuesCount.maxDefinitionLevelValuesCount();
totalValuesCount += valuesCount.totalValuesCount();
}
}
}
offset += positionsCount;
return new ValuesCount(totalValuesCount, maxDefinitionValuesCount);
}
};
}
}
static class ColumnArrayDefLevelWriterProvider
implements DefLevelWriterProvider
{
private final ColumnarArray columnarArray;
private final int maxDefinitionLevel;
ColumnArrayDefLevelWriterProvider(ColumnarArray columnarArray, int maxDefinitionLevel)
{
this.columnarArray = requireNonNull(columnarArray, "columnarArray is null");
this.maxDefinitionLevel = maxDefinitionLevel;
}
@Override
public DefinitionLevelWriter getDefinitionLevelWriter(Optional nestedWriterOptional, ColumnDescriptorValuesWriter encoder)
{
checkArgument(nestedWriterOptional.isPresent(), "nestedWriter should be present for column map definition level writer");
return new DefinitionLevelWriter()
{
private final DefinitionLevelWriter nestedWriter = nestedWriterOptional.orElseThrow();
private int offset;
@Override
public ValuesCount writeDefinitionLevels()
{
return writeDefinitionLevels(columnarArray.getPositionCount());
}
@Override
public ValuesCount writeDefinitionLevels(int positionsCount)
{
checkValidPosition(offset, positionsCount, columnarArray.getPositionCount());
int maxDefinitionValuesCount = 0;
int totalValuesCount = 0;
if (!columnarArray.mayHaveNull()) {
for (int position = offset; position < offset + positionsCount; ) {
int arrayLength = columnarArray.getLength(position);
if (arrayLength == 0) {
encoder.writeInteger(maxDefinitionLevel - 1);
totalValuesCount++;
position++;
}
else {
int consecutiveNonEmptyArrayLength = arrayLength;
position++;
while (position < offset + positionsCount) {
arrayLength = columnarArray.getLength(position);
if (arrayLength == 0) {
break;
}
position++;
consecutiveNonEmptyArrayLength += arrayLength;
}
ValuesCount valuesCount = nestedWriter.writeDefinitionLevels(consecutiveNonEmptyArrayLength);
maxDefinitionValuesCount += valuesCount.maxDefinitionLevelValuesCount();
totalValuesCount += valuesCount.totalValuesCount();
}
}
}
else {
for (int position = offset; position < offset + positionsCount; position++) {
if (columnarArray.isNull(position)) {
encoder.writeInteger(maxDefinitionLevel - 2);
totalValuesCount++;
continue;
}
int arrayLength = columnarArray.getLength(position);
if (arrayLength == 0) {
encoder.writeInteger(maxDefinitionLevel - 1);
totalValuesCount++;
}
else {
ValuesCount valuesCount = nestedWriter.writeDefinitionLevels(arrayLength);
maxDefinitionValuesCount += valuesCount.maxDefinitionLevelValuesCount();
totalValuesCount += valuesCount.totalValuesCount();
}
}
}
offset += positionsCount;
return new ValuesCount(totalValuesCount, maxDefinitionValuesCount);
}
};
}
}
private static void checkValidPosition(int offset, int positionsCount, int totalPositionsCount)
{
if (offset < 0 || positionsCount < 0 || offset + positionsCount > totalPositionsCount) {
throw new IndexOutOfBoundsException(format("Invalid offset %s and positionsCount %s in block with %s positions", offset, positionsCount, totalPositionsCount));
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy