DatabaseOperatorImplTest.java

134 lines | 4.327 kB Blame History Raw Download
/*
 * Copyright 2017 LinkedIn Corp.
 *
 * 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 azkaban.db;

import static org.mockito.Mockito.*;

import java.sql.Connection;
import java.sql.SQLException;
import java.util.ArrayList;
import java.util.List;
import javax.sql.DataSource;
import org.apache.commons.dbutils.QueryRunner;
import org.apache.commons.dbutils.ResultSetHandler;
import org.junit.Assert;
import org.junit.Before;
import org.junit.Test;


public class DatabaseOperatorImplTest {

  private static final List<Integer> list = new ArrayList<>();
  private static final int index_2 = 15;
  private static int index_1 = 3;
  private final AzkabanDataSource datasource = new AzDBTestUtility.EmbeddedH2BasicDataSource();
  private final ResultSetHandler<Integer> handler = rs -> {
    if (!rs.next()) {
      return 0;
    }
    return rs.getInt(1);
  };
  private DatabaseOperator dbOperator;
  private QueryRunner queryRunner;
  private Connection conn;

  @Before
  public void setUp() throws Exception {
    this.queryRunner = mock(QueryRunner.class);

    this.conn = this.datasource.getConnection();
    final DataSource mockDataSource = mock(this.datasource.getClass());

    when(this.queryRunner.getDataSource()).thenReturn(mockDataSource);
    when(mockDataSource.getConnection()).thenReturn(this.conn);

    this.dbOperator = new DatabaseOperatorImpl(this.queryRunner);

    list.add(index_1);
    list.add(index_2);

    // valid query returns correct value
    when(this.queryRunner.query("select * from blah where ? = ?", this.handler, "id", 2))
        .thenReturn(index_2);

    // If select an non-existing entry, handler returns 0.
    when(this.queryRunner.query("select * from blah where ? = ?", this.handler, "id", 3))
        .thenReturn(0);

    //If typos, throw Exceptions.
    doThrow(SQLException.class).when(this.queryRunner)
        .query("sele * from blah where ? = ?", this.handler, "id", 2);

    doAnswer(invocation -> {
      index_1 = 26;
      return 1;
    }).when(this.queryRunner).update("update blah set ? = ?", "1", 26);
  }

  @Test
  public void testValidQuery() throws Exception {
    final int res = this.dbOperator.query("select * from blah where ? = ?", this.handler, "id", 2);
    Assert.assertEquals(15, res);
    verify(this.queryRunner).query("select * from blah where ? = ?", this.handler, "id", 2);
  }

  @Test
  public void testInvalidQuery() throws Exception {
    final int res = this.dbOperator.query("select * from blah where ? = ?", this.handler, "id", 3);
    Assert.assertEquals(0, res);
  }

  @Test(expected = SQLException.class)
  public void testTypoSqlStatement() throws Exception {
    System.out.println("testTypoSqlStatement");
    this.dbOperator.query("sele * from blah where ? = ?", this.handler, "id", 2);
  }

  @Test
  public void testTransaction() throws Exception {
    when(this.queryRunner.update(this.conn, "update blah set ? = ?", "1", 26)).thenReturn(1);
    when(this.queryRunner.query(this.conn, "select * from blah where ? = ?", this.handler, "id", 1))
        .thenReturn(26);

    final SQLTransaction<Integer> transaction = transOperator -> {
      transOperator.update("update blah set ? = ?", "1", 26);
      return transOperator.query("select * from blah where ? = ?", this.handler, "id", 1);
    };

    final int res = this.dbOperator.transaction(transaction);
    Assert.assertEquals(26, res);
  }

  @Test
  public void testValidUpdate() throws Exception {
    final int res = this.dbOperator.update("update blah set ? = ?", "1", 26);

    // 1 row is affected
    Assert.assertEquals(1, res);
    Assert.assertEquals(26, index_1);
    verify(this.queryRunner).update("update blah set ? = ?", "1", 26);
  }

  @Test
  public void testInvalidUpdate() throws Exception {
    final int res = this.dbOperator.update("update blah set ? = ?", "3", 26);

    // 0 row is affected
    Assert.assertEquals(0, res);
  }
}