我现在正在寻找更多时间,但没有结果。请帮忙...
这是我要测试的类(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 个选择:
重构您的代码并注入(inject)一些连接工厂。所以你会是 能够在您的测试中模拟它。
在测试中扩展类 DBSelectSchema 并覆盖方法
mensaDB()
因此它将返回模拟连接使用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/