DbUnit を使用して DAO レイヤーをテストしています。XML データセットからデータベースを事前入力し、いくつかのアクションを実行してから、既知の結果に対してアサートしています。
Assertion.assertEquals(expectedDataSet, actualDataSet);
データセットには、浮動小数点数の列が含まれています。次に、これらの列を比較すると、次のようになります。
junit.framework.ComparisonFailure: value (table=OrderLine_T, row=2, col=price) expected:<2.99[]> but was:<2.99[0000009536743]>.
値は同じですが、浮動小数点数はバイナリ形式で正確に表すことができないため、アサーションは失敗します。JUnit にはassertEquals(double expected, double actual, double delta)
. 浮動小数点数の比較のために DbUnit にデルタを設定するにはどうすればよいですか?
初期データセット:
<dataset>
<Customer_T id="1" name="Anthony"/>
<Customer_T id="2" name="John"/>
<Order_T id="1" date="2012-06-07 14:30" customer_id="1" />
<Order_T id="2" date="2012-06-07 15:31" customer_id="2" />
<OrderLine_T id="1" order_id="1" product_id="1" price="2.99" quantity="5" />
<OrderLine_T id="2" order_id="2" product_id="2" price="3.49" quantity="10" />
</dataset>
期待される結果:
<dataset>
<Customer_T id="1" name="Anthony"/>
<Customer_T id="2" name="John"/>
<Order_T id="1" date="2012-06-07 14:30" customer_id="1" />
<Order_T id="2" date="2012-06-07 15:31" customer_id="2" />
<OrderLine_T id="1" order_id="1" product_id="1" price="2.99" quantity="5" />
<OrderLine_T id="2" order_id="2" product_id="2" price="3.49" quantity="10" />
<!-- Below added -->
<Order_T id="3" date="1987-06-07 9:15:10" customer_id="2" />
<OrderLine_T id="3" order_id="3" product_id="1" price="2.99" quantity="2" />
<OrderLine_T id="4" order_id="3" product_id="5" price="3.55" quantity="8" />
</dataset>
コード:
/* Should save order correctly (including order lines) */
@Test
public void save() throws Exception {
/* Create new order */
Set<OrderLine> lines = new HashSet<OrderLine>();
lines.add(new OrderLine(1, (float)2.99, 2));
lines.add(new OrderLine(5, (float)3.55, 8));
Calendar cal = Calendar.getInstance();
cal.set(1987, 6, 7, 9, 15, 10);
Date date = cal.getTime();
Customer customer = customerDAO.findById(2); // John
Order order = new Order(date, lines, customer);
orderDAO.save(order);
entityManager.flush();
/* Assert order is saved */
IDataSet expectedDataSet = new FlatXmlDataSetBuilder().build(Thread.currentThread()
.getContextClassLoader()
.getResourceAsStream("data-set-afterAddOrder.xml"));
IDataSet actualDataSet = getDatabaseConnection().createDataSet();
Assertion.assertEquals(expectedDataSet, actualDataSet);
}
編集:
おそらく、インメモリ HSQLDB を使用していることについて言及する必要があります。MySQLを試してみたところ、成功しました。
成功せずに設定しようとしToleratedDelta
ました:
IDatabaseConnection connection = new DatabaseConnection(((SessionImpl) (entityManager.getDelegate())).connection());
HsqldbDataTypeFactory dataTypeFactory = new HsqldbDataTypeFactory();
dataTypeFactory.addToleratedDelta(new ToleratedDelta("OrderLine_T", "price", 0.01));
connection.getConfig().setProperty(DatabaseConfig.PROPERTY_DATATYPE_FACTORY, dataTypeFactory);
編集2:
スキーマをデータベースにエクスポートするために、hibernate.hbm2ddl.auto = create を使用していました。fredtからアイデアを得た後 、エンティティのフィールドpriceのタイプを BigDecimal に変更し、列の精度とスケール パラメータを追加しました。
@Column(precision=10, scale=4)
private BigDecimal price;
これは に変換されPRICE NUMERIC(10,4)
ます。fredtのおかげで問題は解決しました。