TestBlockBuilder.java
/*
* 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 com.facebook.presto.block;
import com.facebook.presto.common.PageBuilder;
import com.facebook.presto.common.block.Block;
import com.facebook.presto.common.block.BlockBuilder;
import com.facebook.presto.common.type.ArrayType;
import com.facebook.presto.common.type.Type;
import com.google.common.collect.ImmutableList;
import io.airlift.slice.Slice;
import io.airlift.slice.Slices;
import org.testng.SkipException;
import org.testng.annotations.Test;
import java.util.Collections;
import java.util.List;
import java.util.Optional;
import java.util.concurrent.atomic.AtomicBoolean;
import java.util.stream.IntStream;
import static com.facebook.presto.block.BlockAssertions.assertBlockEquals;
import static com.facebook.presto.common.block.ArrayBlock.fromElementBlock;
import static com.facebook.presto.common.type.BigintType.BIGINT;
import static com.facebook.presto.common.type.VarcharType.VARCHAR;
import static org.testng.Assert.assertEquals;
import static org.testng.Assert.assertNotEquals;
import static org.testng.Assert.assertTrue;
import static org.testng.Assert.fail;
public class TestBlockBuilder
{
@Test
public void testMultipleValuesWithNull()
{
BlockBuilder blockBuilder = BIGINT.createBlockBuilder(null, 10);
blockBuilder.appendNull();
BIGINT.writeLong(blockBuilder, 42);
blockBuilder.appendNull();
BIGINT.writeLong(blockBuilder, 42);
Block block = blockBuilder.build();
assertTrue(block.isNull(0));
assertEquals(BIGINT.getLong(block, 1), 42L);
assertTrue(block.isNull(2));
assertEquals(BIGINT.getLong(block, 3), 42L);
}
@Test
public void testNewBlockBuilderLike()
{
ArrayType longArrayType = new ArrayType(BIGINT);
ArrayType arrayType = new ArrayType(longArrayType);
List<Type> channels = ImmutableList.of(BIGINT, VARCHAR, arrayType);
PageBuilder pageBuilder = new PageBuilder(channels);
BlockBuilder bigintBlockBuilder = pageBuilder.getBlockBuilder(0);
BlockBuilder varcharBlockBuilder = pageBuilder.getBlockBuilder(1);
BlockBuilder arrayBlockBuilder = pageBuilder.getBlockBuilder(2);
for (int i = 0; i < 100; i++) {
BIGINT.writeLong(bigintBlockBuilder, i);
VARCHAR.writeSlice(varcharBlockBuilder, Slices.utf8Slice("test" + i));
Block longArrayBlock = new ArrayType(BIGINT)
.createBlockBuilder(null, 1)
.appendStructure(BIGINT.createBlockBuilder(null, 2).writeLong(i).closeEntry().writeLong(i * 2).closeEntry().build());
arrayBlockBuilder.appendStructure(longArrayBlock);
pageBuilder.declarePosition();
}
PageBuilder newPageBuilder = pageBuilder.newPageBuilderLike();
for (int i = 0; i < channels.size(); i++) {
assertEquals(newPageBuilder.getType(i), pageBuilder.getType(i));
// we should get new block builder instances
assertNotEquals(pageBuilder.getBlockBuilder(i), newPageBuilder.getBlockBuilder(i));
assertEquals(newPageBuilder.getBlockBuilder(i).getPositionCount(), 0);
assertTrue(newPageBuilder.getBlockBuilder(i).getRetainedSizeInBytes() < pageBuilder.getBlockBuilder(i).getRetainedSizeInBytes());
}
// Test newBlockBuilderLike with expectedEntries
BlockBuilder newBigintBlockBuilder = bigintBlockBuilder.newBlockBuilderLike(null, 200);
assertEquals(newBigintBlockBuilder.getPositionCount(), 0);
assertEquals(newBigintBlockBuilder.getRetainedSizeInBytes(), 80);
newBigintBlockBuilder.writeLong(0);
assertEquals(newBigintBlockBuilder.getPositionCount(), 1);
// Reserved 200 longs and booleans for nulls array
assertEquals(newBigintBlockBuilder.getRetainedSizeInBytes(), 1880);
BlockBuilder newVarcharBlockBuilder = varcharBlockBuilder.newBlockBuilderLike(null, 200);
assertEquals(newVarcharBlockBuilder.getPositionCount(), 0);
assertEquals(newVarcharBlockBuilder.getRetainedSizeInBytes(), 164);
newVarcharBlockBuilder.writeLong(0);
newVarcharBlockBuilder.closeEntry();
assertEquals(newVarcharBlockBuilder.getPositionCount(), 1);
// Reserved 200 varchars of average length 5.9, and 201 ints for offsets and 200 booleans for nulls
assertEquals(newVarcharBlockBuilder.getRetainedSizeInBytes(), 2360);
BlockBuilder newArrayBlockBuilder = arrayBlockBuilder.newBlockBuilderLike(null, 200);
assertEquals(newArrayBlockBuilder.getPositionCount(), 0);
assertEquals(newArrayBlockBuilder.getRetainedSizeInBytes(), 248);
newArrayBlockBuilder.appendStructure(fromElementBlock(1, Optional.empty(), IntStream.range(0, 2).toArray(), newBigintBlockBuilder.build()));
assertEquals(newArrayBlockBuilder.getPositionCount(), 1);
// Reserved 200 ARRAY(ARRAY(BIGINT)), and 201 ints for offsets and 200 booleans for nulls
assertEquals(newArrayBlockBuilder.getRetainedSizeInBytes(), 5848);
}
@Test
public void testNewBlockBuilderLikeForLargeBlockBuilder()
{
if (true) {
throw new SkipException("https://github.com/prestodb/presto/issues/15653 - Skipped because it OOMs");
}
List<Type> channels = ImmutableList.of(VARCHAR);
PageBuilder pageBuilder = new PageBuilder(channels);
BlockBuilder largeVarcharBlockBuilder = pageBuilder.getBlockBuilder(0);
// Construct a string of 64 * 16 = 2^11 bytes
Slice largeSlice = Slices.utf8Slice(String.join("", Collections.nCopies(64, "CowMonsterKing:)")));
// Write the string to the largeVarcharBlockBuilder for 2^20 times
for (int i = 0; i < 1_048_576; i++) {
VARCHAR.writeSlice(largeVarcharBlockBuilder, largeSlice);
pageBuilder.declarePosition();
}
BlockBuilder newVarcharBlockBuilder = largeVarcharBlockBuilder.newBlockBuilderLike(null, 1_048_576 * 8);
assertEquals(newVarcharBlockBuilder.getPositionCount(), 0);
assertEquals(newVarcharBlockBuilder.getRetainedSizeInBytes(), 164);
// We are not going to test real reservation here because allocating large amount of memory fails the Travis.
}
@Test
public void testGetPositions()
{
BlockBuilder blockBuilder = BIGINT.createFixedSizeBlockBuilder(5).appendNull().writeLong(42L).appendNull().writeLong(43L).appendNull();
int[] positions = new int[] {0, 1, 1, 1, 4};
// test getPositions for block builder
assertBlockEquals(BIGINT, blockBuilder.getPositions(positions, 0, positions.length), BIGINT.createFixedSizeBlockBuilder(5).appendNull().writeLong(42).writeLong(42).writeLong(42).appendNull().build());
assertBlockEquals(BIGINT, blockBuilder.getPositions(positions, 1, 4), BIGINT.createFixedSizeBlockBuilder(5).writeLong(42).writeLong(42).writeLong(42).appendNull().build());
assertBlockEquals(BIGINT, blockBuilder.getPositions(positions, 2, 1), BIGINT.createFixedSizeBlockBuilder(5).writeLong(42).build());
assertBlockEquals(BIGINT, blockBuilder.getPositions(positions, 0, 0), BIGINT.createFixedSizeBlockBuilder(5).build());
assertBlockEquals(BIGINT, blockBuilder.getPositions(positions, 1, 0), BIGINT.createFixedSizeBlockBuilder(5).build());
// out of range
assertInvalidGetPositions(blockBuilder, new int[] {-1}, 0, 1);
assertInvalidGetPositions(blockBuilder, new int[] {6}, 0, 1);
assertInvalidGetPositions(blockBuilder, new int[] {6}, 1, 1);
assertInvalidGetPositions(blockBuilder, new int[] {6}, -1, 1);
assertInvalidGetPositions(blockBuilder, new int[] {6}, 2, -1);
// test getPositions for block
Block block = blockBuilder.build();
assertBlockEquals(BIGINT, block.getPositions(positions, 0, positions.length), BIGINT.createFixedSizeBlockBuilder(5).appendNull().writeLong(42).writeLong(42).writeLong(42).appendNull().build());
assertBlockEquals(BIGINT, block.getPositions(positions, 1, 4), BIGINT.createFixedSizeBlockBuilder(5).writeLong(42).writeLong(42).writeLong(42).appendNull().build());
assertBlockEquals(BIGINT, block.getPositions(positions, 2, 1), BIGINT.createFixedSizeBlockBuilder(5).writeLong(42).build());
assertBlockEquals(BIGINT, block.getPositions(positions, 0, 0), BIGINT.createFixedSizeBlockBuilder(5).build());
assertBlockEquals(BIGINT, block.getPositions(positions, 1, 0), BIGINT.createFixedSizeBlockBuilder(5).build());
// out of range
assertInvalidGetPositions(block, new int[] {-1}, 0, 1);
assertInvalidGetPositions(block, new int[] {6}, 0, 1);
assertInvalidGetPositions(block, new int[] {6}, 1, 1);
assertInvalidGetPositions(block, new int[] {6}, -1, 1);
assertInvalidGetPositions(block, new int[] {6}, 2, -1);
// assert we should not copy ids
AtomicBoolean isIdentical = new AtomicBoolean(false);
block.getPositions(positions, 0, positions.length - 1).retainedBytesForEachPart((part, size) -> {
if (part == positions) {
isIdentical.set(true);
}
});
assertTrue(isIdentical.get());
}
private static void assertInvalidGetPositions(Block block, int[] positions, int offset, int length)
{
try {
block.getPositions(positions, offset, length).getLong(0);
fail("Expected to fail");
}
catch (IllegalArgumentException e) {
assertTrue(e.getMessage().startsWith("position is not valid"));
}
catch (IndexOutOfBoundsException e) {
assertTrue(e.getMessage().startsWith("Invalid offset"));
}
}
}