Use the whole frame when writing rows. (#17094)

* Use the whole frame when writing rows.

This patch makes the following adjustments to enable writing larger
single rows to frames:

1) RowBasedFrameWriter: Max out allocation size on the final doubling.
   i.e., if the final allocation "naturally" would be 1 MiB but the
   max frame size is 900 KiB, use 900 KiB rather than failing the 1 MiB
   allocation.

2) AppendableMemory: In reserveAdditional, release the last block if it
   is empty. This eliminates waste when a frame writer uses a
   successive-doubling approach to find the right allocation size.

3) ArenaMemoryAllocator: Reclaim memory from the last allocation when
   the last allocation is closed.

Prior to these changes, a single row could be much smaller than the
frame size and still fail to be added to the frame.

* Style.

* Fix test.
This commit is contained in:
Gian Merlino 2024-09-19 00:42:03 -07:00 committed by GitHub
parent b9a4c73e52
commit 3d45f9829c
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
9 changed files with 347 additions and 90 deletions

View File

@ -1848,7 +1848,7 @@ public class MSQWindowTest extends MSQTestBase
.setSql( .setSql(
"select cityName, added, SUM(added) OVER () cc from wikipedia") "select cityName, added, SUM(added) OVER () cc from wikipedia")
.setQueryContext(customContext) .setQueryContext(customContext)
.setExpectedMSQFault(new TooManyRowsInAWindowFault(15921, 200)) .setExpectedMSQFault(new TooManyRowsInAWindowFault(15922, 200))
.verifyResults(); .verifyResults();
} }

View File

@ -54,10 +54,10 @@ public class AppendableMemory implements Closeable
// One holder for every Memory we've allocated. // One holder for every Memory we've allocated.
private final List<ResourceHolder<WritableMemory>> blockHolders = new ArrayList<>(); private final List<ResourceHolder<WritableMemory>> blockHolders = new ArrayList<>();
// The amount of space that has been used from each Memory block. Same length as "memoryHolders". // The amount of space that has been used from each Memory block. Same length as "blockHolders".
private final IntList limits = new IntArrayList(); private final IntList limits = new IntArrayList();
// The global starting position for each Memory block (blockNumber -> position). Same length as "memoryHolders". // The global starting position for each Memory block (blockNumber -> position). Same length as "blockHolders".
private final LongArrayList globalStartPositions = new LongArrayList(); private final LongArrayList globalStartPositions = new LongArrayList();
// Whether the blocks we've allocated are "packed"; meaning all non-final block limits equal the allocationSize. // Whether the blocks we've allocated are "packed"; meaning all non-final block limits equal the allocationSize.
@ -104,6 +104,36 @@ public class AppendableMemory implements Closeable
return cursor; return cursor;
} }
/**
* Maximum number that can be successfully passed to {@link #reserveAdditional(int)}.
*/
public int availableToReserve()
{
final int currentBlockIdx = currentBlockNumber();
final long availableInCurrentBlock;
final boolean currentBlockIsEmpty;
if (currentBlockIdx < 0) {
availableInCurrentBlock = 0;
currentBlockIsEmpty = false;
} else {
final int usedInCurrentBlock = limits.getInt(currentBlockIdx);
availableInCurrentBlock = blockHolders.get(currentBlockIdx).get().getCapacity() - usedInCurrentBlock;
currentBlockIsEmpty = usedInCurrentBlock == 0;
}
// If currentBlockIsEmpty, add availableInCurrentBlock to account for reclamation in reclaimLastBlockIfEmpty().
final long availableInAllocator = allocator.available() + (currentBlockIsEmpty ? availableInCurrentBlock : 0);
return (int) Math.min(
Integer.MAX_VALUE,
Math.max(
availableInAllocator,
availableInCurrentBlock
)
);
}
/** /**
* Ensure that at least "bytes" amount of space is available after the cursor. Allocates a new block if needed. * Ensure that at least "bytes" amount of space is available after the cursor. Allocates a new block if needed.
* Note: the amount of bytes is guaranteed to be in a *single* block. * Note: the amount of bytes is guaranteed to be in a *single* block.
@ -126,11 +156,13 @@ public class AppendableMemory implements Closeable
return true; return true;
} }
releaseLastBlockIfEmpty();
if (bytes > allocator.available()) { if (bytes > allocator.available()) {
return false; return false;
} }
final int idx = blockHolders.size() - 1; final int idx = currentBlockNumber();
if (idx < 0 || bytes + limits.getInt(idx) > blockHolders.get(idx).get().getCapacity()) { if (idx < 0 || bytes + limits.getInt(idx) > blockHolders.get(idx).get().getCapacity()) {
// Allocation needed. // Allocation needed.
@ -228,6 +260,9 @@ public class AppendableMemory implements Closeable
cursor.set(currentBlockMemory, newLimit, currentBlockMemory.getCapacity() - newLimit); cursor.set(currentBlockMemory, newLimit, currentBlockMemory.getCapacity() - newLimit);
} }
/**
* Current used size, in bytes.
*/
public long size() public long size()
{ {
long sz = 0; long sz = 0;
@ -295,12 +330,21 @@ public class AppendableMemory implements Closeable
cursor.set(blockMemory, 0, blockMemory.getCapacity()); cursor.set(blockMemory, 0, blockMemory.getCapacity());
} }
private int currentBlockNumber() private void releaseLastBlockIfEmpty()
{ {
if (blockHolders.isEmpty()) { final int lastBlockNumber = currentBlockNumber();
return NO_BLOCK; if (lastBlockNumber != NO_BLOCK && limits.getInt(lastBlockNumber) == 0) {
} else { blockHolders.remove(lastBlockNumber).close();
return blockHolders.size() - 1; limits.removeInt(lastBlockNumber);
} }
} }
/**
* Returns the index into {@link #blockHolders} and {@link #limits} of the current block, or {@link #NO_BLOCK}
* if there are no blocks.
*/
private int currentBlockNumber()
{
return blockHolders.size() - 1;
}
} }

View File

@ -37,6 +37,7 @@ public class ArenaMemoryAllocator implements MemoryAllocator
private final WritableMemory arena; private final WritableMemory arena;
private long allocations = 0; private long allocations = 0;
private long position = 0; private long position = 0;
private WritableMemory lastAllocation;
private ArenaMemoryAllocator(WritableMemory arena) private ArenaMemoryAllocator(WritableMemory arena)
{ {
@ -64,20 +65,23 @@ public class ArenaMemoryAllocator implements MemoryAllocator
@Override @Override
public Optional<ResourceHolder<WritableMemory>> allocate(final long size) public Optional<ResourceHolder<WritableMemory>> allocate(final long size)
{ {
if (position + size < arena.getCapacity()) { if (position + size <= arena.getCapacity()) {
final long start = position; final long start = position;
allocations++; allocations++;
position += size; position += size;
final WritableMemory memory = arena.writableRegion(start, size, ByteOrder.LITTLE_ENDIAN);
lastAllocation = memory;
return Optional.of( return Optional.of(
new ResourceHolder<WritableMemory>() new ResourceHolder<WritableMemory>()
{ {
private WritableMemory memory = arena.writableRegion(start, size, ByteOrder.LITTLE_ENDIAN); boolean closed;
@Override @Override
public WritableMemory get() public WritableMemory get()
{ {
if (memory == null) { if (closed) {
throw new ISE("Already closed"); throw new ISE("Already closed");
} }
@ -87,10 +91,21 @@ public class ArenaMemoryAllocator implements MemoryAllocator
@Override @Override
public void close() public void close()
{ {
memory = null; if (closed) {
return;
}
closed = true;
//noinspection ObjectEquality
if (memory == lastAllocation) {
// Last allocation closed; decrement position to enable partial arena reuse.
position -= memory.getCapacity();
lastAllocation = null;
}
if (--allocations == 0) { if (--allocations == 0) {
// All allocations closed; reset position to enable arena reuse. // All allocations closed; reset position to enable full arena reuse.
position = 0; position = 0;
} }
} }

View File

@ -77,8 +77,10 @@ public class HeapMemoryAllocator implements MemoryAllocator
@Override @Override
public void close() public void close()
{ {
memory = null; if (memory != null) {
bytesAllocated -= size; memory = null;
bytesAllocated -= size;
}
} }
} }
); );

View File

@ -23,6 +23,7 @@ import com.google.common.base.Throwables;
import com.google.common.primitives.Ints; import com.google.common.primitives.Ints;
import org.apache.datasketches.memory.Memory; import org.apache.datasketches.memory.Memory;
import org.apache.datasketches.memory.WritableMemory; import org.apache.datasketches.memory.WritableMemory;
import org.apache.druid.error.DruidException;
import org.apache.druid.frame.Frame; import org.apache.druid.frame.Frame;
import org.apache.druid.frame.FrameType; import org.apache.druid.frame.FrameType;
import org.apache.druid.frame.allocation.AppendableMemory; import org.apache.druid.frame.allocation.AppendableMemory;
@ -313,10 +314,22 @@ public class RowBasedFrameWriter implements FrameWriter
// Reset to beginning of loop. // Reset to beginning of loop.
i = -1; i = -1;
final int priorAllocation = BASE_DATA_ALLOCATION_SIZE * reserveMultiple;
// Try again with a bigger allocation. // Try again with a bigger allocation.
reserveMultiple *= 2; reserveMultiple *= 2;
if (!dataMemory.reserveAdditional(Ints.checkedCast((long) BASE_DATA_ALLOCATION_SIZE * reserveMultiple))) { final int nextAllocation = Math.min(
dataMemory.availableToReserve(),
Ints.checkedCast((long) BASE_DATA_ALLOCATION_SIZE * reserveMultiple)
);
if (nextAllocation > priorAllocation) {
if (!dataMemory.reserveAdditional(nextAllocation)) {
// Shouldn't see this unless availableToReserve lied to us.
throw DruidException.defensive("Unexpected failure of dataMemory.reserveAdditional");
}
} else {
return false; return false;
} }

View File

@ -0,0 +1,29 @@
/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you 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 org.apache.druid.frame.allocation;
public class ArenaMemoryAllocatorTest extends BaseMemoryAllocatorTest
{
@Override
protected MemoryAllocator makeAllocator(int capacity)
{
return ArenaMemoryAllocator.createOnHeap(capacity);
}
}

View File

@ -0,0 +1,166 @@
/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you 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 org.apache.druid.frame.allocation;
import org.apache.datasketches.memory.WritableMemory;
import org.apache.druid.collections.ResourceHolder;
import org.junit.Assert;
import org.junit.Test;
import java.util.Optional;
/**
* Tests for {@link MemoryAllocator}, subclassed for each concrete implementation.
*/
public abstract class BaseMemoryAllocatorTest
{
private static final int ALLOCATOR_SIZE = 10;
protected abstract MemoryAllocator makeAllocator(int capacity);
@Test
public void testAllocationInSinglePass()
{
MemoryAllocator memoryAllocator = makeAllocator(ALLOCATOR_SIZE);
Optional<ResourceHolder<WritableMemory>> memoryResourceHolderOptional = memoryAllocator.allocate(ALLOCATOR_SIZE);
Assert.assertTrue(memoryResourceHolderOptional.isPresent());
ResourceHolder<WritableMemory> memoryResourceHolder = memoryResourceHolderOptional.get();
WritableMemory memory = memoryResourceHolder.get();
for (int i = 0; i < ALLOCATOR_SIZE; ++i) {
memory.putByte(i, (byte) 0xFF);
}
}
@Test
public void testAllocationInMultiplePasses()
{
MemoryAllocator memoryAllocator = makeAllocator(ALLOCATOR_SIZE);
Optional<ResourceHolder<WritableMemory>> memoryResourceHolderOptional1 = memoryAllocator.allocate(ALLOCATOR_SIZE
- 4);
Assert.assertTrue(memoryResourceHolderOptional1.isPresent());
ResourceHolder<WritableMemory> memoryResourceHolder1 = memoryResourceHolderOptional1.get();
WritableMemory memory1 = memoryResourceHolder1.get();
Optional<ResourceHolder<WritableMemory>> memoryResourceHolderOptional2 = memoryAllocator.allocate(4);
Assert.assertTrue(memoryResourceHolderOptional2.isPresent());
ResourceHolder<WritableMemory> memoryResourceHolder2 = memoryResourceHolderOptional2.get();
WritableMemory memory2 = memoryResourceHolder2.get();
for (int i = 0; i < ALLOCATOR_SIZE - 4; ++i) {
memory1.putByte(i, (byte) 0xFF);
}
for (int i = 0; i < 4; ++i) {
memory2.putByte(i, (byte) 0xFE);
}
// Readback to ensure that value hasn't been overwritten
for (int i = 0; i < ALLOCATOR_SIZE - 4; ++i) {
Assert.assertEquals((byte) 0xFF, memory1.getByte(i));
}
for (int i = 0; i < 4; ++i) {
Assert.assertEquals((byte) 0xFE, memory2.getByte(i));
}
}
@Test
public void testReleaseAllocationTwice()
{
final MemoryAllocator memoryAllocator = makeAllocator(ALLOCATOR_SIZE);
final int allocationSize = 4;
final Optional<ResourceHolder<WritableMemory>> holder1 = memoryAllocator.allocate(allocationSize);
final Optional<ResourceHolder<WritableMemory>> holder2 = memoryAllocator.allocate(allocationSize);
Assert.assertTrue(holder1.isPresent());
Assert.assertTrue(holder2.isPresent());
Assert.assertEquals(ALLOCATOR_SIZE - allocationSize * 2, memoryAllocator.available());
// Release the second allocation.
holder2.get().close();
Assert.assertEquals(ALLOCATOR_SIZE - allocationSize, memoryAllocator.available());
// Release again-- does nothing.
holder2.get().close();
Assert.assertEquals(ALLOCATOR_SIZE - allocationSize, memoryAllocator.available());
// Release the first allocation.
holder1.get().close();
Assert.assertEquals(ALLOCATOR_SIZE, memoryAllocator.available());
}
@Test
public void testReleaseLastAllocationFirst()
{
final MemoryAllocator memoryAllocator = makeAllocator(ALLOCATOR_SIZE);
final int allocationSize = 4;
final Optional<ResourceHolder<WritableMemory>> holder1 = memoryAllocator.allocate(allocationSize);
final Optional<ResourceHolder<WritableMemory>> holder2 = memoryAllocator.allocate(allocationSize);
Assert.assertTrue(holder1.isPresent());
Assert.assertTrue(holder2.isPresent());
Assert.assertEquals(ALLOCATOR_SIZE - allocationSize * 2, memoryAllocator.available());
// Release the second allocation.
holder2.get().close();
Assert.assertEquals(ALLOCATOR_SIZE - allocationSize, memoryAllocator.available());
// Release the first allocation.
holder1.get().close();
Assert.assertEquals(ALLOCATOR_SIZE, memoryAllocator.available());
}
@Test
public void testReleaseLastAllocationLast()
{
final MemoryAllocator memoryAllocator = makeAllocator(ALLOCATOR_SIZE);
final int allocationSize = 4;
final Optional<ResourceHolder<WritableMemory>> holder1 = memoryAllocator.allocate(allocationSize);
final Optional<ResourceHolder<WritableMemory>> holder2 = memoryAllocator.allocate(allocationSize);
Assert.assertTrue(holder1.isPresent());
Assert.assertTrue(holder2.isPresent());
Assert.assertEquals(ALLOCATOR_SIZE - allocationSize * 2, memoryAllocator.available());
// Don't check memoryAllocator.available() after holder1 is closed; behavior is not consistent between arena
// and heap. Arena won't reclaim this allocation because it wasn't the final one; heap will reclaim it.
// They converge to fully-reclaimed once all allocations are closed.
holder1.get().close();
holder2.get().close();
Assert.assertEquals(ALLOCATOR_SIZE, memoryAllocator.available());
}
@Test
public void testOverallocationInSinglePass()
{
MemoryAllocator memoryAllocator = makeAllocator(ALLOCATOR_SIZE);
Optional<ResourceHolder<WritableMemory>> memoryResourceHolderOptional =
memoryAllocator.allocate(ALLOCATOR_SIZE + 1);
Assert.assertFalse(memoryResourceHolderOptional.isPresent());
}
@Test
public void testOverallocationInMultiplePasses()
{
MemoryAllocator memoryAllocator = makeAllocator(ALLOCATOR_SIZE);
Optional<ResourceHolder<WritableMemory>> memoryResourceHolderOptional =
memoryAllocator.allocate(ALLOCATOR_SIZE - 4);
Assert.assertTrue(memoryResourceHolderOptional.isPresent());
Assert.assertFalse(memoryAllocator.allocate(5).isPresent());
}
}

View File

@ -19,78 +19,11 @@
package org.apache.druid.frame.allocation; package org.apache.druid.frame.allocation;
import org.apache.datasketches.memory.WritableMemory; public class HeapMemoryAllocatorTest extends BaseMemoryAllocatorTest
import org.apache.druid.collections.ResourceHolder;
import org.junit.Assert;
import org.junit.Test;
import java.util.Optional;
public class HeapMemoryAllocatorTest
{ {
private static final int ALLOCATOR_SIZE = 10; @Override
protected MemoryAllocator makeAllocator(int capacity)
@Test
public void testAllocationInSinglePass()
{ {
MemoryAllocator heapMemoryAllocator = new HeapMemoryAllocator(ALLOCATOR_SIZE); return new HeapMemoryAllocator(capacity);
Optional<ResourceHolder<WritableMemory>> memoryResourceHolderOptional = heapMemoryAllocator.allocate(ALLOCATOR_SIZE);
Assert.assertTrue(memoryResourceHolderOptional.isPresent());
ResourceHolder<WritableMemory> memoryResourceHolder = memoryResourceHolderOptional.get();
WritableMemory memory = memoryResourceHolder.get();
for (int i = 0; i < ALLOCATOR_SIZE; ++i) {
memory.putByte(i, (byte) 0xFF);
}
} }
@Test
public void testAllocationInMultiplePasses()
{
MemoryAllocator heapMemoryAllocator = new HeapMemoryAllocator(ALLOCATOR_SIZE);
Optional<ResourceHolder<WritableMemory>> memoryResourceHolderOptional1 = heapMemoryAllocator.allocate(ALLOCATOR_SIZE
- 4);
Assert.assertTrue(memoryResourceHolderOptional1.isPresent());
ResourceHolder<WritableMemory> memoryResourceHolder1 = memoryResourceHolderOptional1.get();
WritableMemory memory1 = memoryResourceHolder1.get();
Optional<ResourceHolder<WritableMemory>> memoryResourceHolderOptional2 = heapMemoryAllocator.allocate(4);
Assert.assertTrue(memoryResourceHolderOptional2.isPresent());
ResourceHolder<WritableMemory> memoryResourceHolder2 = memoryResourceHolderOptional2.get();
WritableMemory memory2 = memoryResourceHolder2.get();
for (int i = 0; i < ALLOCATOR_SIZE - 4; ++i) {
memory1.putByte(i, (byte) 0xFF);
}
for (int i = 0; i < 4; ++i) {
memory2.putByte(i, (byte) 0xFE);
}
// Readback to ensure that value hasn't been overwritten
for (int i = 0; i < ALLOCATOR_SIZE - 4; ++i) {
Assert.assertEquals((byte) 0xFF, memory1.getByte(i));
}
for (int i = 0; i < 4; ++i) {
Assert.assertEquals((byte) 0xFE, memory2.getByte(i));
}
}
@Test
public void testOverallocationInSinglePass()
{
MemoryAllocator heapMemoryAllocator = new HeapMemoryAllocator(ALLOCATOR_SIZE);
Optional<ResourceHolder<WritableMemory>> memoryResourceHolderOptional =
heapMemoryAllocator.allocate(ALLOCATOR_SIZE + 1);
Assert.assertFalse(memoryResourceHolderOptional.isPresent());
}
@Test
public void testOverallocationInMultiplePasses()
{
MemoryAllocator heapMemoryAllocator = new HeapMemoryAllocator(ALLOCATOR_SIZE);
Optional<ResourceHolder<WritableMemory>> memoryResourceHolderOptional =
heapMemoryAllocator.allocate(ALLOCATOR_SIZE - 4);
Assert.assertTrue(memoryResourceHolderOptional.isPresent());
Assert.assertFalse(heapMemoryAllocator.allocate(5).isPresent());
}
} }

View File

@ -20,21 +20,76 @@
package org.apache.druid.frame.write; package org.apache.druid.frame.write;
import com.google.common.collect.ImmutableList; import com.google.common.collect.ImmutableList;
import com.google.common.collect.ImmutableMap;
import org.apache.druid.data.input.MapBasedRow;
import org.apache.druid.data.input.Row;
import org.apache.druid.frame.Frame;
import org.apache.druid.frame.allocation.AppendableMemory; import org.apache.druid.frame.allocation.AppendableMemory;
import org.apache.druid.frame.allocation.ArenaMemoryAllocatorFactory;
import org.apache.druid.frame.allocation.HeapMemoryAllocator; import org.apache.druid.frame.allocation.HeapMemoryAllocator;
import org.apache.druid.frame.field.LongFieldWriter; import org.apache.druid.frame.field.LongFieldWriter;
import org.apache.druid.frame.read.FrameReader;
import org.apache.druid.frame.testutil.FrameTestUtil;
import org.apache.druid.java.util.common.StringUtils;
import org.apache.druid.java.util.common.guava.Sequences;
import org.apache.druid.segment.ColumnSelectorFactory;
import org.apache.druid.segment.RowAdapters;
import org.apache.druid.segment.RowBasedColumnSelectorFactory;
import org.apache.druid.segment.column.ColumnType; import org.apache.druid.segment.column.ColumnType;
import org.apache.druid.segment.column.RowSignature; import org.apache.druid.segment.column.RowSignature;
import org.apache.druid.testing.InitializedNullHandlingTest;
import org.easymock.EasyMock; import org.easymock.EasyMock;
import org.junit.Assert; import org.junit.Assert;
import org.junit.Test; import org.junit.Test;
import java.util.Arrays;
import java.util.Collections; import java.util.Collections;
public class RowBasedFrameWriterTest public class RowBasedFrameWriterTest extends InitializedNullHandlingTest
{ {
@Test @Test
public void testAddSelectionWithException() public void test_addSelection_singleLargeRow()
{
final RowSignature signature =
RowSignature.builder()
.add("n", ColumnType.LONG)
.add("s", ColumnType.STRING)
.build();
final byte[] largeUtf8 = new byte[990000];
Arrays.fill(largeUtf8, (byte) 'F');
final String largeString = StringUtils.fromUtf8(largeUtf8);
final Row largeRow = new MapBasedRow(0L, ImmutableMap.of("n", 3L, "s", largeString));
final FrameWriterFactory frameWriterFactory = FrameWriters.makeRowBasedFrameWriterFactory(
new ArenaMemoryAllocatorFactory(1_000_000),
signature,
ImmutableList.of(),
false
);
final ColumnSelectorFactory columnSelectorFactory = RowBasedColumnSelectorFactory.create(
RowAdapters.standardRow(),
() -> largeRow,
signature,
false,
false
);
final Frame frame;
try (final FrameWriter frameWriter = frameWriterFactory.newFrameWriter(columnSelectorFactory)) {
Assert.assertTrue(frameWriter.addSelection());
frame = Frame.wrap(frameWriter.toByteArray());
}
FrameTestUtil.assertRowsEqual(
Sequences.simple(Collections.singletonList(ImmutableList.of(3L, largeString))),
FrameTestUtil.readRowsFromCursorFactory(FrameReader.create(signature).makeCursorFactory(frame))
);
}
@Test
public void test_addSelection_withException()
{ {
String colName = "colName"; String colName = "colName";
String errorMsg = "Frame writer exception"; String errorMsg = "Frame writer exception";