Skip to content

Commit 9846e93

Browse files
committed
DBUnit
1 parent bb18ef3 commit 9846e93

File tree

4 files changed

+157
-0
lines changed

4 files changed

+157
-0
lines changed
Lines changed: 157 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,157 @@
1+
---
2+
title: "DBUnitでいろいろはまった"
3+
date: 2024/12/04 00:00:00
4+
postid: a
5+
tag:
6+
- Java
7+
- DBUnit
8+
- テスト
9+
category:
10+
- Programming
11+
thumbnail: /images/20241204a/thumbnail.jpg
12+
author: 澁川喜規
13+
lede: "Javaでデータベースを使うプロジェクトだったのでDBUnit使うぜ、と導入したのですが、細かいところで引っかかったりしたので備忘メモです。"
14+
---
15+
<img src="/images/20241204a/dbunit.jpg" alt="" width="500" height="300">
16+
17+
本ブログは、[ソフトウェアテスト - Qiita Advent Calendar 2024 - Qiita](https://qiita.com/advent-calendar/2024/softwaretesting) の4日目です。
18+
19+
3日目は、ぱいん (pinecandy)さんの[テストについて相談を受けたときにいつもしていること - 飴ブロ(仮](https://pineapplecandy.hatenadiary.jp/entry/test-talk-over)です。
20+
21+
Javaでデータベースを使うプロジェクトだったのでDBUnit使うぜ、と導入したのですが、細かいところで引っかかったりしたので備忘メモです。
22+
23+
# org.dbunit.database.AmbiguousTableNameException
24+
25+
このプロジェクトでは複数のスキーマがあり、一部重複したテーブル名があったため、このような例外が出ていました。共通スキーマsystemというのがあったとしたら、それを``JdbcDatabaseTester``に渡して設定します。
26+
27+
```java
28+
databaseTester = new JdbcDatabaseTester(this.dataSourceDriver,
29+
this.dataSourceUrl,
30+
this.dataSourceUser,
31+
this.dataSourcePassword,
32+
"system"); // これ
33+
34+
IDataSet dataSet = new FlatXmlDataSetBuilder().build(
35+
new File("src/test/resources/dbunit/fixture.xml"));
36+
databaseTester.setDataSet(replacementDataSet);
37+
databaseTester.setSetUpOperation(DatabaseOperation.CLEAN_INSERT);
38+
databaseTester.onSetup();
39+
```
40+
41+
# 複数のスキーマにデータを入れたい!
42+
43+
1つのデータを定義したけど、一部のデータは別のスキーマに入れたい、という場合です。取込は2回書くのですが、データセットをデータテスターに渡すときに、フィルターでラップしてから渡します。片方は``IncludeTableFilter``、もう片方は``ExcludeTableFilter``を使います。
44+
45+
雑にベタに書いたので、もしかしたらFlatXmlDataSetBuilderとかは
46+
47+
```java
48+
var appOnlyTables = new String[] {
49+
"t_work",
50+
"t_log"};
51+
databaseTesterApp = new JdbcDatabaseTester(this.dataSourceDriver,
52+
this.dataSourceUrl,
53+
this.dataSourceUser,
54+
this.dataSourcePassword,
55+
"app");
56+
IDataSet dataSetApp = new FlatXmlDataSetBuilder().build(
57+
new File("src/test/resources/dbunit/fixture.xml"));
58+
59+
// このフィルターでテーブルを選別
60+
var dataFilterApp = new IncludeTableFilter(appOnlyTables);
61+
var filterdDatasetApp = new FilteredDataSet(dataFilterApp, dataSetApp);
62+
63+
databaseTesterApp.setDataSet(filterdDatasetApp);
64+
databaseTesterApp.setSetUpOperation(DatabaseOperation.CLEAN_INSERT);
65+
databaseTesterApp.onSetup();
66+
67+
databaseTesterSys = new JdbcDatabaseTester(this.dataSourceDriver,
68+
this.dataSourceUrl,
69+
this.dataSourceUser,
70+
this.dataSourcePassword,
71+
"system");
72+
IDataSet dataSetSys = new FlatXmlDataSetBuilder().build(
73+
new File("src/test/resources/dbunit/fixture.xml"));
74+
75+
// このフィルターでテーブルを選別
76+
var dataFilterSys = new ExcludeTableFilter(appOnlyTables);
77+
var filterdDatasetSys = new FilteredDataSet(dataFilterSys, dataSetSys);
78+
79+
databaseTesterSys.setDataSet(filterdDatasetSys);
80+
databaseTesterSys.setSetUpOperation(DatabaseOperation.CLEAN_INSERT);
81+
databaseTesterSys.onSetup();
82+
```
83+
84+
# NULLを表現したい!
85+
86+
DBUnit標準だと、読み込んだとおりに文字列で解釈しますので、空文字列だと空文字列。NULLとか書くと"NULL"という文字列になります。データセットを置換するラッパーを使い、特定のテキストをオブジェクトに変換するラッパーを作成してデータセットにかぶせることで、nullが表現できます。
87+
88+
```java
89+
IDataSet dataSet = new FlatXmlDataSetBuilder().build(
90+
new File("src/test/resources/dbunit/fixture.xml"));
91+
92+
// このフィルターで変換
93+
var replacementDataSet = new ReplacementDataSet(dataSet);
94+
replacementDataSet.addReplacementObject("[NULL]", null);
95+
96+
databaseTesterSys.setDataSet(replacementDataSet);
97+
databaseTesterSys.setSetUpOperation(DatabaseOperation.CLEAN_INSERT);
98+
databaseTesterSys.onSetup();
99+
```
100+
101+
# WindowsとLinuxのフォルダを吸収したい
102+
103+
データベースのログテーブルには処理したファイル名が入るが、WindowsだとC:\になってしまうし、Linuxだと/tmpになってしまうケースに対応する場合も、おなじデータセットを置換するラッパーを使います。
104+
105+
先ほどは``addReplacementObject()``でしたが、こちらでは``addReplacementSubstring()``メソッドです。部分一致で変換できるので``[tmp]/deleted-file.txt``みたいにデータセットに書いておくと、OSの作業フォルダに書き換えてくれます。
106+
107+
```java
108+
this.tmpFolderPath = Paths.get(System.getProperty("java.io.tmpdir")).resolve("機能ID");
109+
110+
IDataSet dataSet = new FlatXmlDataSetBuilder().build(
111+
new File("src/test/resources/dbunit/fixture.xml"));
112+
113+
var replacementDataSet = new ReplacementDataSet(dataSet);
114+
replacementDataSet.addReplacementSubstring("[tmp]",
115+
this.tmpFolderPath.toString());
116+
117+
databaseTesterSys.setDataSet(replacementDataSet);
118+
databaseTesterSys.setSetUpOperation(DatabaseOperation.CLEAN_INSERT);
119+
databaseTesterSys.onSetup();
120+
```
121+
122+
# XMLで、最初のタグにない要素は無視される
123+
124+
データセットはExcelとかよりもXMLの方が便利ですね。テーブルごとにまとめれば良いかというと、従属テーブルなんかは親テーブルのそばにあった方が分かりやすいですしね。
125+
126+
ということでXML型式を好んで使っていたのですが、1つ罠がありました。次のようなデータがあったとします。
127+
ユーザーを退会した場合にデータが入るNULLABLEな ``unregisterd_at`` カラムがあり、2つ目のユーザーにだけあります。ですが、このデータは無視されます。というのも、なぜかそのテーブルの要素が持っている属性以外は無視されるとのことでした。↑↑に書いたフィルタを追加して `unregisterd_at="[NULL]"` という属性を最初のユーザーにも設定しなければなりません。
128+
129+
```xml
130+
<?xml version="1.0" encoding="UTF-8"?>
131+
<dataset>
132+
<!-- ユーザーマスタ -->
133+
<m_user id="10" name="ユーザー1" />
134+
<m_user id="20" name="ユーザー2" unregisterd_at="2024/12/01"/>
135+
</dataset>
136+
```
137+
138+
# 新たに作られたパーティションが、careateDataSet()の結果に表れない
139+
140+
PL/pgSQLで書かれたパーティション改廃関数を実行した後に、``careateDataSet()``を呼んで正しくテーブルが作られたり削除されたか確認するテストを作成したのですが、古いパーティションの削除処理は正しく処理され、結果のテーブルリストから消えていたのだが、新たに分割して作った新しいパーティションが見えなかったということがありました。
141+
142+
コネクションプールが悪さしていたみたいです。HikariCPをラップしたミドルウェア経由でPL/pgSQLの関数を呼んでいたのを、Javaの機能を直接呼んで実行してからDBUnitで実行結果を取得したら削除したパーティションも追加したパーティションの情報も反映された結果が取得できました。
143+
144+
```java
145+
try (Connection connection = DriverManager.getConnection(this.dataSourceUrl + "?currentSchema=cmn",
146+
this.dataSourceUser, this.dataSourcePassword)) {
147+
try (Statement stmt = connection.createStatement()) {
148+
stmt.execute("SELECT 関数呼び出し('param1', 'param2')");
149+
}
150+
}
151+
```
152+
153+
# まとめ
154+
155+
単にトラブルシュートの備忘録なので、かっこいい締めの言葉とかオチとかはありません。
156+
157+
いろいろハマりはしましたが、なかなか便利なやつなので、Goとか他の言語にも欲しいですね。

source/images/20241204a/dbunit-logo - コピー.jpg:Zone.Identifier

Whitespace-only changes.

source/images/20241204a/dbunit.jpg

13.6 KB
Loading
11.3 KB
Loading

0 commit comments

Comments
 (0)