|
@@ -4,22 +4,34 @@ import cn.hutool.core.bean.BeanUtil;
|
|
import cn.hutool.core.collection.CollUtil;
|
|
import cn.hutool.core.collection.CollUtil;
|
|
import cn.hutool.core.lang.Assert;
|
|
import cn.hutool.core.lang.Assert;
|
|
import cn.hutool.core.util.RandomUtil;
|
|
import cn.hutool.core.util.RandomUtil;
|
|
|
|
+import co.elastic.clients.elasticsearch.ElasticsearchClient;
|
|
|
|
+import co.elastic.clients.elasticsearch._types.SortOrder;
|
|
|
|
+import co.elastic.clients.elasticsearch._types.query_dsl.BoolQuery;
|
|
|
|
+import co.elastic.clients.elasticsearch.core.SearchRequest;
|
|
|
|
+import co.elastic.clients.elasticsearch.core.SearchResponse;
|
|
|
|
+import co.elastic.clients.elasticsearch.core.search.Hit;
|
|
|
|
+import co.elastic.clients.elasticsearch.core.search.HitsMetadata;
|
|
import com.atguigu.tingshu.album.AlbumFeignClient;
|
|
import com.atguigu.tingshu.album.AlbumFeignClient;
|
|
import com.atguigu.tingshu.model.album.AlbumAttributeValue;
|
|
import com.atguigu.tingshu.model.album.AlbumAttributeValue;
|
|
import com.atguigu.tingshu.model.album.AlbumInfo;
|
|
import com.atguigu.tingshu.model.album.AlbumInfo;
|
|
import com.atguigu.tingshu.model.album.BaseCategoryView;
|
|
import com.atguigu.tingshu.model.album.BaseCategoryView;
|
|
import com.atguigu.tingshu.model.search.AlbumInfoIndex;
|
|
import com.atguigu.tingshu.model.search.AlbumInfoIndex;
|
|
import com.atguigu.tingshu.model.search.AttributeValueIndex;
|
|
import com.atguigu.tingshu.model.search.AttributeValueIndex;
|
|
|
|
+import com.atguigu.tingshu.query.search.AlbumIndexQuery;
|
|
import com.atguigu.tingshu.search.repository.AlbumInfoIndexRepository;
|
|
import com.atguigu.tingshu.search.repository.AlbumInfoIndexRepository;
|
|
import com.atguigu.tingshu.search.service.SearchService;
|
|
import com.atguigu.tingshu.search.service.SearchService;
|
|
import com.atguigu.tingshu.user.client.UserFeignClient;
|
|
import com.atguigu.tingshu.user.client.UserFeignClient;
|
|
|
|
+import com.atguigu.tingshu.vo.search.AlbumInfoIndexVo;
|
|
|
|
+import com.atguigu.tingshu.vo.search.AlbumSearchResponseVo;
|
|
import com.atguigu.tingshu.vo.user.UserInfoVo;
|
|
import com.atguigu.tingshu.vo.user.UserInfoVo;
|
|
import lombok.extern.slf4j.Slf4j;
|
|
import lombok.extern.slf4j.Slf4j;
|
|
|
|
+import org.apache.commons.lang3.StringUtils;
|
|
import org.springframework.beans.factory.annotation.Autowired;
|
|
import org.springframework.beans.factory.annotation.Autowired;
|
|
import org.springframework.stereotype.Service;
|
|
import org.springframework.stereotype.Service;
|
|
|
|
|
|
import java.math.BigDecimal;
|
|
import java.math.BigDecimal;
|
|
import java.util.List;
|
|
import java.util.List;
|
|
|
|
+import java.util.Map;
|
|
import java.util.concurrent.CompletableFuture;
|
|
import java.util.concurrent.CompletableFuture;
|
|
import java.util.concurrent.Executor;
|
|
import java.util.concurrent.Executor;
|
|
import java.util.stream.Collectors;
|
|
import java.util.stream.Collectors;
|
|
@@ -43,6 +55,9 @@ public class SearchServiceImpl implements SearchService {
|
|
@Autowired //先按类型注入 再按名称注入
|
|
@Autowired //先按类型注入 再按名称注入
|
|
private Executor threadPoolTaskExecutor;
|
|
private Executor threadPoolTaskExecutor;
|
|
|
|
|
|
|
|
+ @Autowired
|
|
|
|
+ private ElasticsearchClient elasticsearchClient;
|
|
|
|
+
|
|
//@Autowired
|
|
//@Autowired
|
|
//@Qualifier("threadPoolTaskExecutor")
|
|
//@Qualifier("threadPoolTaskExecutor")
|
|
//private Executor executor;
|
|
//private Executor executor;
|
|
@@ -57,6 +72,7 @@ public class SearchServiceImpl implements SearchService {
|
|
*/
|
|
*/
|
|
@Override
|
|
@Override
|
|
public void upperAlbum(Long albumId) {
|
|
public void upperAlbum(Long albumId) {
|
|
|
|
+ log.info("主线程:{},专辑索引库对象", Thread.currentThread().getName());
|
|
//1.创建索引库对象
|
|
//1.创建索引库对象
|
|
AlbumInfoIndex albumInfoIndex = new AlbumInfoIndex();
|
|
AlbumInfoIndex albumInfoIndex = new AlbumInfoIndex();
|
|
|
|
|
|
@@ -64,6 +80,7 @@ public class SearchServiceImpl implements SearchService {
|
|
CompletableFuture<AlbumInfo> albumInfoCompletableFuture = CompletableFuture.supplyAsync(() -> {
|
|
CompletableFuture<AlbumInfo> albumInfoCompletableFuture = CompletableFuture.supplyAsync(() -> {
|
|
//2.1 远程调用专辑服务根据专辑ID查询专辑信息包含标签列表
|
|
//2.1 远程调用专辑服务根据专辑ID查询专辑信息包含标签列表
|
|
AlbumInfo albumInfo = albumFeignClient.getAlbumInfo(albumId).getData();
|
|
AlbumInfo albumInfo = albumFeignClient.getAlbumInfo(albumId).getData();
|
|
|
|
+ log.info("子线程:{},执行查询专辑", Thread.currentThread().getName());
|
|
Assert.notNull(albumInfo, "专辑:{}不存在", albumId);
|
|
Assert.notNull(albumInfo, "专辑:{}不存在", albumId);
|
|
//2.2 为索引库对象中属性赋值
|
|
//2.2 为索引库对象中属性赋值
|
|
BeanUtil.copyProperties(albumInfo, albumInfoIndex);
|
|
BeanUtil.copyProperties(albumInfo, albumInfoIndex);
|
|
@@ -81,6 +98,7 @@ public class SearchServiceImpl implements SearchService {
|
|
|
|
|
|
//3.封装专辑索引库分类信息
|
|
//3.封装专辑索引库分类信息
|
|
CompletableFuture<Void> categoryCompletableFuture = albumInfoCompletableFuture.thenAcceptAsync(albumInfo -> {
|
|
CompletableFuture<Void> categoryCompletableFuture = albumInfoCompletableFuture.thenAcceptAsync(albumInfo -> {
|
|
|
|
+ log.info("子线程:{},执行查询分类", Thread.currentThread().getName());
|
|
BaseCategoryView categoryView = albumFeignClient.getCategoryView(albumInfo.getCategory3Id()).getData();
|
|
BaseCategoryView categoryView = albumFeignClient.getCategoryView(albumInfo.getCategory3Id()).getData();
|
|
Assert.notNull(categoryView, "专辑分类:{}不存在", albumInfo.getCategory3Id());
|
|
Assert.notNull(categoryView, "专辑分类:{}不存在", albumInfo.getCategory3Id());
|
|
albumInfoIndex.setCategory1Id(categoryView.getCategory1Id());
|
|
albumInfoIndex.setCategory1Id(categoryView.getCategory1Id());
|
|
@@ -89,12 +107,13 @@ public class SearchServiceImpl implements SearchService {
|
|
|
|
|
|
//4.封装专辑索引库主播信息
|
|
//4.封装专辑索引库主播信息
|
|
CompletableFuture<Void> userCompletableFuture = albumInfoCompletableFuture.thenAcceptAsync(albumInfo -> {
|
|
CompletableFuture<Void> userCompletableFuture = albumInfoCompletableFuture.thenAcceptAsync(albumInfo -> {
|
|
|
|
+ log.info("子线程:{},执行查询用户", Thread.currentThread().getName());
|
|
UserInfoVo userInfoVo = userFeignClient.getUserInfoVo(albumInfo.getUserId()).getData();
|
|
UserInfoVo userInfoVo = userFeignClient.getUserInfoVo(albumInfo.getUserId()).getData();
|
|
Assert.notNull(userInfoVo, "用户:{}不存在", albumInfo.getUserId());
|
|
Assert.notNull(userInfoVo, "用户:{}不存在", albumInfo.getUserId());
|
|
albumInfoIndex.setAnnouncerName(userInfoVo.getNickname());
|
|
albumInfoIndex.setAnnouncerName(userInfoVo.getNickname());
|
|
}, threadPoolTaskExecutor);
|
|
}, threadPoolTaskExecutor);
|
|
|
|
|
|
- //5.封装专辑索引库统计信息 暂时采用随机生成方式
|
|
|
|
|
|
+ //5.封装专辑索引库统计信息 暂时采用随机生成方式 TODO 上线后需要实际查询统计数值
|
|
CompletableFuture<Void> statCompletableFuture = CompletableFuture.runAsync(() -> {
|
|
CompletableFuture<Void> statCompletableFuture = CompletableFuture.runAsync(() -> {
|
|
//5.1 随机产生四个数值对应 播放数,订阅数,购买数,评论数
|
|
//5.1 随机产生四个数值对应 播放数,订阅数,购买数,评论数
|
|
int num1 = RandomUtil.randomInt(1000, 5000);
|
|
int num1 = RandomUtil.randomInt(1000, 5000);
|
|
@@ -129,10 +148,179 @@ public class SearchServiceImpl implements SearchService {
|
|
|
|
|
|
/**
|
|
/**
|
|
* 下架索引库专辑
|
|
* 下架索引库专辑
|
|
|
|
+ *
|
|
* @param albumId
|
|
* @param albumId
|
|
*/
|
|
*/
|
|
@Override
|
|
@Override
|
|
public void lowerAlbum(Long albumId) {
|
|
public void lowerAlbum(Long albumId) {
|
|
albumInfoIndexRepository.deleteById(albumId);
|
|
albumInfoIndexRepository.deleteById(albumId);
|
|
}
|
|
}
|
|
|
|
+
|
|
|
|
+ /**
|
|
|
|
+ * 多条件条件分页检索
|
|
|
|
+ *
|
|
|
|
+ * @param albumIndexQuery
|
|
|
|
+ * @return
|
|
|
|
+ */
|
|
|
|
+ @Override
|
|
|
|
+ public AlbumSearchResponseVo search(AlbumIndexQuery albumIndexQuery) {
|
|
|
|
+ try {
|
|
|
|
+ //一、构建检索请求对象
|
|
|
|
+ SearchRequest searchRequest = this.buildDSL(albumIndexQuery);
|
|
|
|
+ System.err.println("本次检索DSL:");
|
|
|
|
+ System.err.println(searchRequest.toString());
|
|
|
|
+ //二、调用ES客户端执行检索
|
|
|
|
+ SearchResponse<AlbumInfoIndex> searchResponse = elasticsearchClient.search(searchRequest, AlbumInfoIndex.class);
|
|
|
|
+ //三、解析封装ES响应结果
|
|
|
|
+ return this.parseResult(searchResponse, albumIndexQuery);
|
|
|
|
+ } catch (Exception e) {
|
|
|
|
+ log.error("[检索服务]专辑检索异常:{}", e);
|
|
|
|
+ throw new RuntimeException(e);
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ private static final String INDEX_NAME = "albuminfo";
|
|
|
|
+
|
|
|
|
+ /**
|
|
|
|
+ * 基于检索条件封装完整检索请求对象
|
|
|
|
+ *
|
|
|
|
+ * @param albumIndexQuery
|
|
|
|
+ * @return
|
|
|
|
+ */
|
|
|
|
+ @Override
|
|
|
|
+ public SearchRequest buildDSL(AlbumIndexQuery albumIndexQuery) {
|
|
|
|
+ //1.创建检索请求 构建器对象->封装请求路径索引名称以及完整请求体参数
|
|
|
|
+ SearchRequest.Builder builder = new SearchRequest.Builder();
|
|
|
|
+ builder.index(INDEX_NAME);
|
|
|
|
+ //2.封装请求体参数 query。通过布尔查询封装 三大条件:1.关键字 2.分类 3.标签
|
|
|
|
+ //2.1 创建封装三大条件bool查询对象
|
|
|
|
+ BoolQuery.Builder allConditionBoolQueryBuilder = new BoolQuery.Builder();
|
|
|
|
+ //2.2 设置关键字条件查询
|
|
|
|
+ String keyword = albumIndexQuery.getKeyword();
|
|
|
|
+ if (StringUtils.isNotBlank(keyword)) {
|
|
|
|
+ allConditionBoolQueryBuilder.must(m -> m.bool(
|
|
|
|
+ b -> b.should(s -> s.match(m1 -> m1.field("albumTitle").query(keyword)))
|
|
|
|
+ .should(s -> s.match(m1 -> m1.field("albumIntro").query(keyword)))
|
|
|
|
+ .should(s -> s.term(t -> t.field("announcerName").value(keyword)))
|
|
|
|
+ ));
|
|
|
|
+ }
|
|
|
|
+ //2.3 设置分类条件过滤
|
|
|
|
+ if (albumIndexQuery.getCategory1Id() != null) {
|
|
|
|
+ allConditionBoolQueryBuilder.filter(f -> f.term(t -> t.field("category1Id").value(albumIndexQuery.getCategory1Id())));
|
|
|
|
+ }
|
|
|
|
+ if (albumIndexQuery.getCategory2Id() != null) {
|
|
|
|
+ allConditionBoolQueryBuilder.filter(f -> f.term(t -> t.field("category2Id").value(albumIndexQuery.getCategory2Id())));
|
|
|
|
+ }
|
|
|
|
+ if (albumIndexQuery.getCategory3Id() != null) {
|
|
|
|
+ allConditionBoolQueryBuilder.filter(f -> f.term(t -> t.field("category3Id").value(albumIndexQuery.getCategory3Id())));
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ //2.4 设置标签条件过滤 可能有多组标签 每一组标签条件产生一个嵌套查询
|
|
|
|
+ List<String> attributeList = albumIndexQuery.getAttributeList();
|
|
|
|
+ if (CollUtil.isNotEmpty(attributeList)) {
|
|
|
|
+ //2.4.1 遍历标签列表 每遍历一次设置过滤Nested查询
|
|
|
|
+ for (String attribute : attributeList) {
|
|
|
|
+ //标签条件=标签ID:标签值ID
|
|
|
|
+ String[] split = attribute.split(":");
|
|
|
|
+ if (split != null && split.length == 2) {
|
|
|
|
+ allConditionBoolQueryBuilder.filter(f -> f.nested(
|
|
|
|
+ n -> n.path("attributeValueIndexList")
|
|
|
|
+ .query(q -> q.bool(
|
|
|
|
+ b -> b.must(m -> m.term(t -> t.field("attributeValueIndexList.attributeId").value(split[0])))
|
|
|
|
+ .must(m -> m.term(t -> t.field("attributeValueIndexList.valueId").value(split[1])))
|
|
|
|
+ ))
|
|
|
|
+ ));
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+ //2.5 将三大条件的布尔查询对象设置到"query"中
|
|
|
|
+ builder.query(allConditionBoolQueryBuilder.build()._toQuery());
|
|
|
|
+ //2.2 封装分页条件 from=起始文档offset size=页大小
|
|
|
|
+ Integer pageNo = albumIndexQuery.getPageNo();
|
|
|
|
+ Integer pageSize = albumIndexQuery.getPageSize();
|
|
|
|
+ Integer from = (pageNo - 1) * pageSize;
|
|
|
|
+ builder.from(from).size(pageSize);
|
|
|
|
+
|
|
|
|
+ //2.3 封装排序条件 sort
|
|
|
|
+ //2.3.1 获取排序条件 形式=>字段标识1/2/3:asc/desc 1:综合排序 2:发布时间 3:播放量
|
|
|
|
+ String order = albumIndexQuery.getOrder();
|
|
|
|
+ if (StringUtils.isNotBlank(order)) {
|
|
|
|
+ //2.3.2 如果有需要自定义排序,才设置请求参数:sort
|
|
|
|
+ String[] split = order.split(":");
|
|
|
|
+ if (split != null && split.length == 2) {
|
|
|
|
+ //2.3.3 得到排序字段 跟 排序方式
|
|
|
|
+ String sortField = "";
|
|
|
|
+ switch (split[0]) {
|
|
|
|
+ case "1":
|
|
|
|
+ sortField = "hotScore";
|
|
|
|
+ break;
|
|
|
|
+ case "2":
|
|
|
|
+ sortField = "playStatNum";
|
|
|
|
+ break;
|
|
|
|
+ case "3":
|
|
|
|
+ sortField = "createTime";
|
|
|
|
+ break;
|
|
|
|
+ }
|
|
|
|
+ String finalSortField = sortField;
|
|
|
|
+ builder.sort(s -> s.field(f -> f.field(finalSortField).order(split[1].equals("desc") ? SortOrder.Desc : SortOrder.Asc)));
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+ //2.4 封装高亮条件 highlight
|
|
|
|
+ if (StringUtils.isNotBlank(keyword)) {
|
|
|
|
+ builder.highlight(h -> h.fields("albumTitle", f -> f.preTags("<font style='color:red'>").postTags("</font>")));
|
|
|
|
+ }
|
|
|
|
+ //2.5 设置文档响应字段
|
|
|
|
+ builder.source(s -> s.filter(f -> f.excludes("attributeValueIndexList", "hotScore", "commentStatNum", "buyStatNum", "subscribeStatNum", "announcerName")));
|
|
|
|
+ //3.基于构建器对象产生检索请求对象
|
|
|
|
+ return builder.build();
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ /**
|
|
|
|
+ * 解析ES响应结果封装自定义VO
|
|
|
|
+ *
|
|
|
|
+ * @param searchResponse
|
|
|
|
+ * @param albumIndexQuery
|
|
|
|
+ * @return
|
|
|
|
+ */
|
|
|
|
+ @Override
|
|
|
|
+ public AlbumSearchResponseVo parseResult(SearchResponse<AlbumInfoIndex> searchResponse, AlbumIndexQuery albumIndexQuery) {
|
|
|
|
+ //1.创建响应VO对象
|
|
|
|
+ AlbumSearchResponseVo vo = new AlbumSearchResponseVo();
|
|
|
|
+ //2.封装VO中分页信息
|
|
|
|
+ //2.1 从检索条件对象中获取页码、页大小
|
|
|
|
+ Integer pageNo = albumIndexQuery.getPageNo();
|
|
|
|
+ Integer pageSize = albumIndexQuery.getPageSize();
|
|
|
|
+ //2.2 从响应结果对象中获取总记录数,计算总页数
|
|
|
|
+ HitsMetadata<AlbumInfoIndex> hits = searchResponse.hits();
|
|
|
|
+ long total = hits.total().value();
|
|
|
|
+ long totalPage = total % pageSize == 0 ? total / pageSize : total / pageSize + 1;
|
|
|
|
+
|
|
|
|
+ vo.setPageNo(pageNo);
|
|
|
|
+ vo.setPageSize(pageSize);
|
|
|
|
+ vo.setTotal(total);
|
|
|
|
+ vo.setTotalPages(totalPage);
|
|
|
|
+ //3.封装VO中业务数据集合
|
|
|
|
+ //3.1 获取检索结果中hits对象中包含hits数组
|
|
|
|
+ List<Hit<AlbumInfoIndex>> hitList = hits.hits();
|
|
|
|
+
|
|
|
|
+ //3.2 遍历hits数组,得到文档对象中_source对应文档对象
|
|
|
|
+ if (CollUtil.isNotEmpty(hitList)) {
|
|
|
|
+ List<AlbumInfoIndexVo> albumInfoVoList = hitList
|
|
|
|
+ .stream()
|
|
|
|
+ .map(hit -> {
|
|
|
|
+ AlbumInfoIndex albumInfoIndex = hit.source();
|
|
|
|
+ //3.3 如果有高亮字段,则替换文档对象中对应字段的值
|
|
|
|
+ Map<String, List<String>> map = hit.highlight();
|
|
|
|
+ if (CollUtil.isNotEmpty(map)) {
|
|
|
|
+ List<String> list = map.get("albumTitle");
|
|
|
|
+ String highLigthText = list.get(0);
|
|
|
|
+ albumInfoIndex.setAlbumTitle(highLigthText);
|
|
|
|
+ }
|
|
|
|
+ return BeanUtil.copyProperties(albumInfoIndex, AlbumInfoIndexVo.class);
|
|
|
|
+ }).collect(Collectors.toList());
|
|
|
|
+ vo.setList(albumInfoVoList);
|
|
|
|
+ }
|
|
|
|
+ //4.响应vo
|
|
|
|
+ return vo;
|
|
|
|
+ }
|
|
}
|
|
}
|