java - Junit Mockito 测试一切

标签 java testing junit mockito code-coverage

我现在正在寻找更多时间,但没有结果。请帮忙...

这是我要测试的类(class):

public class DBSelectSchema extends Database {

    private static final Logger LOG = Logger
            .getLogger(DBSelectSchema.class.getName());
    private Connection conn = null;

    public DBSelectSchema() {
        super();
    }

    /**
     * This method will return the version of the database.
     * 
     * @return version
     * @throws Exception
     */
    public JSONObject getVersionFromDB() throws SQLException {
        ResultSet rs = null;
        JSONObject version = new JSONObject();
        PreparedStatement query = null;

        try {
            conn = mensaDB();
            query = conn.prepareStatement("SELECT number FROM version");

            rs = query.executeQuery();

            if (rs.isBeforeFirst()) {
                rs.next();
                version.put(HTTP.HTTP, HTTP.OK);
                version.put("version", rs.getString("number"));
            } else {
                version.put(HTTP.HTTP, HTTP.NO_CONTENT);
                version.put(HTTP.ERROR, "Die SQL Abfrage lieferte kein Result!");
            }

            rs.close();
            query.close();
            conn.close();

        } catch (SQLException sqlError) {
            String message = ERROR.SQL_EXCEPTION;
            LOG.log(Level.SEVERE, message, sqlError);
            return version;

        } catch (JSONException jsonError) {
            String message = ERROR.JSON_EXCEPTION;
            LOG.log(Level.SEVERE, message, jsonError);
            return version;
        }

        return version;
    }

我试图在每个分支中实现 100% 的代码覆盖率。 我如何模拟 ResultSet rs、JSONObject 版本和PreparedStatement 查询来执行/返回我想要的内容:

目前我正在这样测试:

@Test
    public void getVersionFromDB_RS_FALSE() throws SQLException, JSONException {
        MockitoAnnotations.initMocks(this);

        Mockito.when(dbSelMocked.mensaDB()).thenReturn(conn);
        Mockito.when(conn.prepareStatement(Mockito.anyString())).thenReturn(query);
        Mockito.when(query.executeQuery()).thenReturn(rs);
        Mockito.when(rs.isBeforeFirst()).thenReturn(false);

        JSONObject returnObj = dbSelMocked.getVersionFromDB();

        assert(...);
    }

但这仅在 3 个变量是类变量(如 Connection conn)而不是局部变量时有效。但我不希望它们(甚至连接)不是全局的。

===编辑1===

如果所有变量都是本地变量,它的工作原理如下:

@Test
    public void getVersionFromDB_RS_FALSE() throws SQLException, JSONException {
        System.out.println("####################");
        System.out.println("started test: getVersionFromDB_RS_FALSE");
        System.out.println("####################");

        Connection conn = Mockito.mock(Connection.class);
        PreparedStatement query = Mockito.mock(PreparedStatement.class);
        ResultSet rs = Mockito.mock(ResultSet.class);

        MockitoAnnotations.initMocks(this);


        Mockito.when(dbSelMocked.mensaDB()).thenReturn(conn);
        Mockito.when(conn.prepareStatement(Mockito.anyString())).thenReturn(query);
        Mockito.when(query.executeQuery()).thenReturn(rs);
        Mockito.when(rs.isBeforeFirst()).thenReturn(false);

        JSONObject returnObj = dbSelMocked.getVersionFromDB();

        assertTrue(returnObj.has("error"));
    }

但是我无法再在另一个测试中模拟 JSONObject 版本:( 我怎样才能做到这一点?

@Test
    public void getVersionFromDB_JSON_EXCEPTION() throws SQLException, JSONException {
        System.out.println("####################");
        System.out.println("started test: getVersionFromDB_JSON_EXCEPTION");
        System.out.println("####################");
        JSONObject version = Mockito.mock(JSONObject.class);

        MockitoAnnotations.initMocks(this);

        doThrow(new JSONException("DBSelectSchemaIT THROWS JSONException")).when(version).put(anyString(), any());

        JSONObject returnObj = dbSelMocked.getVersionFromDB();

        System.out.println(returnObj.toString());

        assertTrue(returnObj.equals(null));
    }

我认为它在真正的方法中被覆盖...因为它不会抛出异常并且该方法不会失败。

最佳答案

您的测试代码有多个问题。

  • 测试冗长且脆弱
  • 多个测试需要相同(详细)的设置
  • 您不测试真实对象,而是使用类的模拟进行测试

前两个问题可以通过将重复代码提取到设置方法中来解决(我为 Mockito 添加了静态导入以减少噪音):

@Before
public void setUp() throws Exception {
    Connection conn = mock(Connection.class);
    PreparedStatement query = mock(PreparedStatement.class);
    when(dbSelMocked.mensaDB()).thenReturn(conn);
    when(conn.prepareStatement(anyString())).thenReturn(query);
    when(query.executeQuery()).thenReturn(rs);

    rs = mock(ResultSet.class); // rs is field
}

现在,在每个测试中,您可以配置 rs 以返回您需要的任何内容:

@Test
public void getVersionFromDB_RS_FALSE() throws Exception {
    // Given
    when(rs.isBeforeFirst()).thenReturn(false);

    // When
    JSONObject returnObj = dbSelMocked.getVersionFromDB();

    // Then
    assertTrue(returnObj.has("error"));
}

现在最重要的问题:您正在模拟类 DBSelectSchema 以返回连接模拟。测试中的模拟类可能会导致不同的难以发现的问题。

要解决这个问题,您有 3 个选择:

  1. 重构您的代码并注入(inject)一些连接工厂。所以你会是 能够在您的测试中模拟它。

  2. 在测试中扩展类 DBSelectSchema 并覆盖方法 mensaDB() 因此它将返回模拟连接

  3. 使用H2等嵌入式数据库并将测试数据放入“number”表中 在调用 getVersionFromDB() 之前

选项#1

将连接的创建提取到一个单独的类,并在您的DBSelectSchema中使用它:

public class ConnectionFactory {
    public Connection getConnection() {
       // here goes implementation of mensaDB()
    }
}

然后将其注入(inject)到您的 DBSelectSchema:

public DBSelectSchema(ConnectionFactory connFactory) {
    this.connFactory = connFactory;
}

现在您的测试可以使用真实 DBSelectSchema 类和模拟 ConnectionFactory

    ConnectionFactory connFactory = mock(ConnectionFactory.class);
    dbSel = new DBSelectSchema(connFactory); 

选项#2

您可以在测试中创建几乎真实的类:

    final Connection conn = mock(Connection.class);
    dbSel = new DBSelectSchema() {
        @Override
        public Connection mensaDB() {
            return conn;
        }
    }; 

选项#3

此选项是最可取的,因为您将调用真正的 SQL 命令并模拟整个数据库而不是类。在这里使用普通 JDBC 需要付出一些努力,但这是值得的。请记住,SQL 方言可能与生产中使用的数据库不同。

@Before
public void setUp() throws Exception {
    Class.forName("org.h2.Driver");
    conn = DriverManager.getConnection("jdbc:h2:mem:test;INIT=RUNSCRIPT FROM 'classpath:schema.sql'");
}

@After
public void tearDown() throws Exception {
    conn.close();
}

然后在您的测试中,您只需将所需的记录添加到数据库即可:

 @Test
 public void getVersionFromDB() throws Exception {
    // Given
    conn.prepareStatement("INSERT INTO version(number) VALUES (1)").execute();

    // When
    JSONObject returnObj = dbSel.getVersionFromDB();

    // Then
    assert(...);
}

显然,DBSelectSchema必须使用相同的连接,因此可以与选项#1和#2结合使用,

关于java - Junit Mockito 测试一切,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/25520477/

相关文章:

asp.net-mvc - asp.net mvc 用户负载测试

java - 使用多种数据库类型的Spring Boot集成测试

go - 当从 Golang 中的另一个包调用时,全局变量变为 nil

junit - 我们什么时候应该使用 Mockery 和 JUnit4Mockery?

java - 如何对 gwt MVP Activity 和地点项目进行测试?

java - Spring 和 hibernate 错误 "Declaring class is not found in the inheritance state hierarchy"

java - LibGDX 闪烁

java - Java中数字的最大质因数

java - 具有 Maincivity 的两项 Activity 会导致后退按钮无法正常工作

java - 为什么 IntelliJ 能够看到我的应用程序上下文,但当我运行测试时他们却看不到它?