継続的な統合の道-データ・アクセス・レイヤのユニット・テスト(続き)
6743 ワード
前編では、テスト用データソースの構成が完了しました.次に、実行可能なテストの構築を続行します.
三、DBUnitによるデータ管理
テストのメンテナンスは、メンテナンスが容易で多重化可能な方法でデータを管理できることを期待しています.より良い方法がない前に、DBUnitを一時的に選択します.(反省:実は私はずっと起こっていないことを心配して、事が全然進まない.すでに存在している、最も簡単なところから始めて、正しい処理方法です.)
pom.xmlにはdbunitパッケージとspringtestdbunitパッケージが導入され、後者はDBUnitを注釈で使用することを提供します.
DBUnitはxmlファイルを使用してデータセットを管理し、サードパーティのライブラリを使用することでJSON形式を容易にサポートできます.ここではxmlを使用します.
このデータファイルは/src/test/resources/に格納され、テスト例と同じレベルのディレクトリに格納されます.区別を容易にするために、Daoクラス名-テストされたメソッド名-datasetを採用しました.xmlの命名方法、例えば:UserDao-findByname-dataxml.set.以降、テスト・インスタンスを変更する必要がある場合は、名前に基づいて対応するデータセットを簡単に見つけることができ、他のテスト・インスタンスに影響を与えることはありません.
注意:
1. ここのElementとそのAttribute名は、エンティティクラスではなくデータベースの構造に対応します.
2. 同じデータ・オブジェクトを初期化する場合、初期化する必要があるフィールドの数が異なります.たとえば、1つのデータを初期化する必要があるフィールドは8つで、もう1つは4つです.では、必ずフィールド数を前にしてください.
四、テスト用例の作成
例を書く前に、テストされたコードを見てみましょう.使用する2つのエンティティークラス:
テストされたFloorDao:
テスト例も簡単です.
コードにより,DatabaseSetupによるテストデータの導入が完了していることが明らかになった.ここでは、各テストメソッドの前に異なるファイルを導入し、すべてのメソッドが1つのファイルで含めることができる場合は、クラスの前にDatabaseSetupを使用してデータファイルを導入することもできます.
これで、完全なデータ層テストの例が提示され、実行できます.しかし、実際の過程はそれほど順調ではなく、次の文章では出会った問題をまとめなければならない.
三、DBUnitによるデータ管理
テストのメンテナンスは、メンテナンスが容易で多重化可能な方法でデータを管理できることを期待しています.より良い方法がない前に、DBUnitを一時的に選択します.(反省:実は私はずっと起こっていないことを心配して、事が全然進まない.すでに存在している、最も簡単なところから始めて、正しい処理方法です.)
pom.xmlにはdbunitパッケージとspringtestdbunitパッケージが導入され、後者はDBUnitを注釈で使用することを提供します.
<dependency>
<groupId>org.dbunit</groupId>
<artifactId>dbunit</artifactId>
<version>2.4.9</version>
<scope>test</scope>
</dependency>
<dependency>
<groupId>com.github.springtestdbunit</groupId>
<artifactId>spring-test-dbunit</artifactId>
<version>1.0.1</version>
<scope>test</scope>
</dependency>
DBUnitはxmlファイルを使用してデータセットを管理し、サードパーティのライブラリを使用することでJSON形式を容易にサポートできます.ここではxmlを使用します.
<?xml version="1.0" encoding="utf-8"?>
<dataset>
<building id="1" name="SOHO"/>
<building id="2" name="New Gate Plaza"/>
<floor id="1" floor_num="2" building="1"/>
<floor id="2" floor_num="3" building="1"/>
<floor id="3" floor_num="5" building="2"/>
</dataset>
このデータファイルは/src/test/resources/に格納され、テスト例と同じレベルのディレクトリに格納されます.区別を容易にするために、Daoクラス名-テストされたメソッド名-datasetを採用しました.xmlの命名方法、例えば:UserDao-findByname-dataxml.set.以降、テスト・インスタンスを変更する必要がある場合は、名前に基づいて対応するデータセットを簡単に見つけることができ、他のテスト・インスタンスに影響を与えることはありません.
注意:
1. ここのElementとそのAttribute名は、エンティティクラスではなくデータベースの構造に対応します.
2. 同じデータ・オブジェクトを初期化する場合、初期化する必要があるフィールドの数が異なります.たとえば、1つのデータを初期化する必要があるフィールドは8つで、もう1つは4つです.では、必ずフィールド数を前にしてください.
四、テスト用例の作成
例を書く前に、テストされたコードを見てみましょう.使用する2つのエンティティークラス:
package com.noyaxe.myapp.entity;
import javax.persistence.Entity;
import javax.persistence.Table;
@Entity
@Table(name = "building")
public class Building extends IdEntity {
private String name;
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
}
package com.noyaxe.myapp.entity;
import javax.persistence.Column;
import javax.persistence.Entity;
import javax.persistence.JoinColumn;
import javax.persistence.ManyToOne;
import javax.persistence.Table;
@Entity
@Table(name = "floor")
public class Floor extends IdEntity {
private Integer floorNum;
private Building building;
@Column(name = "floor_num")
public Integer getFloorNum() {
return floorNum;
}
public void setFloorNum(Integer floorNum) {
this.floorNum = floorNum;
}
@ManyToOne(optional = false)
@JoinColumn(name = "building")
public Building getBuilding() {
return building;
}
public void setBuilding(Building building) {
this.building = building;
}
}
テストされたFloorDao:
package com.noyaxe.myapp.repository;
import com.noyaxe.myapp.entity.Floor;
import org.springframework.data.jpa.repository.JpaSpecificationExecutor;
import org.springframework.data.repository.PagingAndSortingRepository;
import java.util.List;
public interface FloorDao extends JpaSpecificationExecutor<Floor>, PagingAndSortingRepository<Floor, Long> {
public Floor findByBuildingNameAndFloorNum(String building, Integer floorNum);
public List<Floor> findByBuildingName(String building);
}
テスト例も簡単です.
package com.noyaxe.myapp.repository;
import com.github.springtestdbunit.annotation.DatabaseSetup;
import com.noyaxe.myapp.entity.Floor;
import org.junit.Test;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;
import org.springframework.test.context.support.DependencyInjectionTestExecutionListener;
import org.springframework.test.context.support.DirtiesContextTestExecutionListener;
import java.util.List;
import static junit.framework.Assert.assertNull;
import static org.junit.Assert.assertEquals;
import static org.junit.Assert.assertNotNull;
@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration("classpath:applicationContext-test.xml")
@TestExecutionListeners({
DependencyInjectionTestExecutionListener.class,
DirtiesContextTestExecutionListener.class,
TransactionDbUnitTestExecutionListener.class})
public class FloorDaoTest {
@Autowired
private FloorDao floorDao;
@Test
@DatabaseSetup("FloorDao-findbByBuidlingName-dataset.xml")
public void testFindByBuildingName(){ List<Floor> singleFloorList = floorDao.findByBuildingName("SOHO"); assertEquals(1, singleFloorList.size()); List<Floor> twoFloorList = floorDao.findByBuildingName("New Gate Plaza"); assertEquals(2, twoFloorList.size()); List<Floor> emptyFloorList = floorDao.findByBuildingName("Test"); assertEquals(0, emptyFloorList.size()); } @Test
@DatabaseSetup("FloorDao-findbByBuidlingNameAndFloorNum-dataset.xml")
public void testFindByBuildingNameAndFloorNum(){ Floor floor = floorDao.findByBuildingNameAndFloorNum("SOHO", 2); assertNotNull(floor); Floor empty = floorDao.findByBuildingNameAndFloorNum("New Gate Plaza", 7); assertNull(empty); empty = floorDao.findByBuildingNameAndFloorNum("No Building", 7); assertNull(empty); }}
コードにより,DatabaseSetupによるテストデータの導入が完了していることが明らかになった.ここでは、各テストメソッドの前に異なるファイルを導入し、すべてのメソッドが1つのファイルで含めることができる場合は、クラスの前にDatabaseSetupを使用してデータファイルを導入することもできます.
これで、完全なデータ層テストの例が提示され、実行できます.しかし、実際の過程はそれほど順調ではなく、次の文章では出会った問題をまとめなければならない.