All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.scalar.db.api.DistributedStorageConditionalMutationIntegrationTestBase Maven / Gradle / Ivy

package com.scalar.db.api;

import static org.assertj.core.api.Assertions.assertThat;
import static org.assertj.core.api.Assertions.assertThatThrownBy;
import static org.assertj.core.api.Assertions.catchThrowable;

import com.google.common.collect.ImmutableMap;
import com.google.common.collect.ImmutableSet;
import com.google.common.collect.Ordering;
import com.scalar.db.api.ConditionalExpression.Operator;
import com.scalar.db.exception.storage.ExecutionException;
import com.scalar.db.exception.storage.NoMutationException;
import com.scalar.db.io.BigIntColumn;
import com.scalar.db.io.BlobColumn;
import com.scalar.db.io.BooleanColumn;
import com.scalar.db.io.Column;
import com.scalar.db.io.DataType;
import com.scalar.db.io.DoubleColumn;
import com.scalar.db.io.FloatColumn;
import com.scalar.db.io.IntColumn;
import com.scalar.db.io.Key;
import com.scalar.db.io.TextColumn;
import com.scalar.db.service.StorageFactory;
import com.scalar.db.util.TestUtils;
import edu.umd.cs.findbugs.annotations.Nullable;
import java.nio.ByteBuffer;
import java.util.ArrayList;
import java.util.Collections;
import java.util.List;
import java.util.Map;
import java.util.Optional;
import java.util.Properties;
import java.util.Random;
import java.util.concurrent.Callable;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.Future;
import org.junit.jupiter.api.AfterAll;
import org.junit.jupiter.api.BeforeAll;
import org.junit.jupiter.api.BeforeEach;
import org.junit.jupiter.api.Test;
import org.junit.jupiter.api.TestInstance;

@TestInstance(TestInstance.Lifecycle.PER_CLASS)
public abstract class DistributedStorageConditionalMutationIntegrationTestBase {

  private static final String TEST_NAME = "storage_cond_mutation";
  private static final String NAMESPACE = "int_test_" + TEST_NAME;
  private static final String TABLE = "test_table";
  private static final String PARTITION_KEY = "pkey";
  private static final String COL_NAME1 = "c1";
  private static final String COL_NAME2 = "c2";
  private static final String COL_NAME3 = "c3";
  private static final String COL_NAME4 = "c4";
  private static final String COL_NAME5 = "c5";
  private static final String COL_NAME6 = "c6";
  private static final String COL_NAME7 = "c7";

  private static final TableMetadata TABLE_METADATA =
      TableMetadata.newBuilder()
          .addColumn(PARTITION_KEY, DataType.TEXT)
          .addColumn(COL_NAME1, DataType.BOOLEAN)
          .addColumn(COL_NAME2, DataType.INT)
          .addColumn(COL_NAME3, DataType.BIGINT)
          .addColumn(COL_NAME4, DataType.FLOAT)
          .addColumn(COL_NAME5, DataType.DOUBLE)
          .addColumn(COL_NAME6, DataType.TEXT)
          .addColumn(COL_NAME7, DataType.BLOB)
          .addPartitionKey(PARTITION_KEY)
          .build();

  private static final int ATTEMPT_COUNT = 5;
  private static final int THREAD_NUM = 10;

  private DistributedStorageAdmin admin;
  private DistributedStorage storage;
  private String namespace;

  private long seed;
  private ThreadLocal random;

  private List operatorAndDataTypeList;

  private ExecutorService executorService;

  @BeforeAll
  public void beforeAll() throws Exception {
    initialize(TEST_NAME);
    StorageFactory factory = StorageFactory.create(getProperties(TEST_NAME));
    admin = factory.getAdmin();
    namespace = getNamespace();
    createTable();
    storage = factory.getStorage();
    seed = System.currentTimeMillis();
    System.out.println("The seed used in the conditional mutation integration test is " + seed);
    random = ThreadLocal.withInitial(Random::new);
    operatorAndDataTypeList = getOperatorAndDataTypeListForTest();
    executorService = Executors.newFixedThreadPool(getThreadNum());
  }

  protected void initialize(String testName) throws Exception {}

  protected abstract Properties getProperties(String testName);

  protected String getNamespace() {
    return NAMESPACE;
  }

  protected int getThreadNum() {
    return THREAD_NUM;
  }

  private void createTable() throws ExecutionException {
    Map options = getCreationOptions();
    admin.createNamespace(namespace, true, options);
    admin.createTable(namespace, TABLE, TABLE_METADATA, true, options);
  }

  protected Map getCreationOptions() {
    return Collections.emptyMap();
  }

  @BeforeEach
  public void setUp() throws ExecutionException {
    admin.truncateTable(namespace, TABLE);
  }

  @AfterAll
  public void afterAll() throws ExecutionException {
    dropTable();
    admin.close();
    storage.close();
  }

  private void dropTable() throws ExecutionException {
    admin.dropTable(namespace, TABLE);
    admin.dropNamespace(namespace);
  }

  protected List getOperatorAndDataTypeListForTest() {
    List ret = new ArrayList<>();
    for (Operator operator : Operator.values()) {
      for (DataType dataType : DataType.values()) {
        ret.add(new OperatorAndDataType(operator, dataType));
      }
    }
    return ret;
  }

  @Test
  public void put_withPutIfWithSingleCondition_shouldPutProperly()
      throws java.util.concurrent.ExecutionException, InterruptedException {
    executeInParallel(
        (operator, dataType) -> {
          put_withPutIfWithSingleConditionWithSameValue_shouldPutProperly(operator, dataType);
          put_withPutIfWithSingleConditionWithRandomValue_shouldPutProperly(operator, dataType);
          put_withPutIfWithInitialDataWithNullValuesWithSingleCondition_shouldPutProperly(
              operator, dataType);
          put_withPutIfWithInitialDataWithoutValuesWithSingleCondition_shouldPutProperly(
              operator, dataType);
        });
  }

  private void put_withPutIfWithSingleConditionWithSameValue_shouldPutProperly(
      Operator operator, DataType dataType) throws ExecutionException {
    // Arrange
    Map> initialData = putInitialDataWithRandomValues(operator, dataType);

    String columnName = getColumnName(dataType);
    Column columnToCompare = initialData.get(columnName);

    MutationCondition condition =
        ConditionBuilder.putIf(buildConditionalExpression(columnToCompare, operator)).build();
    Put put = preparePutWithRandomValues(operator, dataType).withCondition(condition);

    boolean shouldMutate = shouldMutate(initialData.get(columnName), columnToCompare, operator);

    // Act Assert
    put_withPutIf_shouldPutProperly(
        initialData,
        put,
        shouldMutate,
        prepareGet(operator, dataType),
        description(initialData, put, columnToCompare, operator));
  }

  private void put_withPutIfWithSingleConditionWithRandomValue_shouldPutProperly(
      Operator operator, DataType dataType) throws ExecutionException {
    for (int i = 0; i < ATTEMPT_COUNT; i++) {
      // Arrange
      Map> initialData = putInitialDataWithRandomValues(operator, dataType);

      String columnName = getColumnName(dataType);
      Column columnToCompare = getColumnWithRandomValue(random.get(), columnName, dataType);

      MutationCondition condition =
          ConditionBuilder.putIf(buildConditionalExpression(columnToCompare, operator)).build();
      Put put = preparePutWithRandomValues(operator, dataType).withCondition(condition);

      boolean shouldMutate = shouldMutate(initialData.get(columnName), columnToCompare, operator);

      // Act Assert
      put_withPutIf_shouldPutProperly(
          initialData,
          put,
          shouldMutate,
          prepareGet(operator, dataType),
          description(initialData, put, columnToCompare, operator));
    }
  }

  private void put_withPutIfWithInitialDataWithNullValuesWithSingleCondition_shouldPutProperly(
      Operator operator, DataType dataType) throws ExecutionException {
    // Arrange
    Map> initialData = putInitialDataWithNullValues(operator, dataType);

    String columnName = getColumnName(dataType);
    Column columnToCompare = getColumnWithRandomValue(random.get(), columnName, dataType);

    MutationCondition condition =
        ConditionBuilder.putIf(buildConditionalExpression(columnToCompare, operator)).build();
    Put put = preparePutWithRandomValues(operator, dataType).withCondition(condition);

    boolean shouldMutate = shouldMutate(initialData.get(columnName), columnToCompare, operator);

    // Act Assert
    put_withPutIf_shouldPutProperly(
        initialData,
        put,
        shouldMutate,
        prepareGet(operator, dataType),
        description(initialData, put, columnToCompare, operator));
  }

  private void put_withPutIfWithInitialDataWithoutValuesWithSingleCondition_shouldPutProperly(
      Operator operator, DataType dataType) throws ExecutionException {
    // Arrange
    Map> initialData = putInitialDataWithoutValues(operator, dataType);

    String columnName = getColumnName(dataType);
    Column columnToCompare = getColumnWithRandomValue(random.get(), columnName, dataType);

    MutationCondition condition =
        ConditionBuilder.putIf(buildConditionalExpression(columnToCompare, operator)).build();
    Put put = preparePutWithRandomValues(operator, dataType).withCondition(condition);

    boolean shouldMutate = shouldMutate(initialData.get(columnName), columnToCompare, operator);

    // Act Assert
    put_withPutIf_shouldPutProperly(
        initialData,
        put,
        shouldMutate,
        prepareGet(operator, dataType),
        description(initialData, put, columnToCompare, operator));
  }

  @Test
  public void put_withPutIfWithMultipleConditions_shouldPutProperly()
      throws java.util.concurrent.ExecutionException, InterruptedException {
    executeInParallel(
        (firstOperator, firstDataType, secondOperator, secondDataType) -> {
          put_withPutIfWithMultipleConditionsWithSameValue_shouldPutProperly(
              firstOperator, firstDataType, secondOperator, secondDataType);
          put_withPutIfWithMultipleConditionsWithRandomValue_shouldPutProperly(
              firstOperator, firstDataType, secondOperator, secondDataType);
          put_withPutIfWithInitialDataWithNullValuesWithMultipleConditions_shouldPutProperly(
              firstOperator, firstDataType, secondOperator, secondDataType);
          put_withPutIfWithInitialDataWithoutValuesWithMultipleConditions_shouldPutProperly(
              firstOperator, firstDataType, secondOperator, secondDataType);
        });
  }

  private void put_withPutIfWithMultipleConditionsWithSameValue_shouldPutProperly(
      Operator firstOperator,
      DataType firstDataType,
      Operator secondOperator,
      DataType secondDataType)
      throws ExecutionException {
    // Arrange
    Map> initialData =
        putInitialDataWithRandomValues(
            firstOperator, firstDataType, secondOperator, secondDataType);

    String firstColumnName = getColumnName(firstDataType);
    Column firstColumnToCompare = initialData.get(firstColumnName);

    String secondColumnName = getColumnName(secondDataType);
    Column secondColumnToCompare = initialData.get(secondColumnName);

    MutationCondition condition =
        ConditionBuilder.putIf(buildConditionalExpression(firstColumnToCompare, firstOperator))
            .and(buildConditionalExpression(secondColumnToCompare, secondOperator))
            .build();
    Put put =
        preparePutWithRandomValues(firstOperator, firstDataType, secondOperator, secondDataType)
            .withCondition(condition);

    boolean shouldMutate =
        shouldMutate(initialData.get(firstColumnName), firstColumnToCompare, firstOperator)
            && shouldMutate(
                initialData.get(secondColumnName), secondColumnToCompare, secondOperator);

    // Act Assert
    put_withPutIf_shouldPutProperly(
        initialData,
        put,
        shouldMutate,
        prepareGet(firstOperator, firstDataType, secondOperator, secondDataType),
        description(
            initialData,
            put,
            firstColumnToCompare,
            firstOperator,
            secondColumnToCompare,
            secondOperator));
  }

  private void put_withPutIfWithMultipleConditionsWithRandomValue_shouldPutProperly(
      Operator firstOperator,
      DataType firstDataType,
      Operator secondOperator,
      DataType secondDataType)
      throws ExecutionException {
    for (int i = 0; i < ATTEMPT_COUNT; i++) {
      // Arrange
      Map> initialData =
          putInitialDataWithRandomValues(
              firstOperator, firstDataType, secondOperator, secondDataType);

      String firstColumnName = getColumnName(firstDataType);
      Column firstColumnToCompare =
          getColumnWithRandomValue(random.get(), firstColumnName, firstDataType);

      String secondColumnName = getColumnName(secondDataType);
      Column secondColumnToCompare =
          getColumnWithRandomValue(random.get(), secondColumnName, secondDataType);

      MutationCondition condition =
          ConditionBuilder.putIf(buildConditionalExpression(firstColumnToCompare, firstOperator))
              .and(buildConditionalExpression(secondColumnToCompare, secondOperator))
              .build();
      Put put =
          preparePutWithRandomValues(firstOperator, firstDataType, secondOperator, secondDataType)
              .withCondition(condition);

      boolean shouldMutate =
          shouldMutate(initialData.get(firstColumnName), firstColumnToCompare, firstOperator)
              && shouldMutate(
                  initialData.get(secondColumnName), secondColumnToCompare, secondOperator);

      // Act Assert
      put_withPutIf_shouldPutProperly(
          initialData,
          put,
          shouldMutate,
          prepareGet(firstOperator, firstDataType, secondOperator, secondDataType),
          description(
              initialData,
              put,
              firstColumnToCompare,
              firstOperator,
              secondColumnToCompare,
              secondOperator));
    }
  }

  private void put_withPutIfWithInitialDataWithNullValuesWithMultipleConditions_shouldPutProperly(
      Operator firstOperator,
      DataType firstDataType,
      Operator secondOperator,
      DataType secondDataType)
      throws ExecutionException {
    // Arrange
    Map> initialData =
        putInitialDataWithNullValues(firstOperator, firstDataType, secondOperator, secondDataType);

    String firstColumnName = getColumnName(firstDataType);
    Column firstColumnToCompare =
        getColumnWithRandomValue(random.get(), firstColumnName, firstDataType);

    String secondColumnName = getColumnName(secondDataType);
    Column secondColumnToCompare =
        getColumnWithRandomValue(random.get(), secondColumnName, secondDataType);

    MutationCondition condition =
        ConditionBuilder.putIf(buildConditionalExpression(firstColumnToCompare, firstOperator))
            .and(buildConditionalExpression(secondColumnToCompare, secondOperator))
            .build();
    Put put =
        preparePutWithRandomValues(firstOperator, firstDataType, secondOperator, secondDataType)
            .withCondition(condition);

    boolean shouldMutate =
        shouldMutate(initialData.get(firstColumnName), firstColumnToCompare, firstOperator)
            && shouldMutate(
                initialData.get(secondColumnName), secondColumnToCompare, secondOperator);

    // Act Assert
    put_withPutIf_shouldPutProperly(
        initialData,
        put,
        shouldMutate,
        prepareGet(firstOperator, firstDataType, secondOperator, secondDataType),
        description(
            initialData,
            put,
            firstColumnToCompare,
            firstOperator,
            secondColumnToCompare,
            secondOperator));
  }

  private void put_withPutIfWithInitialDataWithoutValuesWithMultipleConditions_shouldPutProperly(
      Operator firstOperator,
      DataType firstDataType,
      Operator secondOperator,
      DataType secondDataType)
      throws ExecutionException {
    // Arrange
    Map> initialData =
        putInitialDataWithoutValues(firstOperator, firstDataType, secondOperator, secondDataType);

    String firstColumnName = getColumnName(firstDataType);
    Column firstColumnToCompare =
        getColumnWithRandomValue(random.get(), firstColumnName, firstDataType);

    String secondColumnName = getColumnName(secondDataType);
    Column secondColumnToCompare =
        getColumnWithRandomValue(random.get(), secondColumnName, secondDataType);

    MutationCondition condition =
        ConditionBuilder.putIf(buildConditionalExpression(firstColumnToCompare, firstOperator))
            .and(buildConditionalExpression(secondColumnToCompare, secondOperator))
            .build();
    Put put =
        preparePutWithRandomValues(firstOperator, firstDataType, secondOperator, secondDataType)
            .withCondition(condition);

    boolean shouldMutate =
        shouldMutate(initialData.get(firstColumnName), firstColumnToCompare, firstOperator)
            && shouldMutate(
                initialData.get(secondColumnName), secondColumnToCompare, secondOperator);

    // Act Assert
    put_withPutIf_shouldPutProperly(
        initialData,
        put,
        shouldMutate,
        prepareGet(firstOperator, firstDataType, secondOperator, secondDataType),
        description(
            initialData,
            put,
            firstColumnToCompare,
            firstOperator,
            secondColumnToCompare,
            secondOperator));
  }

  private void put_withPutIf_shouldPutProperly(
      Map> initialData,
      Put put,
      boolean shouldMutate,
      Get get,
      String description)
      throws ExecutionException {
    Throwable thrown = catchThrowable(() -> storage.put(put));
    if (shouldMutate) {
      assertThat(thrown).describedAs(description).isNull();
    } else {
      assertThat(thrown).describedAs(description).isInstanceOf(NoMutationException.class);
    }

    Optional result = storage.get(get);
    assertThat(result).describedAs(description).isPresent();
    assertThat(result.get().getContainedColumnNames())
        .describedAs(description)
        .isEqualTo(
            ImmutableSet.of(
                PARTITION_KEY,
                COL_NAME1,
                COL_NAME2,
                COL_NAME3,
                COL_NAME4,
                COL_NAME5,
                COL_NAME6,
                COL_NAME7));

    Map> expected = shouldMutate ? put.getColumns() : initialData;
    assertThat(result.get().isNull(COL_NAME1))
        .describedAs(description)
        .isEqualTo(expected.get(COL_NAME1).hasNullValue());
    assertThat(result.get().getBoolean(COL_NAME1))
        .describedAs(description)
        .isEqualTo(expected.get(COL_NAME1).getBooleanValue());
    assertThat(result.get().isNull(COL_NAME2))
        .describedAs(description)
        .isEqualTo(expected.get(COL_NAME2).hasNullValue());
    assertThat(result.get().getInt(COL_NAME2))
        .describedAs(description)
        .isEqualTo(expected.get(COL_NAME2).getIntValue());
    assertThat(result.get().isNull(COL_NAME3))
        .describedAs(description)
        .isEqualTo(expected.get(COL_NAME3).hasNullValue());
    assertThat(result.get().getBigInt(COL_NAME3))
        .describedAs(description)
        .isEqualTo(expected.get(COL_NAME3).getBigIntValue());
    assertThat(result.get().isNull(COL_NAME4))
        .describedAs(description)
        .isEqualTo(expected.get(COL_NAME4).hasNullValue());
    assertThat(result.get().getFloat(COL_NAME4))
        .describedAs(description)
        .isEqualTo(expected.get(COL_NAME4).getFloatValue());
    assertThat(result.get().isNull(COL_NAME5))
        .describedAs(description)
        .isEqualTo(expected.get(COL_NAME5).hasNullValue());
    assertThat(result.get().getDouble(COL_NAME5))
        .describedAs(description)
        .isEqualTo(expected.get(COL_NAME5).getDoubleValue());
    assertThat(result.get().getText(COL_NAME6))
        .describedAs(description)
        .isEqualTo(expected.get(COL_NAME6).getTextValue());
    assertThat(result.get().getBlob(COL_NAME7))
        .describedAs(description)
        .isEqualTo(expected.get(COL_NAME7).getBlobValue());
  }

  @Test
  public void put_withPutIfExistsWhenRecordExists_shouldPutProperly() throws ExecutionException {
    random.get().setSeed(seed);

    // Arrange
    putInitialDataWithRandomValues();

    Put put = preparePutWithRandomValues().withCondition(ConditionBuilder.putIfExists());

    // Act
    storage.put(put);

    // Assert
    Optional result = storage.get(prepareGet());
    assertThat(result).isPresent();
    assertThat(result.get().getContainedColumnNames())
        .isEqualTo(
            ImmutableSet.of(
                PARTITION_KEY,
                COL_NAME1,
                COL_NAME2,
                COL_NAME3,
                COL_NAME4,
                COL_NAME5,
                COL_NAME6,
                COL_NAME7));
    assertThat(result.get().getBoolean(COL_NAME1)).isEqualTo(put.getBooleanValue(COL_NAME1));
    assertThat(result.get().getInt(COL_NAME2)).isEqualTo(put.getIntValue(COL_NAME2));
    assertThat(result.get().getBigInt(COL_NAME3)).isEqualTo(put.getBigIntValue(COL_NAME3));
    assertThat(result.get().getFloat(COL_NAME4)).isEqualTo(put.getFloatValue(COL_NAME4));
    assertThat(result.get().getDouble(COL_NAME5)).isEqualTo(put.getDoubleValue(COL_NAME5));
    assertThat(result.get().getText(COL_NAME6)).isEqualTo(put.getTextValue(COL_NAME6));
    assertThat(result.get().getBlob(COL_NAME7)).isEqualTo(put.getBlobValue(COL_NAME7));
  }

  @Test
  public void put_withPutIfExistsWhenRecordDoesNotExist_shouldThrowNoMutationException()
      throws ExecutionException {
    random.get().setSeed(seed);

    // Arrange
    Put put = preparePutWithRandomValues().withCondition(ConditionBuilder.putIfExists());

    // Act Assert
    assertThatThrownBy(() -> storage.put(put)).isInstanceOf(NoMutationException.class);

    Optional result = storage.get(prepareGet());
    assertThat(result).isNotPresent();
  }

  @Test
  public void put_withPutIfNotExistsWhenRecordDoesNotExist_shouldPutProperly()
      throws ExecutionException {
    random.get().setSeed(seed);

    // Arrange
    Put put = preparePutWithRandomValues().withCondition(ConditionBuilder.putIfNotExists());

    // Act
    storage.put(put);

    // Assert
    Optional result = storage.get(prepareGet());
    assertThat(result).isPresent();
    assertThat(result.get().getContainedColumnNames())
        .isEqualTo(
            ImmutableSet.of(
                PARTITION_KEY,
                COL_NAME1,
                COL_NAME2,
                COL_NAME3,
                COL_NAME4,
                COL_NAME5,
                COL_NAME6,
                COL_NAME7));
    assertThat(result.get().getBoolean(COL_NAME1)).isEqualTo(put.getBooleanValue(COL_NAME1));
    assertThat(result.get().getInt(COL_NAME2)).isEqualTo(put.getIntValue(COL_NAME2));
    assertThat(result.get().getBigInt(COL_NAME3)).isEqualTo(put.getBigIntValue(COL_NAME3));
    assertThat(result.get().getFloat(COL_NAME4)).isEqualTo(put.getFloatValue(COL_NAME4));
    assertThat(result.get().getDouble(COL_NAME5)).isEqualTo(put.getDoubleValue(COL_NAME5));
    assertThat(result.get().getText(COL_NAME6)).isEqualTo(put.getTextValue(COL_NAME6));
    assertThat(result.get().getBlob(COL_NAME7)).isEqualTo(put.getBlobValue(COL_NAME7));
  }

  @Test
  public void put_withPutIfNotExistsWhenRecordExists_shouldThrowNoMutationException()
      throws ExecutionException {
    random.get().setSeed(seed);

    // Arrange
    Map> initialData = putInitialDataWithRandomValues();

    Put put = preparePutWithRandomValues().withCondition(ConditionBuilder.putIfNotExists());

    // Act Assert
    assertThatThrownBy(() -> storage.put(put)).isInstanceOf(NoMutationException.class);

    Optional result = storage.get(prepareGet());
    assertThat(result).isPresent();
    assertThat(result.get().getContainedColumnNames())
        .isEqualTo(
            ImmutableSet.of(
                PARTITION_KEY,
                COL_NAME1,
                COL_NAME2,
                COL_NAME3,
                COL_NAME4,
                COL_NAME5,
                COL_NAME6,
                COL_NAME7));
    assertThat(result.get().getBoolean(COL_NAME1))
        .isEqualTo(initialData.get(COL_NAME1).getBooleanValue());
    assertThat(result.get().getInt(COL_NAME2)).isEqualTo(initialData.get(COL_NAME2).getIntValue());
    assertThat(result.get().getBigInt(COL_NAME3))
        .isEqualTo(initialData.get(COL_NAME3).getBigIntValue());
    assertThat(result.get().getFloat(COL_NAME4))
        .isEqualTo(initialData.get(COL_NAME4).getFloatValue());
    assertThat(result.get().getDouble(COL_NAME5))
        .isEqualTo(initialData.get(COL_NAME5).getDoubleValue());
    assertThat(result.get().getText(COL_NAME6))
        .isEqualTo(initialData.get(COL_NAME6).getTextValue());
    assertThat(result.get().getBlob(COL_NAME7))
        .isEqualTo(initialData.get(COL_NAME7).getBlobValue());
  }

  @Test
  public void delete_withDeleteIfWithSingleCondition_shouldPutProperly()
      throws java.util.concurrent.ExecutionException, InterruptedException {
    executeInParallel(
        (operator, dataType) -> {
          delete_withDeleteIfWithSingleConditionWithSameValue_shouldPutProperly(operator, dataType);
          delete_withDeleteWithSingleConditionWithRandomValue_shouldPutProperly(operator, dataType);
          delete_withDeleteIfWithInitialDataWithNullValuesWithSingleCondition_shouldPutProperly(
              operator, dataType);
          delete_withDeleteIfWithInitialDataWithoutValuesWithSingleCondition_shouldPutProperly(
              operator, dataType);
        });
  }

  private void delete_withDeleteIfWithSingleConditionWithSameValue_shouldPutProperly(
      Operator operator, DataType dataType) throws ExecutionException {
    // Arrange
    Map> initialData = putInitialDataWithRandomValues(operator, dataType);

    String columnName = getColumnName(dataType);
    Column columnToCompare = initialData.get(columnName);

    MutationCondition condition =
        ConditionBuilder.deleteIf(buildConditionalExpression(columnToCompare, operator)).build();
    Delete delete = prepareDelete(operator, dataType).withCondition(condition);

    boolean shouldMutate = shouldMutate(initialData.get(columnName), columnToCompare, operator);

    // Act Assert
    delete_withDeleteIf_shouldPutProperly(
        initialData,
        delete,
        shouldMutate,
        prepareGet(operator, dataType),
        description(initialData, delete, columnToCompare, operator));
  }

  private void delete_withDeleteWithSingleConditionWithRandomValue_shouldPutProperly(
      Operator operator, DataType dataType) throws ExecutionException {
    for (int i = 0; i < ATTEMPT_COUNT; i++) {
      // Arrange
      Map> initialData = putInitialDataWithRandomValues(operator, dataType);

      String columnName = getColumnName(dataType);
      Column columnToCompare = getColumnWithRandomValue(random.get(), columnName, dataType);

      MutationCondition condition =
          ConditionBuilder.deleteIf(buildConditionalExpression(columnToCompare, operator)).build();
      Delete delete = prepareDelete(operator, dataType).withCondition(condition);

      boolean shouldMutate = shouldMutate(initialData.get(columnName), columnToCompare, operator);

      // Act Assert
      delete_withDeleteIf_shouldPutProperly(
          initialData,
          delete,
          shouldMutate,
          prepareGet(operator, dataType),
          description(initialData, delete, columnToCompare, operator));
    }
  }

  private void
      delete_withDeleteIfWithInitialDataWithNullValuesWithSingleCondition_shouldPutProperly(
          Operator operator, DataType dataType) throws ExecutionException {
    // Arrange
    Map> initialData = putInitialDataWithNullValues(operator, dataType);

    String columnName = getColumnName(dataType);
    Column columnToCompare = getColumnWithRandomValue(random.get(), columnName, dataType);

    MutationCondition condition =
        ConditionBuilder.deleteIf(buildConditionalExpression(columnToCompare, operator)).build();
    Delete delete = prepareDelete(operator, dataType).withCondition(condition);

    boolean shouldMutate = shouldMutate(initialData.get(columnName), columnToCompare, operator);

    // Act Assert
    delete_withDeleteIf_shouldPutProperly(
        initialData,
        delete,
        shouldMutate,
        prepareGet(operator, dataType),
        description(initialData, delete, columnToCompare, operator));
  }

  private void delete_withDeleteIfWithInitialDataWithoutValuesWithSingleCondition_shouldPutProperly(
      Operator operator, DataType dataType) throws ExecutionException {
    // Arrange
    Map> initialData = putInitialDataWithoutValues(operator, dataType);

    String columnName = getColumnName(dataType);
    Column columnToCompare = getColumnWithRandomValue(random.get(), columnName, dataType);

    MutationCondition condition =
        ConditionBuilder.deleteIf(buildConditionalExpression(columnToCompare, operator)).build();
    Delete delete = prepareDelete(operator, dataType).withCondition(condition);

    boolean shouldMutate = shouldMutate(initialData.get(columnName), columnToCompare, operator);

    // Act Assert
    delete_withDeleteIf_shouldPutProperly(
        initialData,
        delete,
        shouldMutate,
        prepareGet(operator, dataType),
        description(initialData, delete, columnToCompare, operator));
  }

  @Test
  public void delete_withDeleteIfWithMultipleConditions_shouldPutProperly()
      throws java.util.concurrent.ExecutionException, InterruptedException {
    executeInParallel(
        (firstOperator, firstDataType, secondOperator, secondDataType) -> {
          delete_withDeleteIfWithMultipleConditionsWithSameValue_shouldPutProperly(
              firstOperator, firstDataType, secondOperator, secondDataType);
          delete_withDeleteIfWithMultipleConditionsWithRandomValue_shouldPutProperly(
              firstOperator, firstDataType, secondOperator, secondDataType);
          delete_withDeleteIfWithInitialDataWithNullValuesWithMultipleConditions_shouldPutProperly(
              firstOperator, firstDataType, secondOperator, secondDataType);
          delete_withDeleteIfWithInitialDataWithoutValuesWithMultipleConditions_shouldPutProperly(
              firstOperator, firstDataType, secondOperator, secondDataType);
        });
  }

  private void delete_withDeleteIfWithMultipleConditionsWithSameValue_shouldPutProperly(
      Operator firstOperator,
      DataType firstDataType,
      Operator secondOperator,
      DataType secondDataType)
      throws ExecutionException {
    // Arrange
    Map> initialData =
        putInitialDataWithRandomValues(
            firstOperator, firstDataType, secondOperator, secondDataType);

    String firstColumnName = getColumnName(firstDataType);
    Column firstColumnToCompare = initialData.get(firstColumnName);

    String secondColumnName = getColumnName(secondDataType);
    Column secondColumnToCompare = initialData.get(secondColumnName);

    MutationCondition condition =
        ConditionBuilder.deleteIf(buildConditionalExpression(firstColumnToCompare, firstOperator))
            .and(buildConditionalExpression(secondColumnToCompare, secondOperator))
            .build();

    boolean shouldMutate =
        shouldMutate(initialData.get(firstColumnName), firstColumnToCompare, firstOperator)
            && shouldMutate(
                initialData.get(secondColumnName), secondColumnToCompare, secondOperator);

    Delete delete =
        prepareDelete(firstOperator, firstDataType, secondOperator, secondDataType)
            .withCondition(condition);

    // Act Assert
    delete_withDeleteIf_shouldPutProperly(
        initialData,
        delete,
        shouldMutate,
        prepareGet(firstOperator, firstDataType, secondOperator, secondDataType),
        description(
            initialData,
            delete,
            firstColumnToCompare,
            firstOperator,
            secondColumnToCompare,
            secondOperator));
  }

  private void delete_withDeleteIfWithMultipleConditionsWithRandomValue_shouldPutProperly(
      Operator firstOperator,
      DataType firstDataType,
      Operator secondOperator,
      DataType secondDataType)
      throws ExecutionException {
    for (int i = 0; i < ATTEMPT_COUNT; i++) {
      // Arrange
      Map> initialData =
          putInitialDataWithRandomValues(
              firstOperator, firstDataType, secondOperator, secondDataType);

      String firstColumnName = getColumnName(firstDataType);
      Column firstColumnToCompare =
          getColumnWithRandomValue(random.get(), firstColumnName, firstDataType);

      String secondColumnName = getColumnName(secondDataType);
      Column secondColumnToCompare =
          getColumnWithRandomValue(random.get(), secondColumnName, secondDataType);

      MutationCondition condition =
          ConditionBuilder.deleteIf(buildConditionalExpression(firstColumnToCompare, firstOperator))
              .and(buildConditionalExpression(secondColumnToCompare, secondOperator))
              .build();

      boolean shouldMutate =
          shouldMutate(initialData.get(firstColumnName), firstColumnToCompare, firstOperator)
              && shouldMutate(
                  initialData.get(secondColumnName), secondColumnToCompare, secondOperator);

      Delete delete =
          prepareDelete(firstOperator, firstDataType, secondOperator, secondDataType)
              .withCondition(condition);

      // Act Assert
      delete_withDeleteIf_shouldPutProperly(
          initialData,
          delete,
          shouldMutate,
          prepareGet(firstOperator, firstDataType, secondOperator, secondDataType),
          description(
              initialData,
              delete,
              firstColumnToCompare,
              firstOperator,
              secondColumnToCompare,
              secondOperator));
    }
  }

  private void
      delete_withDeleteIfWithInitialDataWithNullValuesWithMultipleConditions_shouldPutProperly(
          Operator firstOperator,
          DataType firstDataType,
          Operator secondOperator,
          DataType secondDataType)
          throws ExecutionException {
    // Arrange
    Map> initialData =
        putInitialDataWithNullValues(firstOperator, firstDataType, secondOperator, secondDataType);

    String firstColumnName = getColumnName(firstDataType);
    Column firstColumnToCompare =
        getColumnWithRandomValue(random.get(), firstColumnName, firstDataType);

    String secondColumnName = getColumnName(secondDataType);
    Column secondColumnToCompare =
        getColumnWithRandomValue(random.get(), secondColumnName, secondDataType);

    MutationCondition condition =
        ConditionBuilder.deleteIf(buildConditionalExpression(firstColumnToCompare, firstOperator))
            .and(buildConditionalExpression(secondColumnToCompare, secondOperator))
            .build();

    boolean shouldMutate =
        shouldMutate(initialData.get(firstColumnName), firstColumnToCompare, firstOperator)
            && shouldMutate(
                initialData.get(secondColumnName), secondColumnToCompare, secondOperator);

    Delete delete =
        prepareDelete(firstOperator, firstDataType, secondOperator, secondDataType)
            .withCondition(condition);

    // Act Assert
    delete_withDeleteIf_shouldPutProperly(
        initialData,
        delete,
        shouldMutate,
        prepareGet(firstOperator, firstDataType, secondOperator, secondDataType),
        description(
            initialData,
            delete,
            firstColumnToCompare,
            firstOperator,
            secondColumnToCompare,
            secondOperator));
  }

  private void
      delete_withDeleteIfWithInitialDataWithoutValuesWithMultipleConditions_shouldPutProperly(
          Operator firstOperator,
          DataType firstDataType,
          Operator secondOperator,
          DataType secondDataType)
          throws ExecutionException {
    // Arrange
    Map> initialData =
        putInitialDataWithoutValues(firstOperator, firstDataType, secondOperator, secondDataType);

    String firstColumnName = getColumnName(firstDataType);
    Column firstColumnToCompare =
        getColumnWithRandomValue(random.get(), firstColumnName, firstDataType);

    String secondColumnName = getColumnName(secondDataType);
    Column secondColumnToCompare =
        getColumnWithRandomValue(random.get(), secondColumnName, secondDataType);

    MutationCondition condition =
        ConditionBuilder.deleteIf(buildConditionalExpression(firstColumnToCompare, firstOperator))
            .and(buildConditionalExpression(secondColumnToCompare, secondOperator))
            .build();

    boolean shouldMutate =
        shouldMutate(initialData.get(firstColumnName), firstColumnToCompare, firstOperator)
            && shouldMutate(
                initialData.get(secondColumnName), secondColumnToCompare, secondOperator);

    Delete delete =
        prepareDelete(firstOperator, firstDataType, secondOperator, secondDataType)
            .withCondition(condition);

    // Act Assert
    delete_withDeleteIf_shouldPutProperly(
        initialData,
        delete,
        shouldMutate,
        prepareGet(firstOperator, firstDataType, secondOperator, secondDataType),
        description(
            initialData,
            delete,
            firstColumnToCompare,
            firstOperator,
            secondColumnToCompare,
            secondOperator));
  }

  private void delete_withDeleteIf_shouldPutProperly(
      Map> initialData,
      Delete delete,
      boolean shouldMutate,
      Get get,
      String description)
      throws ExecutionException {
    Throwable thrown = catchThrowable(() -> storage.delete(delete));
    if (shouldMutate) {
      assertThat(thrown).describedAs(description).isNull();

      Optional result = storage.get(get);
      assertThat(result).describedAs(description).isNotPresent();
    } else {
      assertThat(thrown).describedAs(description).isInstanceOf(NoMutationException.class);

      Optional result = storage.get(get);
      assertThat(result).describedAs(description).isPresent();
      assertThat(result.get().getContainedColumnNames())
          .describedAs(description)
          .isEqualTo(
              ImmutableSet.of(
                  PARTITION_KEY,
                  COL_NAME1,
                  COL_NAME2,
                  COL_NAME3,
                  COL_NAME4,
                  COL_NAME5,
                  COL_NAME6,
                  COL_NAME7));

      assertThat(result.get().isNull(COL_NAME1))
          .describedAs(description)
          .isEqualTo(initialData.get(COL_NAME1).hasNullValue());
      assertThat(result.get().getBoolean(COL_NAME1))
          .describedAs(description)
          .isEqualTo(initialData.get(COL_NAME1).getBooleanValue());
      assertThat(result.get().isNull(COL_NAME2))
          .describedAs(description)
          .isEqualTo(initialData.get(COL_NAME2).hasNullValue());
      assertThat(result.get().getInt(COL_NAME2))
          .describedAs(description)
          .isEqualTo(initialData.get(COL_NAME2).getIntValue());
      assertThat(result.get().isNull(COL_NAME3))
          .describedAs(description)
          .isEqualTo(initialData.get(COL_NAME3).hasNullValue());
      assertThat(result.get().getBigInt(COL_NAME3))
          .describedAs(description)
          .isEqualTo(initialData.get(COL_NAME3).getBigIntValue());
      assertThat(result.get().isNull(COL_NAME4))
          .describedAs(description)
          .isEqualTo(initialData.get(COL_NAME4).hasNullValue());
      assertThat(result.get().getFloat(COL_NAME4))
          .describedAs(description)
          .isEqualTo(initialData.get(COL_NAME4).getFloatValue());
      assertThat(result.get().isNull(COL_NAME5))
          .describedAs(description)
          .isEqualTo(initialData.get(COL_NAME5).hasNullValue());
      assertThat(result.get().getDouble(COL_NAME5))
          .describedAs(description)
          .isEqualTo(initialData.get(COL_NAME5).getDoubleValue());
      assertThat(result.get().getText(COL_NAME6))
          .describedAs(description)
          .isEqualTo(initialData.get(COL_NAME6).getTextValue());
      assertThat(result.get().getBlob(COL_NAME7))
          .describedAs(description)
          .isEqualTo(initialData.get(COL_NAME7).getBlobValue());
    }
  }

  @Test
  public void delete_withDeleteIfExistsWhenRecordExists_shouldPutProperly()
      throws ExecutionException {
    random.get().setSeed(seed);

    // Arrange
    putInitialDataWithRandomValues();

    Delete delete = prepareDelete().withCondition(ConditionBuilder.deleteIfExists());

    // Act
    storage.delete(delete);

    // Assert
    Optional result = storage.get(prepareGet());
    assertThat(result).isNotPresent();
  }

  @Test
  public void delete_withDeleteIfExistsWhenRecordDoesNotExist_shouldThrowNoMutationException()
      throws ExecutionException {
    random.get().setSeed(seed);

    // Arrange
    Delete delete = prepareDelete().withCondition(ConditionBuilder.deleteIfExists());

    // Act Assert
    assertThatThrownBy(() -> storage.delete(delete)).isInstanceOf(NoMutationException.class);

    Optional result = storage.get(prepareGet());
    assertThat(result).isNotPresent();
  }

  private Get prepareGet() {
    return prepareGet(Operator.EQ, DataType.INT);
  }

  private Get prepareGet(Operator operator, DataType dataType) {
    return prepareGet(operator, dataType, null, null);
  }

  private Get prepareGet(
      Operator firstOperator,
      DataType firstDataType,
      @Nullable Operator secondOperator,
      @Nullable DataType secondDataType) {
    return new Get(
            Key.ofText(
                PARTITION_KEY,
                getPartitionKeyValue(firstOperator, firstDataType, secondOperator, secondDataType)))
        .withConsistency(Consistency.LINEARIZABLE)
        .forNamespace(namespace)
        .forTable(TABLE);
  }

  private Put preparePutWithRandomValues() {
    return preparePutWithRandomValues(Operator.EQ, DataType.INT);
  }

  private Put preparePutWithRandomValues(Operator operator, DataType dataType) {
    return preparePutWithRandomValues(operator, dataType, null, null);
  }

  private Put preparePutWithRandomValues(
      Operator firstOperator,
      DataType firstDataType,
      @Nullable Operator secondOperator,
      @Nullable DataType secondDataType) {
    return new Put(
            Key.ofText(
                PARTITION_KEY,
                getPartitionKeyValue(firstOperator, firstDataType, secondOperator, secondDataType)))
        .withValue(getColumnWithRandomValue(random.get(), COL_NAME1, DataType.BOOLEAN))
        .withValue(getColumnWithRandomValue(random.get(), COL_NAME2, DataType.INT))
        .withValue(getColumnWithRandomValue(random.get(), COL_NAME3, DataType.BIGINT))
        .withValue(getColumnWithRandomValue(random.get(), COL_NAME4, DataType.FLOAT))
        .withValue(getColumnWithRandomValue(random.get(), COL_NAME5, DataType.DOUBLE))
        .withValue(getColumnWithRandomValue(random.get(), COL_NAME6, DataType.TEXT))
        .withValue(getColumnWithRandomValue(random.get(), COL_NAME7, DataType.BLOB))
        .withConsistency(Consistency.LINEARIZABLE)
        .forNamespace(namespace)
        .forTable(TABLE);
  }

  private Delete prepareDelete() {
    return prepareDelete(Operator.EQ, DataType.INT);
  }

  private Delete prepareDelete(Operator operator, DataType dataType) {
    return prepareDelete(operator, dataType, null, null);
  }

  private Delete prepareDelete(
      Operator firstOperator,
      DataType firstDataType,
      @Nullable Operator secondOperator,
      @Nullable DataType secondDataType) {
    return new Delete(
            Key.ofText(
                PARTITION_KEY,
                getPartitionKeyValue(firstOperator, firstDataType, secondOperator, secondDataType)))
        .withConsistency(Consistency.LINEARIZABLE)
        .forNamespace(namespace)
        .forTable(TABLE);
  }

  private Map> putInitialDataWithRandomValues() throws ExecutionException {
    return putInitialDataWithRandomValues(Operator.EQ, DataType.INT);
  }

  private Map> putInitialDataWithRandomValues(
      Operator operator, DataType dataType) throws ExecutionException {
    return putInitialDataWithRandomValues(operator, dataType, null, null);
  }

  private Map> putInitialDataWithRandomValues(
      Operator firstOperator,
      DataType firstDataType,
      @Nullable Operator secondOperator,
      @Nullable DataType secondDataType)
      throws ExecutionException {
    try {
      storage.delete(
          prepareDelete(firstOperator, firstDataType, secondOperator, secondDataType)
              .withCondition(ConditionBuilder.deleteIfExists()));
    } catch (NoMutationException ignored) {
      // ignored
    }

    Put initialPut =
        preparePutWithRandomValues(firstOperator, firstDataType, secondOperator, secondDataType)
            .withCondition(ConditionBuilder.putIfNotExists());
    storage.put(initialPut);
    return initialPut.getColumns();
  }

  private Map> putInitialDataWithNullValues(Operator operator, DataType dataType)
      throws ExecutionException {
    return putInitialDataWithNullValues(operator, dataType, null, null);
  }

  private Map> putInitialDataWithNullValues(
      Operator firstOperator,
      DataType firstDataType,
      @Nullable Operator secondOperator,
      @Nullable DataType secondDataType)
      throws ExecutionException {
    try {
      storage.delete(
          prepareDelete(firstOperator, firstDataType, secondOperator, secondDataType)
              .withCondition(ConditionBuilder.deleteIfExists()));
    } catch (NoMutationException ignored) {
      // ignored
    }

    Put initialPut =
        preparePutWithNullValues(firstOperator, firstDataType, secondOperator, secondDataType)
            .withCondition(ConditionBuilder.putIfNotExists());
    storage.put(initialPut);
    return initialPut.getColumns();
  }

  private Put preparePutWithNullValues(
      Operator firstOperator,
      DataType firstDataType,
      @Nullable Operator secondOperator,
      @Nullable DataType secondDataType) {
    return new Put(
            Key.ofText(
                PARTITION_KEY,
                getPartitionKeyValue(firstOperator, firstDataType, secondOperator, secondDataType)))
        .withBooleanValue(COL_NAME1, null)
        .withIntValue(COL_NAME2, null)
        .withBigIntValue(COL_NAME3, null)
        .withFloatValue(COL_NAME4, null)
        .withDoubleValue(COL_NAME5, null)
        .withTextValue(COL_NAME6, null)
        .withBlobValue(COL_NAME7, (ByteBuffer) null)
        .withConsistency(Consistency.LINEARIZABLE)
        .forNamespace(namespace)
        .forTable(TABLE);
  }

  private Map> putInitialDataWithoutValues(Operator operator, DataType dataType)
      throws ExecutionException {
    return putInitialDataWithoutValues(operator, dataType, null, null);
  }

  private Map> putInitialDataWithoutValues(
      Operator firstOperator,
      DataType firstDataType,
      @Nullable Operator secondOperator,
      @Nullable DataType secondDataType)
      throws ExecutionException {
    try {
      storage.delete(
          prepareDelete(firstOperator, firstDataType, secondOperator, secondDataType)
              .withCondition(ConditionBuilder.deleteIfExists()));
    } catch (NoMutationException ignored) {
      // ignored
    }

    Put initialPut =
        preparePutWithoutValues(firstOperator, firstDataType, secondOperator, secondDataType)
            .withCondition(ConditionBuilder.putIfNotExists());
    storage.put(initialPut);
    return ImmutableMap.>builder()
        .put(
            PARTITION_KEY,
            TextColumn.of(
                PARTITION_KEY,
                getPartitionKeyValue(firstOperator, firstDataType, secondOperator, secondDataType)))
        .put(COL_NAME1, BooleanColumn.ofNull(COL_NAME1))
        .put(COL_NAME2, IntColumn.ofNull(COL_NAME2))
        .put(COL_NAME3, BigIntColumn.ofNull(COL_NAME3))
        .put(COL_NAME4, FloatColumn.ofNull(COL_NAME4))
        .put(COL_NAME5, DoubleColumn.ofNull(COL_NAME5))
        .put(COL_NAME6, TextColumn.ofNull(COL_NAME6))
        .put(COL_NAME7, BlobColumn.ofNull(COL_NAME7))
        .build();
  }

  private Put preparePutWithoutValues(
      Operator firstOperator,
      DataType firstDataType,
      @Nullable Operator secondOperator,
      @Nullable DataType secondDataType) {
    return new Put(
            Key.ofText(
                PARTITION_KEY,
                getPartitionKeyValue(firstOperator, firstDataType, secondOperator, secondDataType)))
        .withConsistency(Consistency.LINEARIZABLE)
        .forNamespace(namespace)
        .forTable(TABLE);
  }

  private String getPartitionKeyValue(
      Operator firstOperator,
      DataType firstDataType,
      @Nullable Operator secondOperator,
      @Nullable DataType secondDataType) {
    return firstOperator.name()
        + "-"
        + firstDataType.name()
        + (secondOperator == null ? "" : "-" + secondOperator.name())
        + (secondDataType == null ? "" : "-" + secondDataType.name());
  }

  private String getColumnName(DataType dataType) {
    switch (dataType) {
      case BOOLEAN:
        return COL_NAME1;
      case INT:
        return COL_NAME2;
      case BIGINT:
        return COL_NAME3;
      case FLOAT:
        return COL_NAME4;
      case DOUBLE:
        return COL_NAME5;
      case TEXT:
        return COL_NAME6;
      case BLOB:
        return COL_NAME7;
      default:
        throw new AssertionError();
    }
  }

  private ConditionalExpression buildConditionalExpression(
      Column columnToCompare, Operator operator) {
    if (operator == Operator.IS_NULL || operator == Operator.IS_NOT_NULL) {
      switch (columnToCompare.getDataType()) {
        case BOOLEAN:
          return ConditionBuilder.buildConditionalExpression(
              BooleanColumn.ofNull(columnToCompare.getName()), operator);
        case INT:
          return ConditionBuilder.buildConditionalExpression(
              IntColumn.ofNull(columnToCompare.getName()), operator);
        case BIGINT:
          return ConditionBuilder.buildConditionalExpression(
              BigIntColumn.ofNull(columnToCompare.getName()), operator);
        case FLOAT:
          return ConditionBuilder.buildConditionalExpression(
              FloatColumn.ofNull(columnToCompare.getName()), operator);
        case DOUBLE:
          return ConditionBuilder.buildConditionalExpression(
              DoubleColumn.ofNull(columnToCompare.getName()), operator);
        case TEXT:
          return ConditionBuilder.buildConditionalExpression(
              TextColumn.ofNull(columnToCompare.getName()), operator);
        case BLOB:
          return ConditionBuilder.buildConditionalExpression(
              BlobColumn.ofNull(columnToCompare.getName()), operator);
        default:
          throw new AssertionError();
      }
    }
    return ConditionBuilder.buildConditionalExpression(columnToCompare, operator);
  }

  protected boolean shouldMutate(
      Column initialColumn, Column columnToCompare, Operator operator) {
    switch (operator) {
      case EQ:
        return !initialColumn.hasNullValue()
            && Ordering.natural().compare(initialColumn, columnToCompare) == 0;
      case NE:
        return !initialColumn.hasNullValue()
            && Ordering.natural().compare(initialColumn, columnToCompare) != 0;
      case GT:
        return !initialColumn.hasNullValue()
            && Ordering.natural().compare(initialColumn, columnToCompare) > 0;
      case GTE:
        return !initialColumn.hasNullValue()
            && Ordering.natural().compare(initialColumn, columnToCompare) >= 0;
      case LT:
        return !initialColumn.hasNullValue()
            && Ordering.natural().compare(initialColumn, columnToCompare) < 0;
      case LTE:
        return !initialColumn.hasNullValue()
            && Ordering.natural().compare(initialColumn, columnToCompare) <= 0;
      case IS_NULL:
        return initialColumn.hasNullValue();
      case IS_NOT_NULL:
        return !initialColumn.hasNullValue();
      default:
        throw new AssertionError();
    }
  }

  protected Column getColumnWithRandomValue(
      Random random, String columnName, DataType dataType) {
    return TestUtils.getColumnWithRandomValue(random, columnName, dataType, true);
  }

  private String description(
      Map> initialData,
      Mutation mutation,
      Column columnToCompare,
      Operator operator) {
    return String.format(
        "initialData: %s, mutation: %s, columnToCompare: %s, operator: %s",
        initialData, mutation, columnToCompare, operator);
  }

  private String description(
      Map> initialData,
      Mutation mutation,
      Column firstColumnToCompare,
      Operator firstOperator,
      Column secondColumnToCompare,
      Operator secondOperator) {
    return String.format(
        "initialData: %s, mutation: %s, firstColumnToCompare: %s, firstOperator: %s, "
            + "secondColumnToCompare: %s, secondOperator: %s",
        initialData,
        mutation,
        firstColumnToCompare,
        firstOperator,
        secondColumnToCompare,
        secondOperator);
  }

  private void executeInParallel(TestForSingleCondition test)
      throws java.util.concurrent.ExecutionException, InterruptedException {
    List> testCallables = new ArrayList<>();
    for (OperatorAndDataType operatorAndDataType : operatorAndDataTypeList) {
      testCallables.add(
          () -> {
            random.get().setSeed(seed);

            test.execute(operatorAndDataType.getOperator(), operatorAndDataType.getDataType());
            return null;
          });
    }
    executeInParallel(testCallables);
  }

  private void executeInParallel(TestForMultipleConditions test)
      throws java.util.concurrent.ExecutionException, InterruptedException {
    List> testCallables = new ArrayList<>();
    for (OperatorAndDataType firstOperatorAndDataType : operatorAndDataTypeList) {
      for (OperatorAndDataType secondOperatorAndDataType : operatorAndDataTypeList) {
        testCallables.add(
            () -> {
              random.get().setSeed(seed);

              test.execute(
                  firstOperatorAndDataType.getOperator(),
                  firstOperatorAndDataType.getDataType(),
                  secondOperatorAndDataType.getOperator(),
                  secondOperatorAndDataType.getDataType());
              return null;
            });
      }
    }
    executeInParallel(testCallables);
  }

  private void executeInParallel(List> testCallables)
      throws InterruptedException, java.util.concurrent.ExecutionException {
    List> futures = executorService.invokeAll(testCallables);
    for (Future future : futures) {
      future.get();
    }
  }

  @FunctionalInterface
  private interface TestForSingleCondition {
    void execute(Operator operator, DataType dataType) throws Exception;
  }

  @FunctionalInterface
  private interface TestForMultipleConditions {
    void execute(
        Operator firstOperator,
        DataType firstDataType,
        Operator secondOperator,
        DataType secondDataType)
        throws Exception;
  }

  public static class OperatorAndDataType {
    private final Operator operator;
    private final DataType dataType;

    public OperatorAndDataType(Operator operator, DataType dataType) {
      this.operator = operator;
      this.dataType = dataType;
    }

    public Operator getOperator() {
      return operator;
    }

    public DataType getDataType() {
      return dataType;
    }
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy