【发布时间】:2017-08-09 15:00:21
【问题描述】:
我正在使用 Spock 和 Mockito 框架编写单元测试,但偶然发现了 Mockito 中的一个限制,我无法优雅地解决。
以下代码解析 .csv 文件并返回 TradedInstrument 对象的集合:
@ManagedOperation
public Collection<TradedInstrument> loadTradedInstrumentFromBatchFile() {
Collection<TradedInstrument> tradedInstrumentsFrombatchFile = new ArrayList<>();
try (BufferedReader br = new BufferedReader(new InputStreamReader(
this.getClass().getResourceAsStream("/" + tradedInstrumentBatchFilename)))) {
String line;
while ((line = br.readLine()) != null) {
String[] instrumentAttributes = line.split(",");
TradedInstrument tradedInstrument = new TradedInstrument();
tradedInstrument.setInstId(instrumentAttributes[0]);
tradedInstrument.setPriceSource(PriceSource.valueOf(instrumentAttributes[1]));
tradedInstrument.setPrice(new BigDecimal(instrumentAttributes[2]));
tradedInstrument.setDateCreated(businessDateDao.getBusinessDate());
insertTradedInstrument(tradedInstrument);
tradedInstrumentsFrombatchFile.add(tradedInstrument);
LOGGER.info("Loaded traded instrument: " + tradedInstrument + " from batch file: " + tradedInstrumentBatchFilename);
}
} catch (NullPointerException e) {
String errorMessage = "An ERROR occurred locating the traded instrument upload batch file: " + tradedInstrumentBatchFilename;
LOGGER.error(errorMessage);
throw new PriceServiceException(errorMessage, e);
} catch (IOException e) {
String errorMessage = "An ERROR occurred reading the traded instrument upload batch file: " + tradedInstrumentBatchFilename;
LOGGER.error(errorMessage);
throw new PriceServiceException(errorMessage, e);
} catch (Exception e) {
String errorMessage = "An ERROR occurred creating traded instrument using data from upload batch file: " + tradedInstrumentBatchFilename;
LOGGER.error(errorMessage, e);
throw new PriceServiceException(errorMessage, e);
}
return tradedInstrumentsFrombatchFile;
}
我有一个名为“tradedInstrumentBatchFile.csv”的批处理文件,内容如下:
1234,ICAP,0.4956 2345,BBG,0.8456 8456,NASDAQ,0.3567 5967,REUTERS,0.8675
我在 Groovy 中编写了以下 Spock 测试:
def "should load traded instrument from batch file"() {
given:
TradedInstrumentSubscriber tradedInstrumentSubscriber = Mock()
TradedInstrumentTable tradedInstrumentDao = Mock()
VendorTopicDao vendorTopicDao = Mock()
TradedInstrumentListener listener = Mock()
BusinessDateDao businessDateDao = Mock()
PriceService priceService = new PriceService(tradedInstrumentSubscriber, listener, vendorTopicDao, tradedInstrumentDao, businessDateDao)
priceService.setTradedInstrumentBatchFilename("tradedInstrumentBatchFile.csv")
businessDateDao.getBusinessDate() >> new Date()
when:
priceService.loadTradedInstrumentFromBatchFile()
then:
1 * tradedInstrumentDao.insert(_ as TradedInstrument) >> { TradedInstrument arg ->
assert arg.instId == "1234"
assert arg.priceSource == PriceSource.ICAP
assert arg.price == BigDecimal.valueOf(0.4956)
}
1 * tradedInstrumentDao.insert(_ as TradedInstrument) >> { TradedInstrument arg ->
assert arg.instId == "2345"
assert arg.priceSource == PriceSource.BBG
assert arg.price == BigDecimal.valueOf(0.8456)
}
1 * tradedInstrumentDao.insert(_ as TradedInstrument) >> { TradedInstrument arg ->
assert arg.instId == "8456"
assert arg.priceSource == PriceSource.NASDAQ
assert arg.price == BigDecimal.valueOf(0.3567)
}
1 * tradedInstrumentDao.insert(_ as TradedInstrument) >> { TradedInstrument arg ->
assert arg.instId == "5967"
assert arg.priceSource == PriceSource.REUTERS
assert arg.price == BigDecimal.valueOf(0.8675)
}
}
我已经尝试使用 Mockito 进行等效的 Junit 测试:
@Test
public void shouldLoadTradedInstrumentFromBatchFile() {
// Given
VendorTopicDao vendorTopicDao = mock(VendorTopicDao.class);
TradedInstrumentSubscriber tradedInstrumentSubscriber = mock(TradedInstrumentSubscriber.class);
TradedInstrumentTable tradedInstrumentDao = mock(TradedInstrumentTable.class);
TradedInstrumentListener listener = mock(TradedInstrumentListener.class);
BusinessDateDao businessDateDao = mock(BusinessDateDao.class);
PriceService priceService = new PriceService(tradedInstrumentSubscriber, listener, vendorTopicDao, tradedInstrumentDao);
priceService.setTradedInstrumentBatchFilename("tradedInstrumentBatchFile.csv");
ArgumentCaptor<TradedInstrument> captor = ArgumentCaptor.forClass(TradedInstrument.class);
when(businessDateDao.getBusinessDate()).thenReturn(new Date());
// When
priceService.loadTradedInstrumentFromBatchFile();
// Then
verify(tradedInstrumentDao, times(4)).insert(captor.capture());
List<TradedInstrument> tradedInstruments = captor.getAllValues();
TradedInstrument tradedInstrument1 = new TradedInstrument();
tradedInstrument1.setInstId("1234");
tradedInstrument1.setPriceSource(PriceSource.ICAP);
tradedInstrument1.setPrice(BigDecimal.valueOf(0.4956));
TradedInstrument tradedInstrument2 = new TradedInstrument();
tradedInstrument2.setInstId("2345");
tradedInstrument2.setPriceSource(PriceSource.BBG);
tradedInstrument2.setPrice(BigDecimal.valueOf(0.8456));
TradedInstrument tradedInstrument3= new TradedInstrument();
tradedInstrument2.setInstId("8456");
tradedInstrument2.setPriceSource(PriceSource.NASDAQ);
tradedInstrument2.setPrice(BigDecimal.valueOf(0.3567));
TradedInstrument tradedInstrument4= new TradedInstrument();
tradedInstrument2.setInstId("5967");
tradedInstrument2.setPriceSource(PriceSource.REUTERS);
tradedInstrument2.setPrice(BigDecimal.valueOf(0.8675));
assertThat(tradedInstruments, hasItem(tradedInstrument1));
assertThat(tradedInstruments, hasItem(tradedInstrument2));
assertThat(tradedInstruments, hasItem(tradedInstrument3));
assertThat(tradedInstruments, hasItem(tradedInstrument4));
}
Junit 测试失败,因为该测试在比较中包含“dateCreated”字段,而 Spock 测试选择性地省略了该字段。
“dateCreated”字段应该是创建 TradedInstrument 的实际时间,因此有意将其排除在比较之外。
我能看到的唯一解决方案是在 businessDateDao 上添加以下交互:
when(businessDateDao.getBusinessDate()).thenReturn(null);
在 Junit/Mockito 中是否有等效的方法使用 Matcher 来选择性地省略字段的比较?
【问题讨论】:
标签: java groovy mockito spock hamcrest