通過長時間的編碼實現,咱們的主體模塊已經大體完成,由於以前咱們都是零散的對各個微服務自行測試,接下來,咱們須要將全部的服務模塊進行聯調測試,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中就能夠了,這裏就不作贅述。sql
用戶服務單元測試數據庫
/** * 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 ); } }
在這個測試用例中,咱們主要的複雜性是在組件各類查詢條件,這個就須要各位夥伴在理解業務的時候須要萬分上心。 </string></districtfeature.provinceandcity></string>