通過長時間的編碼實現,咱們的主體模塊已經大體完成,由於以前咱們都是零散的對各個微服務自行測試,接下來,咱們須要將全部的服務模塊進行聯調測試,Let's do it.java
咱們在實現各個服務的過程當中,添加了很多的測試文件和測試數據,爲了避免影響咱們最終的展現效果,咱們先將以前的歷史數據清理掉。git
drop database advertisement;複製代碼
依然使用flyway 添加咱們的測試數據:github
INSERT INTO `ad_user` VALUES (10,'Isaac','B2E56F2420D73FEC125D2D51641C5713',1,'2019-08-14 20:29:01','2019-08-14 20:29:01');
INSERT INTO `ad_creative` VALUES (10,'第一個創意',1,1,720,1080,1024,0,1,10,'https://www.life-runner.com','2019-08-14 21:31:31','2019-08-14 21:31:31');
INSERT INTO `ad_plan` VALUES (10,10,'推廣計劃名稱',1,'2019-11-28 00:00:00','2019-11-20 00:00:00','2019-11-19 20:42:27','2019-08-14 20:57:12');
INSERT INTO `ad_unit` VALUES (10,10,'第一個推廣單元',1,1,10000000,'2019-11-20 11:43:26','2019-11-20 11:43:26'),(12,10,'第二個推廣單元',1,1,15000000,'2019-01-01 00:00:00','2019-01-01 00:00:00');
INSERT INTO `ad_unit_district` VALUES (10,10,'陝西省','西安市'),(11,10,'陝西省','西安市'),(12,10,'陝西省','西安市'),(14,10,'山西省','陽泉市');
INSERT INTO `ad_unit_hobby` VALUES (10,10,'登山'),(11,10,'讀書'),(12,10,'寫代碼');
INSERT INTO `ad_unit_keyword` VALUES (10,10,'汽車'),(11,10,'火車'),(12,10,'飛機');
INSERT INTO `relationship_creative_unit` VALUES (10,10,10);複製代碼
可參考 全量索引傳送門 ,或者下載源碼github傳送門 / gitee傳送門 ,運行mscx-ad-db
項目,而後執行 http://localhost:7002/ad-db/export/plan。web
一個合格的開發人員是絕對不能容忍本身的代碼存在傻~ bug 存在的,可是我的總會有犯錯的時候,那麼咱們要怎麼避免此類非業務發展致使的基礎問題呢,這時候,開發的UT就顯得很是Important了。spring
咱們來編寫投放系統的單元測試,以下圖:
單元測試模塊的目錄結構與咱們的正式項目結構保持一致,若是你須要給單元測試編寫特例化配置,把咱們的application.yml
配置文件copy到UT中就能夠了,這裏就不作贅述。數據庫
/**
* UserServiceTest for 用戶服務單元測試
*
* @author <a href="mailto:magicianisaac@gmail.com">Isaac.Zhang | 若初</a>
*/
@RunWith(SpringRunner.class)
@SpringBootTest(
classes = {SponsorApplication.class},
webEnvironment = SpringBootTest.WebEnvironment.NONE
)
public class UserServiceTest {
@Autowired
private IUserService userService;
@Test
// @Transactional
public void testCreateUser() throws AdException {
UserRequestVO userRequestVO = new UserRequestVO("Isaac Zhang");
UserResponseVO responseVO = userService.createUser(userRequestVO);
assert responseVO.getUserName() == "Isaac Zhang";
System.out.printf("建立用戶: %s", JSON.toJSONString(responseVO));
}
}複製代碼
你們能夠看到,在上述代碼中,咱們測試了建立用戶的service方法,特別注意2個點:
app
@Transactional
註解.ad_user
表中,若是咱們不想這個用戶存入表中,就須要加上@Transactional
註解,咱們的建立就不會commit,也就不會被插入到真實數據庫中。 @SpringBootTest
註解classes
代表測試啓動類是哪一個,webEnvironment = SpringBootTest.WebEnvironment.NONE
代表咱們當前的測試並不是一個web環境。 這裏就不針對每個service進行單元測試的編寫,可是你們必定要記住,在真實的企業開發環境中,大的開發團隊必定會對單元測試的代碼覆蓋率有一個要求,通常都不會低於60%
,我我的對本身的行代碼覆蓋率
是 > 80%.這樣才能真實的保證咱們的每個方法都儘可能都執行和驗證到。微服務
你們嘗試依次實現其他的單元測試吧。單元測試
咱們的檢索服務對外只提供一個服務,所以咱們只須要建立一個Test類就能夠了,let's code.測試
package com.sxzhongf.ad.search;
import com.sxzhongf.ad.AdSearchApplication;
import com.sxzhongf.ad.search.vo.SearchRequest;
import com.sxzhongf.ad.search.vo.SearchResponse;
import com.sxzhongf.ad.search.vo.feature.DistrictFeature;
import com.sxzhongf.ad.search.vo.feature.FeatureRelation;
import com.sxzhongf.ad.search.vo.feature.HobbyFeatrue;
import com.sxzhongf.ad.search.vo.feature.KeywordFeature;
import com.sxzhongf.ad.search.vo.media.AdSlot;
import com.sxzhongf.ad.search.vo.media.App;
import com.sxzhongf.ad.search.vo.media.Device;
import com.sxzhongf.ad.search.vo.media.Geo;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;
import org.springframework.test.context.junit4.SpringRunner;
import java.util.Arrays;
import java.util.Collections;
import java.util.List;
/**
* SearchTest for 搜索服務測試用例
*
* @author <a href="mailto:magicianisaac@gmail.com">Isaac.Zhang | 若初</a>
*/
@RunWith(SpringRunner.class)
@SpringBootTest(classes = AdSearchApplication.class, webEnvironment = SpringBootTest.WebEnvironment.NONE)
public class SearchTest {
@Autowired
private ISearch search;
@Test
public void testFetchAds() {
SearchRequest request = new SearchRequest().builder()
.mediaId("isaac-search-mediaId")
.requestInfo(new SearchRequest.RequestInfo(
"request id",
Arrays.asList(
new AdSlot().builder()
.adSlotCode("slot code")
.height(800)
.minCpm(1024)
.positionType(1)
.type(Arrays.asList(1))
.build()
),
buildSimpleApp(),
buildSimpleGeo(),
buildSimpleDevice()
))
.featureInfo(
buildSimpleFeatureInfo(
Arrays.asList("汽車", "火車", "飛機"),
Collections.singletonList(
new DistrictFeature.ProvinceAndCity(
"陝西省", "西安市"
)
),
Arrays.asList("登山", "寫代碼", "飛機"),
FeatureRelation.OR
)
)
.build();
SearchResponse response = search.fetchAds(request);
// assert response.adSlotRelationAds.get(0).contains("key");
System.out.println("開始查詢廣告拉:" + response);
}
/**
* 建立demo {@link App}
*/
private App buildSimpleApp() {
return new App().builder()
.activityName("simple App activityName")
.appCode("simple App appCode")
.appName("simple app name")
.packageName("simple app package name")
.build();
}
/**
* 建立demo {@link Geo}
*/
private Geo buildSimpleGeo() {
return new Geo().builder()
.longitude(Float.valueOf("100.2222222"))
.latitude(Float.valueOf("38.8888888"))
.city("xiaan")
.province("shaanxi")
.build();
}
/**
* 建立demo {@link Device}
*/
private Device buildSimpleDevice() {
return new Device().builder()
.deviceCode("simple device code")
.deviceMacAddr("simple mac addr")
.displaySize("simple display size")
.ip("127.0.0.1")
.model("simple model")
.screenSize("simple screen size")
.serialName("simple serial name")
.build();
}
private SearchRequest.FeatureInfo buildSimpleFeatureInfo(
List<String> keywords,
List<DistrictFeature.ProvinceAndCity> provinceAndCities,
List<String> hobbys,
FeatureRelation featureRelation
) {
return new SearchRequest.FeatureInfo(
new KeywordFeature(keywords),
new DistrictFeature(provinceAndCities),
new HobbyFeatrue(hobbys),
featureRelation
);
}
}
複製代碼
在這個測試用例中,咱們主要的複雜性是在組件各類查詢條件,這個就須要各位夥伴在理解業務的時候須要萬分上心。