|
@@ -4,21 +4,31 @@ import cn.hutool.core.bean.BeanUtil;
|
|
|
import cn.hutool.core.collection.CollUtil;
|
|
|
import cn.hutool.core.lang.Assert;
|
|
|
import cn.hutool.core.util.RandomUtil;
|
|
|
+import cn.hutool.extra.pinyin.PinyinUtil;
|
|
|
import co.elastic.clients.elasticsearch.ElasticsearchClient;
|
|
|
+import co.elastic.clients.elasticsearch._types.FieldValue;
|
|
|
import co.elastic.clients.elasticsearch._types.SortOrder;
|
|
|
+import co.elastic.clients.elasticsearch._types.aggregations.LongTermsAggregate;
|
|
|
+import co.elastic.clients.elasticsearch._types.aggregations.LongTermsBucket;
|
|
|
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.CompletionSuggestOption;
|
|
|
import co.elastic.clients.elasticsearch.core.search.Hit;
|
|
|
import co.elastic.clients.elasticsearch.core.search.HitsMetadata;
|
|
|
+import co.elastic.clients.elasticsearch.core.search.Suggestion;
|
|
|
+import com.alibaba.fastjson.JSON;
|
|
|
import com.atguigu.tingshu.album.AlbumFeignClient;
|
|
|
import com.atguigu.tingshu.model.album.AlbumAttributeValue;
|
|
|
import com.atguigu.tingshu.model.album.AlbumInfo;
|
|
|
+import com.atguigu.tingshu.model.album.BaseCategory3;
|
|
|
import com.atguigu.tingshu.model.album.BaseCategoryView;
|
|
|
import com.atguigu.tingshu.model.search.AlbumInfoIndex;
|
|
|
import com.atguigu.tingshu.model.search.AttributeValueIndex;
|
|
|
+import com.atguigu.tingshu.model.search.SuggestIndex;
|
|
|
import com.atguigu.tingshu.query.search.AlbumIndexQuery;
|
|
|
import com.atguigu.tingshu.search.repository.AlbumInfoIndexRepository;
|
|
|
+import com.atguigu.tingshu.search.repository.SuggestIndexRepository;
|
|
|
import com.atguigu.tingshu.search.service.SearchService;
|
|
|
import com.atguigu.tingshu.user.client.UserFeignClient;
|
|
|
import com.atguigu.tingshu.vo.search.AlbumInfoIndexVo;
|
|
@@ -27,11 +37,12 @@ import com.atguigu.tingshu.vo.user.UserInfoVo;
|
|
|
import lombok.extern.slf4j.Slf4j;
|
|
|
import org.apache.commons.lang3.StringUtils;
|
|
|
import org.springframework.beans.factory.annotation.Autowired;
|
|
|
+import org.springframework.data.elasticsearch.core.suggest.Completion;
|
|
|
import org.springframework.stereotype.Service;
|
|
|
|
|
|
+import java.io.IOException;
|
|
|
import java.math.BigDecimal;
|
|
|
-import java.util.List;
|
|
|
-import java.util.Map;
|
|
|
+import java.util.*;
|
|
|
import java.util.concurrent.CompletableFuture;
|
|
|
import java.util.concurrent.Executor;
|
|
|
import java.util.stream.Collectors;
|
|
@@ -58,6 +69,9 @@ public class SearchServiceImpl implements SearchService {
|
|
|
@Autowired
|
|
|
private ElasticsearchClient elasticsearchClient;
|
|
|
|
|
|
+ @Autowired
|
|
|
+ private SuggestIndexRepository suggestIndexRepository;
|
|
|
+
|
|
|
//@Autowired
|
|
|
//@Qualifier("threadPoolTaskExecutor")
|
|
|
//private Executor executor;
|
|
@@ -96,7 +110,7 @@ public class SearchServiceImpl implements SearchService {
|
|
|
return albumInfo;
|
|
|
}, threadPoolTaskExecutor);
|
|
|
|
|
|
- //3.封装专辑索引库分类信息
|
|
|
+ ////3.封装专辑索引库分类信息
|
|
|
CompletableFuture<Void> categoryCompletableFuture = albumInfoCompletableFuture.thenAcceptAsync(albumInfo -> {
|
|
|
log.info("子线程:{},执行查询分类", Thread.currentThread().getName());
|
|
|
BaseCategoryView categoryView = albumFeignClient.getCategoryView(albumInfo.getCategory3Id()).getData();
|
|
@@ -104,16 +118,16 @@ public class SearchServiceImpl implements SearchService {
|
|
|
albumInfoIndex.setCategory1Id(categoryView.getCategory1Id());
|
|
|
albumInfoIndex.setCategory2Id(categoryView.getCategory2Id());
|
|
|
}, threadPoolTaskExecutor);
|
|
|
-
|
|
|
- //4.封装专辑索引库主播信息
|
|
|
+ //
|
|
|
+ ////4.封装专辑索引库主播信息
|
|
|
CompletableFuture<Void> userCompletableFuture = albumInfoCompletableFuture.thenAcceptAsync(albumInfo -> {
|
|
|
log.info("子线程:{},执行查询用户", Thread.currentThread().getName());
|
|
|
UserInfoVo userInfoVo = userFeignClient.getUserInfoVo(albumInfo.getUserId()).getData();
|
|
|
Assert.notNull(userInfoVo, "用户:{}不存在", albumInfo.getUserId());
|
|
|
albumInfoIndex.setAnnouncerName(userInfoVo.getNickname());
|
|
|
}, threadPoolTaskExecutor);
|
|
|
-
|
|
|
- //5.封装专辑索引库统计信息 暂时采用随机生成方式 TODO 上线后需要实际查询统计数值
|
|
|
+ //
|
|
|
+ ////5.封装专辑索引库统计信息 暂时采用随机生成方式 TODO 上线后需要实际查询统计数值
|
|
|
CompletableFuture<Void> statCompletableFuture = CompletableFuture.runAsync(() -> {
|
|
|
//5.1 随机产生四个数值对应 播放数,订阅数,购买数,评论数
|
|
|
int num1 = RandomUtil.randomInt(1000, 5000);
|
|
@@ -144,6 +158,9 @@ public class SearchServiceImpl implements SearchService {
|
|
|
|
|
|
//7.将专辑索引库信息保存到索引库
|
|
|
albumInfoIndexRepository.save(albumInfoIndex);
|
|
|
+
|
|
|
+ //8.保存专辑名称到提词索引库
|
|
|
+ this.saveSuggest(albumInfoIndex);
|
|
|
}
|
|
|
|
|
|
/**
|
|
@@ -154,6 +171,7 @@ public class SearchServiceImpl implements SearchService {
|
|
|
@Override
|
|
|
public void lowerAlbum(Long albumId) {
|
|
|
albumInfoIndexRepository.deleteById(albumId);
|
|
|
+ suggestIndexRepository.deleteById(albumId);
|
|
|
}
|
|
|
|
|
|
/**
|
|
@@ -180,6 +198,7 @@ public class SearchServiceImpl implements SearchService {
|
|
|
}
|
|
|
|
|
|
private static final String INDEX_NAME = "albuminfo";
|
|
|
+ private static final String SUGGEST_INDEX_NAME = "suggestinfo";
|
|
|
|
|
|
/**
|
|
|
* 基于检索条件封装完整检索请求对象
|
|
@@ -323,4 +342,169 @@ public class SearchServiceImpl implements SearchService {
|
|
|
//4.响应vo
|
|
|
return vo;
|
|
|
}
|
|
|
+
|
|
|
+ /**
|
|
|
+ * 得到置顶三级分类热门专辑TOP6列表
|
|
|
+ *
|
|
|
+ * @param category1Id
|
|
|
+ * @return
|
|
|
+ */
|
|
|
+ @Override
|
|
|
+ public List<Map<String, Object>> getBaseCategoryListByCategory1Id(Long category1Id) {
|
|
|
+ try {
|
|
|
+ //1.根据1级分类ID获取置顶的7个三级分类列表,获取7个三级分类ID
|
|
|
+ //1.1 远程调用专辑服务获取置顶三级分类列表
|
|
|
+ List<BaseCategory3> baseCategory3List = albumFeignClient.findTopBaseCategory3(category1Id).getData();
|
|
|
+ Assert.notNull(baseCategory3List, "根据一级分类ID获取置顶的7个三级分类列表失败");
|
|
|
+ //1.2 将三级分类对象处理得到ES检索需要的FiledValue类型
|
|
|
+ List<FieldValue> fieldValueList = baseCategory3List.stream()
|
|
|
+ .map(baseCategory3 -> FieldValue.of(baseCategory3.getId()))
|
|
|
+ .collect(Collectors.toList());
|
|
|
+
|
|
|
+ //1.3 为了封装结果中得到三级分类对象,将三级分类列表转为Map<三级分类ID,三级分类对象>
|
|
|
+ Map<Long, BaseCategory3> category3Map =
|
|
|
+ baseCategory3List.stream().collect(Collectors.toMap(c3 -> c3.getId(), c3 -> c3));
|
|
|
+
|
|
|
+ //2.采用多关键字精确查询+聚合执行检索ES
|
|
|
+ SearchResponse<AlbumInfoIndex> searchResponse = elasticsearchClient.search(
|
|
|
+ s -> s.index(INDEX_NAME)
|
|
|
+ .size(0)
|
|
|
+ .query(q -> q.terms(t -> t.field("category3Id").terms(t1 -> t1.value(fieldValueList))))
|
|
|
+ .aggregations(
|
|
|
+ "category_agg", a -> a.terms(t -> t.field("category3Id").size(7))
|
|
|
+ .aggregations("top6", a1 -> a1.topHits(t -> t.size(6).sort(s1 -> s1.field(f -> f.field("hotScore").order(SortOrder.Desc))).source(s1 -> s1.filter(f -> f.excludes("attributeValueIndexList")))))
|
|
|
+ ),
|
|
|
+ AlbumInfoIndex.class
|
|
|
+ );
|
|
|
+ //3.解析响应结果封装置顶分类热门专辑列表
|
|
|
+ //3.1 获取分类聚合结果对象
|
|
|
+ LongTermsAggregate categoryAgg = searchResponse.aggregations().get("category_agg").lterms();
|
|
|
+ //3.2 获取分类聚合结果对象中的桶集合
|
|
|
+ List<LongTermsBucket> cagegoryTermsBucketList = categoryAgg.buckets().array();
|
|
|
+ if (CollUtil.isNotEmpty(cagegoryTermsBucketList)) {
|
|
|
+ //3.3 遍历桶集合,每遍历一个桶,封装一个置顶分类热门专辑Map对象
|
|
|
+ List<Map<String, Object>> list = cagegoryTermsBucketList
|
|
|
+ .stream()
|
|
|
+ .map(categoryBucket -> {
|
|
|
+ //3.4 封装置顶分类热门专辑Map
|
|
|
+ Map<String, Object> map = new HashMap<>();
|
|
|
+ //3.4.1 处理分类信息:获取桶中的三级分类ID,得到三级分类对象
|
|
|
+ map.put("baseCategory3", category3Map.get(categoryBucket.key()));
|
|
|
+ //3.4.2 处理热门专辑信息:获取桶中的热门专辑Top6聚合结果,得到热门专辑对象集合
|
|
|
+ List<AlbumInfoIndex> top6AlbumList = categoryBucket.aggregations().get("top6").topHits().hits().hits()
|
|
|
+ .stream()
|
|
|
+ .map(hit -> {
|
|
|
+ String albumInfoIndexStr = hit.source().toString();
|
|
|
+ AlbumInfoIndex albumInfoIndex = JSON.parseObject(albumInfoIndexStr, AlbumInfoIndex.class);
|
|
|
+ return albumInfoIndex;
|
|
|
+ }).collect(Collectors.toList());
|
|
|
+ map.put("list", top6AlbumList);
|
|
|
+ return map;
|
|
|
+ }).collect(Collectors.toList());
|
|
|
+ return list;
|
|
|
+ }
|
|
|
+ return null;
|
|
|
+ } catch (IOException e) {
|
|
|
+ log.error("[搜索服务]首页置顶分类热门专辑检索异常:{}", e);
|
|
|
+ throw new RuntimeException(e);
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * 将专辑名称存入提词索引库
|
|
|
+ *
|
|
|
+ * @param albumInfoIndex
|
|
|
+ */
|
|
|
+ @Override
|
|
|
+ public void saveSuggest(AlbumInfoIndex albumInfoIndex) {
|
|
|
+ SuggestIndex suggestIndex = new SuggestIndex();
|
|
|
+ suggestIndex.setId(albumInfoIndex.getId());
|
|
|
+ String albumTitle = albumInfoIndex.getAlbumTitle();
|
|
|
+ suggestIndex.setTitle(albumTitle);
|
|
|
+ //存入汉字
|
|
|
+ suggestIndex.setKeyword(new Completion(new String[]{albumTitle}));
|
|
|
+ //将汉字转为汉语拼音
|
|
|
+ String pinyin = PinyinUtil.getPinyin(albumTitle, "");
|
|
|
+ suggestIndex.setKeywordPinyin(new Completion(new String[]{pinyin}));
|
|
|
+ //获取拼音首字母
|
|
|
+ String firstLetter = PinyinUtil.getFirstLetter(albumTitle, "");
|
|
|
+ suggestIndex.setKeywordSequence(new Completion(new String[]{firstLetter}));
|
|
|
+ suggestIndexRepository.save(suggestIndex);
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * 根据用户已录入字符返回提示词列表,自动补全效果
|
|
|
+ *
|
|
|
+ * @param keyword
|
|
|
+ * @return
|
|
|
+ */
|
|
|
+ @Override
|
|
|
+ public List<String> completeSuggest(String keyword) {
|
|
|
+ try {
|
|
|
+ //1.采用自动补全API尝试分别查询:汉字、拼音、首字母自动补全字段
|
|
|
+ SearchResponse<SuggestIndex> suggestIndexSearchResponse = elasticsearchClient.search(
|
|
|
+ s -> s.index(SUGGEST_INDEX_NAME)
|
|
|
+ .suggest(s1 -> s1.suggesters(
|
|
|
+ "letter-suggest", s2 -> s2.prefix(keyword).completion(c -> c.field("keywordSequence").size(10).skipDuplicates(true))
|
|
|
+ ).suggesters("pinyin-suggest", s2 -> s2.prefix(keyword).completion(c -> c.field("keywordPinyin").size(10).skipDuplicates(true).fuzzy(f -> f.fuzziness("2").minLength(4))))
|
|
|
+ .suggesters("keyword-suggest", s2 -> s2.prefix(keyword).completion(c -> c.field("keyword").size(10).skipDuplicates(true).fuzzy(f -> f.fuzziness("2").minLength(4))))
|
|
|
+ )
|
|
|
+ ,
|
|
|
+ SuggestIndex.class
|
|
|
+ );
|
|
|
+ //2.解析ES自动补全结果,将结果放入HashSet去重
|
|
|
+ Set<String> hashSet = new HashSet<>();
|
|
|
+ hashSet.addAll(this.parseSuggestResult(suggestIndexSearchResponse, "letter-suggest"));
|
|
|
+ hashSet.addAll(this.parseSuggestResult(suggestIndexSearchResponse, "pinyin-suggest"));
|
|
|
+ hashSet.addAll(this.parseSuggestResult(suggestIndexSearchResponse, "keyword-suggest"));
|
|
|
+
|
|
|
+ //3.如果返回自动补全结果集合长度 小于10个,采用全文检索:检索专辑标题即可,补全到10个
|
|
|
+ if (hashSet.size() < 10) {
|
|
|
+ SearchResponse<AlbumInfoIndex> searchResponse = elasticsearchClient.search(
|
|
|
+ s -> s.index(INDEX_NAME)
|
|
|
+ .size(10)
|
|
|
+ .query(q -> q.match(m -> m.field("albumTitle").query(keyword))
|
|
|
+ ), AlbumInfoIndex.class);
|
|
|
+ List<Hit<AlbumInfoIndex>> hitList = searchResponse.hits().hits();
|
|
|
+ if (CollUtil.isNotEmpty(hitList)) {
|
|
|
+ for (Hit<AlbumInfoIndex> hit : hitList) {
|
|
|
+ AlbumInfoIndex source = hit.source();
|
|
|
+ hashSet.add(source.getAlbumTitle());
|
|
|
+ if (hashSet.size() >= 10) {
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ //4.如果返回自动补全结果集合长度 大于等于10个,则截取10个返回
|
|
|
+ if (hashSet.size() >= 10) {
|
|
|
+ return new ArrayList<>(hashSet).subList(0, 10);
|
|
|
+ }
|
|
|
+ return new ArrayList<>(hashSet);
|
|
|
+ } catch (IOException e) {
|
|
|
+ log.error("[搜索服务]自动补全检索异常:{}", e);
|
|
|
+ throw new RuntimeException(e);
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ /**
|
|
|
+ * 解析建议词结果
|
|
|
+ *
|
|
|
+ * @param suggestIndexSearchResponse ES响应对象
|
|
|
+ * @param suggestName 自定义名称
|
|
|
+ * @return
|
|
|
+ */
|
|
|
+ @Override
|
|
|
+ public Collection<String> parseSuggestResult(SearchResponse<SuggestIndex> suggestIndexSearchResponse, String suggestName) {
|
|
|
+ List<String> list = new ArrayList<>();
|
|
|
+ List<Suggestion<SuggestIndex>> suggestions = suggestIndexSearchResponse.suggest().get(suggestName);
|
|
|
+ for (Suggestion<SuggestIndex> suggestion : suggestions) {
|
|
|
+ for (CompletionSuggestOption<SuggestIndex> option : suggestion.completion().options()) {
|
|
|
+ SuggestIndex suggestIndex = option.source();
|
|
|
+ list.add(suggestIndex.getTitle());
|
|
|
+ }
|
|
|
+ }
|
|
|
+ return list;
|
|
|
+ }
|
|
|
}
|