通过长时间的编码实现,咱们的主体模块已经大体完成,由于以前咱们都是零散的对各个微服务自行测试,接下来,咱们须要将全部的服务模块进行联调测试,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
);
}
}
复制代码
在这个测试用例中,咱们主要的复杂性是在组件各类查询条件,这个就须要各位伙伴在理解业务的时候须要万分上心。