Browse Source

Merge branch 'dev-zenas-20241018' of wangfan/adweb3-server into master

zhangqiang 5 months ago
parent
commit
76def60abe

+ 2 - 2
jeecg-module-system/jeecg-system-biz/src/main/java/org/jeecg/modules/adweb/common/util/GeoIpUtil.java

@@ -13,7 +13,7 @@ import org.jeecg.modules.adweb.common.dto.CountryAreaApiDto;
 import org.slf4j.Logger;
 import org.slf4j.LoggerFactory;
 import org.springframework.beans.factory.annotation.Value;
-import org.springframework.stereotype.Service;
+import org.springframework.stereotype.Component;
 
 import java.io.File;
 import java.io.IOException;
@@ -27,7 +27,7 @@ import java.util.TimeZone;
  * @author: Cyan
  * @time: 2019/12/16 10:33
  */
-@Service
+@Component
 public class GeoIpUtil {
 
     private static final Logger log = LoggerFactory.getLogger(GeoIpUtil.class);

+ 4 - 7
jeecg-module-system/jeecg-system-biz/src/main/java/org/jeecg/modules/adweb/enquiry/service/impl/AdwebEnquiryServiceImpl.java

@@ -8,14 +8,15 @@ import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl;
 import com.fasterxml.jackson.core.JsonProcessingException;
 import com.fasterxml.jackson.databind.ObjectMapper;
 import com.xkcoding.http.util.StringUtil;
+
 import jakarta.annotation.Resource;
+
 import lombok.extern.slf4j.Slf4j;
+
 import org.apache.commons.lang.StringUtils;
 import org.jeecg.common.constant.CacheConstant;
-import org.jeecg.common.system.vo.DictModel;
 import org.jeecg.common.system.vo.DictPropertyModel;
 import org.jeecg.common.util.FastJsonUtil;
-
 import org.jeecg.modules.adweb.common.constant.NumConstant;
 import org.jeecg.modules.adweb.common.dto.CountryAreaApiDto;
 import org.jeecg.modules.adweb.common.service.FeishuService;
@@ -48,7 +49,6 @@ import org.springframework.stereotype.Service;
 import org.springframework.web.client.RestTemplate;
 
 import java.util.*;
-import java.util.function.BiConsumer;
 import java.util.regex.Matcher;
 import java.util.regex.Pattern;
 import java.util.stream.Collectors;
@@ -77,9 +77,6 @@ public class AdwebEnquiryServiceImpl extends ServiceImpl<AdwebEnquiryMapper, Adw
     private IAdwebPublicEnquiryRuleService AdwebPublicEnquiryRuleService;
 
     @Resource
-    private GeoIpUtil geoIpService;
-
-    @Resource
     private ISysDictService dictService;
 
     @Resource
@@ -634,7 +631,7 @@ public class AdwebEnquiryServiceImpl extends ServiceImpl<AdwebEnquiryMapper, Adw
         adwebEnquiry.setFromEmail(enquiryDto.getFromEmail());
         adwebEnquiry.setPluginName(enquiryDto.getPluginName());
         adwebEnquiry.setRecordId(enquiryDto.getRecordId());
-        adwebEnquiry.setModifyRecordCtime(geoIpService.getLocalhostTime(enquiryDto.getFromIp(), enquiryDto.getRecordCtime()));
+        adwebEnquiry.setModifyRecordCtime(geoIpUtil.getLocalhostTime(enquiryDto.getFromIp(), enquiryDto.getRecordCtime()));
         adwebEnquiry.setRecordCtime(enquiryDto.getRecordCtime());
         adwebEnquiry.setSiteCode(adwebSite.getCode());
         adwebEnquiry.setModular(enquiryDto.getPluginName());

+ 4 - 6
jeecg-module-system/jeecg-system-biz/src/main/java/org/jeecg/modules/adweb/seo/controller/SeoKeywordsController.java

@@ -26,7 +26,6 @@ import org.jeecg.common.system.base.controller.JeecgController;
 import org.jeecg.common.system.query.QueryGenerator;
 import org.jeecg.modules.adweb.seo.dto.AvesApiSearchKeywordsDTO;
 import org.jeecg.modules.adweb.seo.entity.SeoKeywords;
-import org.jeecg.modules.adweb.seo.service.ISearchKeywordsService;
 import org.jeecg.modules.adweb.seo.service.ISeoKeywordsService;
 import org.jeecg.modules.adweb.seo.service.ISeoKeywordsRankService;
 import org.springframework.beans.factory.annotation.Autowired;
@@ -52,11 +51,10 @@ public class SeoKeywordsController extends JeecgController<SeoKeywords, ISeoKeyw
 	@Autowired
 	private ISeoKeywordsService seoKeywordsService;
 
-	@Autowired
-	ISeoKeywordsRankService seoKpiStatisticsService;
+	@Autowired ISeoKeywordsRankService seoKeywordsRankService;
 
 	@Autowired
-	ISearchKeywordsService searchKeywordsService;
+	ISeoKeywordsRankService seoKpiStatisticsService;
 
 	@Autowired
 	private ISysBaseAPI sysBaseAPI;
@@ -234,7 +232,7 @@ public class SeoKeywordsController extends JeecgController<SeoKeywords, ISeoKeyw
 		 if(pageList == null){
 			 return Result.error("未获取到关键词数据");
 		 }
-		 searchKeywordsService.addRankInfoByIntervalTimeByType(pageList.getRecords(), avesApiSearchKeywordsDTO.getKeywordType(),siteCode,subscriptionId,dateList);
+		 seoKeywordsRankService.addRankInfoByIntervalTimeByType(pageList.getRecords(), avesApiSearchKeywordsDTO.getKeywordType(),siteCode,subscriptionId,dateList);
 		 log.info(pageList.toString());
 		 return Result.OK(pageList);
 	 }
@@ -308,7 +306,7 @@ public class SeoKeywordsController extends JeecgController<SeoKeywords, ISeoKeyw
 			response.setHeader("Content-Disposition", "attachment;filename=" + new String((fileName + ".xlsx").getBytes(), StandardCharsets.ISO_8859_1));
 			out = response.getOutputStream();
 			writer.write(list, true);
-			searchKeywordsService.setSizeColumn(sheet, list.get(0).size());
+			seoKeywordsRankService.setSizeColumn(sheet, list.get(0).size());
 			// 写出到文件
 			writer.flush(out, true);
 			// 关闭writer,释放内存

+ 3 - 6
jeecg-module-system/jeecg-system-biz/src/main/java/org/jeecg/modules/adweb/seo/controller/SeoKeywordsRankController.java

@@ -6,7 +6,6 @@ import lombok.extern.slf4j.Slf4j;
 
 import org.jeecg.common.api.vo.Result;
 import org.jeecg.modules.adweb.seo.entity.ComprehensiveStatistics;
-import org.jeecg.modules.adweb.seo.service.ISearchKeywordsService;
 import org.jeecg.modules.adweb.seo.service.ISeoKeywordsRankService;
 import org.jeecg.modules.adweb.site.entity.AdwebSite;
 import org.springframework.beans.factory.annotation.Autowired;
@@ -26,8 +25,6 @@ import java.util.Map;
 public class SeoKeywordsRankController {
     @Autowired private ISeoKeywordsRankService seoKeywordsRankService;
 
-    @Autowired private ISearchKeywordsService searchKeywordsService;
-
     /** 获取当前用户管理的 父 站点信息 @CopyFrom: getAllSites */
     @GetMapping("/getAllSites")
     public Result<List<AdwebSite>> getAllSites() {
@@ -39,8 +36,8 @@ public class SeoKeywordsRankController {
      * 获得综合统计信息
      */
     @GetMapping(value = "/comprehensiveInfo")
-    public Result<?> getComprehensiveInfo(@RequestParam String siteCode, @RequestParam String subscriptionId) throws ParseException {
-        ComprehensiveStatistics statistics = searchKeywordsService.getComprehensiveInfo(siteCode, subscriptionId);
+    public Result<?> getComprehensiveInfo(@RequestParam String siteCode, @RequestParam String planId) throws ParseException {
+        ComprehensiveStatistics statistics = seoKeywordsRankService.getComprehensiveInfo(siteCode, planId);
         return Result.OK(statistics);
     }
 
@@ -49,7 +46,7 @@ public class SeoKeywordsRankController {
      */
     @GetMapping(value = "/getRankInfo")
     public Result<?> getRankInfo(@RequestParam("siteCode") String siteCode, @RequestParam("subscriptionId") String subscriptionId) throws ParseException {
-        Map map = searchKeywordsService.getRankInfo(siteCode, subscriptionId);
+        Map map = seoKeywordsRankService.getRankInfo(siteCode, subscriptionId);
         return Result.OK(map);
     }
 }

+ 18 - 14
jeecg-module-system/jeecg-system-biz/src/main/java/org/jeecg/modules/adweb/seo/entity/ComprehensiveStatistics.java

@@ -10,37 +10,41 @@ import lombok.Data;
 @Data
 public class ComprehensiveStatistics {
     /**
-     * 指定词
+     * 套餐指定词
      */
-    private String appointKeywordNum;
+    private Integer planAppointKeywordNum;
 
     /**
-     * 长尾词
+     * 套餐长尾词
      */
-    private String longTailKeywordNum;
+    private Integer planLongTailKeywordNum;
 
     /**
-     * 外链数
+     * 套餐外链数
      */
-    private String outerLinkNum;
+    private Integer planOuterLinkNum;
 
     /**
-     * 文章数
+     * 套餐文章数
+     */
+    private Integer planArticleNum;
+    /**
+     * 指定词
      */
-    private String articleNum;
+    private Integer appointKeywordNum;
 
     /**
-     * 询盘数
+     * 长尾词
      */
-    private String enquiryNum;
+    private Integer longTailKeywordNum;
 
     /**
-     * 访问量(UV)
+     * 外链数
      */
-    private String uvNum;
+    private Integer outerLinkNum;
 
     /**
-     * 是否展示访问量
+     * 文章数
      */
-    private boolean showUvNum;
+    private Integer articleNum;
 }

+ 0 - 26
jeecg-module-system/jeecg-system-biz/src/main/java/org/jeecg/modules/adweb/seo/service/ISearchKeywordsService.java

@@ -1,26 +0,0 @@
-package org.jeecg.modules.adweb.seo.service;
-
-import org.apache.poi.ss.usermodel.Sheet;
-import org.jeecg.modules.adweb.seo.entity.ComprehensiveStatistics;
-import org.jeecg.modules.adweb.seo.entity.SeoKeywords;
-import org.jeecg.modules.adweb.seo.vo.SeoRankInfoVO;
-
-import java.text.ParseException;
-import java.util.List;
-import java.util.Map;
-
-/**
- * @author Zenas
- */
-public interface ISearchKeywordsService {
-
-    ComprehensiveStatistics getComprehensiveInfo(String siteCode, String historyId) throws ParseException;
-
-    Map getRankInfo(String siteCode, String historyId) throws ParseException;
-
-    List<SeoRankInfoVO> getSeoRankInfo(String siteCode, String historyId) throws ParseException;
-
-    void addRankInfoByIntervalTimeByType(List<SeoKeywords> keywordList, Integer keywordType, String siteCode, String historyId, List<String> dateList) throws Exception;
-
-    void setSizeColumn(Sheet sheet, int size);
-}

+ 17 - 0
jeecg-module-system/jeecg-system-biz/src/main/java/org/jeecg/modules/adweb/seo/service/ISeoKeywordsRankService.java

@@ -1,8 +1,14 @@
 package org.jeecg.modules.adweb.seo.service;
 
+import org.apache.poi.ss.usermodel.Sheet;
+import org.jeecg.modules.adweb.seo.entity.ComprehensiveStatistics;
+import org.jeecg.modules.adweb.seo.entity.SeoKeywords;
+import org.jeecg.modules.adweb.seo.vo.SeoRankInfoVO;
 import org.jeecg.modules.adweb.site.entity.AdwebSite;
 
+import java.text.ParseException;
 import java.util.List;
+import java.util.Map;
 
 /**
  * @Description: seo_kpi_statistics
@@ -13,4 +19,15 @@ import java.util.List;
 public interface ISeoKeywordsRankService {
 
     List<AdwebSite> getAllSites();
+
+    ComprehensiveStatistics getComprehensiveInfo(String siteCode, String subscriptionId);
+
+    Map getRankInfo(String siteCode, String historyId) throws ParseException;
+
+    List<SeoRankInfoVO> getSeoRankInfo(String siteCode, String historyId)
+            throws ParseException;
+
+    void addRankInfoByIntervalTimeByType(List<SeoKeywords> keywordList, Integer keywordType, String siteCode, String historyId, List<String> dateList) throws Exception;
+
+    void setSizeColumn(Sheet sheet, int size);
 }

+ 0 - 259
jeecg-module-system/jeecg-system-biz/src/main/java/org/jeecg/modules/adweb/seo/service/impl/SearchKeywordsServiceImpl.java

@@ -1,259 +0,0 @@
-package org.jeecg.modules.adweb.seo.service.impl;
-
-import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper;
-import com.xkcoding.http.util.StringUtil;
-
-import jakarta.annotation.Resource;
-
-import lombok.extern.slf4j.Slf4j;
-
-import org.apache.poi.ss.usermodel.Cell;
-import org.apache.poi.ss.usermodel.CellType;
-import org.apache.poi.ss.usermodel.Row;
-import org.apache.poi.ss.usermodel.Sheet;
-import org.jeecg.modules.adweb.common.util.ListUtil;
-import org.jeecg.modules.adweb.seo.entity.*;
-import org.jeecg.modules.adweb.seo.mapper.SeoKeywordsMapper;
-import org.jeecg.modules.adweb.seo.service.*;
-import org.jeecg.modules.adweb.seo.vo.RankInfoVO;
-import org.jeecg.modules.adweb.seo.vo.SeoRankInfoVO;
-import org.springframework.beans.factory.annotation.Autowired;
-import org.springframework.stereotype.Service;
-
-import java.text.ParseException;
-import java.util.*;
-import java.util.stream.Collectors;
-
-/**
- * @author Zenas
- */
-@Slf4j
-@Service
-public class SearchKeywordsServiceImpl implements ISearchKeywordsService {
-
-    @Resource private SeoKeywordsMapper seoKeywordsMapper;
-
-    @Autowired private ISeoKeywordsService seoKeywordsService;
-
-    @Autowired private ISeoKeywordsSerpService seoKeywordsSerpService;
-
-    /**
-     * 获得综合统计的数据
-     *
-     * @param siteCode 站点code
-     * @param subscriptionId 订阅表id
-     * @return 综合统计的数据
-     */
-    @Override
-    public ComprehensiveStatistics getComprehensiveInfo(String siteCode, String subscriptionId) {
-        ComprehensiveStatistics comprehensiveStatistics = new ComprehensiveStatistics();
-
-        return comprehensiveStatistics;
-    }
-
-    @Override
-    public Map getRankInfo(String siteCode, String historyId) throws ParseException {
-        List<SeoRankInfoVO> seoRankInfos = this.getSeoRankInfo(siteCode, historyId);
-
-        Map map = new HashMap<>();
-        map.put("appointKeyword", seoRankInfos.get(0));
-        map.put("longTailKeyword", seoRankInfos.get(1));
-        return map;
-    }
-
-    /**
-     * 获取一个站点的关键词排名数量
-     *
-     * @param siteCode 站点code
-     * @return 关键词排名数量
-     */
-    @Override
-    public List<SeoRankInfoVO> getSeoRankInfo(String siteCode, String historyId)
-            throws ParseException {
-        // 排名信息
-        SeoRankInfoVO appointRankInfo = new SeoRankInfoVO();
-        SeoRankInfoVO longTailRankInfo = new SeoRankInfoVO();
-        // 1-10
-        Map<String, Integer> first = dealKeywordsRankData(siteCode, historyId, 1, 10);
-        if (first.containsKey("appoint")) {
-            appointRankInfo.setFirstNum(first.get("appoint"));
-        }
-        if (first.containsKey("longTail")) {
-            longTailRankInfo.setFirstNum(first.get("longTail"));
-        }
-
-        // 11-30
-        Map<String, Integer> second = dealKeywordsRankData(siteCode, historyId, 11, 30);
-        if (second.containsKey("appoint")) {
-            appointRankInfo.setSecondNum(second.get("appoint"));
-        }
-        if (second.containsKey("longTail")) {
-            longTailRankInfo.setSecondNum(second.get("longTail"));
-        }
-        // 31-100
-        Map<String, Integer> third = dealKeywordsRankData(siteCode, historyId, 31, 100);
-        if (third.containsKey("appoint")) {
-            appointRankInfo.setThirdType(third.get("appoint"));
-        }
-        if (third.containsKey("longTail")) {
-            longTailRankInfo.setThirdType(third.get("longTail"));
-        }
-        List<SeoRankInfoVO> list = new ArrayList<>();
-        list.add(appointRankInfo);
-        list.add(longTailRankInfo);
-        return list;
-    }
-
-    private Map<String, Integer> dealKeywordsRankData(
-            String siteCode, String historyId, Integer rankStart, Integer rankEnd) {
-        Map<String, Integer> KeywordsRankMap = new HashMap<>();
-
-        // 指定词
-        List<String> appointDateList =
-                seoKeywordsService.getKeywordsLastSevenDays(siteCode, 1, historyId);
-        String appointDatesStr = "";
-        if (ListUtil.notEmpty(appointDateList)) {
-            for (int i = 0; i < appointDateList.size(); i++) {
-                if (i == appointDateList.size() - 1) {
-                    appointDatesStr += "'" + appointDateList.get(i) + "'";
-                } else {
-                    appointDatesStr += "'" + appointDateList.get(i) + "'" + ",";
-                }
-            }
-            appointDatesStr = "(" + appointDatesStr + ")";
-
-            // 指定词
-            RankInfoVO appointRankInfo =
-                    seoKeywordsMapper.queryKeywordNumByRankRange(
-                            siteCode, historyId, rankStart, rankEnd, appointDatesStr, "1");
-            if (appointRankInfo != null) {
-                KeywordsRankMap.put("appoint", appointRankInfo.getKeywordNum());
-            }
-        }
-
-        // 长尾词
-        List<String> dateList = seoKeywordsService.getKeywordsLastSevenDays(siteCode, 2, historyId);
-        String dateStr = "";
-        if (ListUtil.notEmpty(dateList)) {
-            for (int i = 0; i < dateList.size(); i++) {
-                if (i == dateList.size() - 1) {
-                    dateStr += "'" + dateList.get(i) + "'";
-                } else {
-                    dateStr += "'" + dateList.get(i) + "'" + ",";
-                }
-            }
-            dateStr = "(" + dateStr + ")";
-            RankInfoVO longRankInfo =
-                    seoKeywordsMapper.queryKeywordNumByRankRange(
-                            siteCode, historyId, rankStart, rankEnd, dateStr, "2");
-            if (longRankInfo != null) {
-                KeywordsRankMap.put("longTail", longRankInfo.getKeywordNum());
-            }
-        }
-        return KeywordsRankMap;
-    }
-
-    /**
-     * 给列表中的关键字增加排名信息
-     * 根据关键词类型
-     * @param keywordList 关键词列表
-     *
-     */
-    @Override
-    public void addRankInfoByIntervalTimeByType(List<SeoKeywords> keywordList, Integer keywordType, String siteCode, String historyId, List<String> dateList) throws Exception {
-        if (ListUtil.isEmpty(keywordList)) {
-            return;
-        }
-
-        String datesStr = "";
-        if(ListUtil.isEmpty(dateList)){
-            return;
-        }else{
-            for(int i = 0; i < dateList.size(); i++){
-                if(i == dateList.size() - 1){
-                    datesStr +=  "'" + dateList.get(i) + "'";
-                }else{
-                    datesStr +=  "'" + dateList.get(i) + "'" + ",";
-                }
-            }
-            datesStr = "(" + datesStr + ")";
-        }
-        List<Integer> keywordIds = keywordList.stream().map(SeoKeywords::getId).collect(Collectors.toList());
-        List<SeoKeywordsSerp> serpList = seoKeywordsSerpService.list(new LambdaQueryWrapper<SeoKeywordsSerp>()
-                .in(SeoKeywordsSerp::getKeywordsId, keywordIds)
-                .in(SeoKeywordsSerp::getSeDate, dateList));
-        List<SeoKeywordsSerp> noRepeatSerpList = new ArrayList<>();
-        Map<String, Integer> serpMap = new HashMap<>();
-        if(ListUtil.notEmpty(serpList)){
-            for (SeoKeywordsSerp seoKeywordsSerp : serpList) {
-                seoKeywordsSerp.setKey(seoKeywordsSerp.getKeywordsId() + "_" + seoKeywordsSerp.getSeDate());
-                if(ListUtil.notEmpty(noRepeatSerpList)){
-                    List<SeoKeywordsSerp> have = noRepeatSerpList.stream().filter(o -> o.getKey().equals(seoKeywordsSerp.getKey())).collect(Collectors.toList());
-                    if(ListUtil.isEmpty(have)){
-                        serpMap.put(seoKeywordsSerp.getKey(),seoKeywordsSerp.getRankAbsolute());
-                        noRepeatSerpList.add(seoKeywordsSerp);
-                    }else{
-                        log.info("重复的数据:" + seoKeywordsSerp.getKey());
-                    }
-                }else {
-                    serpMap.put(seoKeywordsSerp.getKey(),seoKeywordsSerp.getRankAbsolute());
-                    noRepeatSerpList.add(seoKeywordsSerp);
-                }
-            }
-        }
-
-        for (SeoKeywords keyword : keywordList) {
-            Map<String, String> rankInfo = new LinkedHashMap<>();
-            for (String date : dateList) {
-
-                // 从Map中获取那天的排名信息
-                Integer rankAbsolute = serpMap.get(keyword.getId() + "_" + date);
-
-                if(rankAbsolute == null){
-                    rankAbsolute = 0;
-                }
-                rankInfo.put(date, rankAbsolute + "");
-            }
-
-            keyword.setRankInfo(rankInfo);
-            keyword.setIpAddress("223.112.18.226");
-
-            keyword.setDateList(dateList);
-            if(StringUtil.isEmpty(keyword.getSearchUrl())){
-                keyword.setSearchUrl("https://www.google.com/");
-            }
-        }
-    }
-
-    /**
-     * 自适应宽度(中文支持)
-     * @param sheet
-     * @param size 因为for循环从0开始,size值为 列数-1
-     */
-    @Override
-    public void setSizeColumn(Sheet sheet, int size) {
-        for (int columnNum = 0; columnNum <= size; columnNum++) {
-            int columnWidth = sheet.getColumnWidth(columnNum) / 256;
-            for (int rowNum = 0; rowNum <= sheet.getLastRowNum(); rowNum++) {
-                Row currentRow;
-                //当前行未被使用过
-                if (sheet.getRow(rowNum) == null) {
-                    currentRow = sheet.createRow(rowNum);
-                } else {
-                    currentRow = sheet.getRow(rowNum);
-                }
-
-                if (currentRow.getCell(columnNum) != null) {
-                    Cell currentCell = currentRow.getCell(columnNum);
-                    if (currentCell.getCellType() == CellType.STRING) {
-                        int length = currentCell.getStringCellValue().getBytes().length;
-                        if (columnWidth < length) {
-                            columnWidth = length;
-                        }
-                    }
-                }
-            }
-            sheet.setColumnWidth(columnNum, columnWidth * 256);
-        }
-    }
-}

+ 279 - 10
jeecg-module-system/jeecg-system-biz/src/main/java/org/jeecg/modules/adweb/seo/service/impl/SeoKeywordsRankServiceImpl.java

@@ -1,25 +1,43 @@
 package org.jeecg.modules.adweb.seo.service.impl;
 
+import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper;
 import com.baomidou.mybatisplus.core.conditions.query.QueryWrapper;
 
+import com.xkcoding.http.util.StringUtil;
+import jakarta.annotation.Resource;
 import lombok.extern.slf4j.Slf4j;
 
+import org.apache.poi.ss.usermodel.Cell;
+import org.apache.poi.ss.usermodel.CellType;
+import org.apache.poi.ss.usermodel.Row;
+import org.apache.poi.ss.usermodel.Sheet;
 import org.apache.shiro.SecurityUtils;
 import org.jeecg.common.system.api.ISysBaseAPI;
 import org.jeecg.common.system.vo.LoginUser;
 import org.jeecg.modules.adweb.common.constant.AdwebConstant;
 import org.jeecg.modules.adweb.common.util.ListUtil;
+import org.jeecg.modules.adweb.seo.entity.ComprehensiveStatistics;
+import org.jeecg.modules.adweb.seo.entity.SeoKeywords;
+import org.jeecg.modules.adweb.seo.entity.SeoKeywordsSerp;
 import org.jeecg.modules.adweb.seo.entity.SeoPlanSubscription;
+import org.jeecg.modules.adweb.seo.mapper.SeoKeywordsMapper;
 import org.jeecg.modules.adweb.seo.service.ISeoKeywordsRankService;
+import org.jeecg.modules.adweb.seo.service.ISeoKeywordsSerpService;
+import org.jeecg.modules.adweb.seo.service.ISeoKeywordsService;
 import org.jeecg.modules.adweb.seo.service.ISeoPlanSubscriptionService;
+import org.jeecg.modules.adweb.seo.vo.RankInfoVO;
+import org.jeecg.modules.adweb.seo.vo.SeoRankInfoVO;
 import org.jeecg.modules.adweb.site.entity.AdwebSite;
 import org.jeecg.modules.adweb.site.service.IAdwebSiteService;
 import org.jeecg.modules.adweb.site.service.ISiteUserPermissionService;
+import org.jeecg.modules.adweb.subscribePlan.entity.SubscribePlan;
+import org.jeecg.modules.adweb.subscribePlan.mapper.SubscribePlanMapper;
 import org.springframework.beans.factory.annotation.Autowired;
 import org.springframework.stereotype.Service;
 
-import java.util.ArrayList;
-import java.util.List;
+import java.text.ParseException;
+import java.util.*;
+import java.util.stream.Collectors;
 
 /**
  * @Description: seo_kpi_statistics @Author: jeecg-boot @Date: 2024-10-09 @Version: V1.0
@@ -36,6 +54,14 @@ public class SeoKeywordsRankServiceImpl implements ISeoKeywordsRankService {
 
     @Autowired private ISeoPlanSubscriptionService seoPlanSubscriptionService;
 
+    @Autowired private ISeoKeywordsService seoKeywordsService;
+
+    @Resource private SeoKeywordsMapper seoKeywordsMapper;
+
+    @Autowired private ISeoKeywordsSerpService seoKeywordsSerpService;
+
+    @Resource private SubscribePlanMapper subscribePlanMapper;
+
     @Override
     public List<AdwebSite> getAllSites() {
         log.info("获取当前用户管理的 父 站点信息");
@@ -74,20 +100,263 @@ public class SeoKeywordsRankServiceImpl implements ISeoKeywordsRankService {
         historyQueryWrapper.select("id", "plan_id", "site_id", "plan_name", "plan_type");
         historyQueryWrapper.in("site_id", siteIds);
         historyQueryWrapper.eq("status", 1);
-        List<SeoPlanSubscription> histories = seoPlanSubscriptionService.list(historyQueryWrapper);
-        if (ListUtil.isEmpty(histories)) {
+        List<SeoPlanSubscription> seoPlanSubscriptions = seoPlanSubscriptionService.list(historyQueryWrapper);
+        if (ListUtil.isEmpty(seoPlanSubscriptions)) {
             return;
         }
 
         for (AdwebSite site : siteList) {
-            for (SeoPlanSubscription history : histories) {
-                if (history.getSiteId().equals(site.getId() + "")) {
-                    site.setPlanId(history.getPlanId());
-                    site.setPlanName(history.getPlanName());
-                    site.setHistoryId(history.getId());
-                    site.setPlanType(history.getPlanType());
+            for (SeoPlanSubscription seoPlanSubscription : seoPlanSubscriptions) {
+                if (seoPlanSubscription.getSiteId().equals(site.getId() + "")) {
+                    site.setPlanId(seoPlanSubscription.getPlanId());
+                    site.setPlanName(seoPlanSubscription.getPlanName());
+                    site.setSubscriptionId(seoPlanSubscription.getId());
+                    site.setPlanType(seoPlanSubscription.getPlanType());
+                }
+            }
+        }
+    }
+
+    /**
+     * 获得综合统计的数据
+     *
+     * @param siteCode 站点code
+     * @param planId 套餐表id
+     * @return 综合统计的数据
+     */
+    @Override
+    public ComprehensiveStatistics getComprehensiveInfo(String siteCode, String planId) {
+        ComprehensiveStatistics comprehensiveStatistics = new ComprehensiveStatistics();
+        SubscribePlan subscribePlan = subscribePlanMapper.selectById(planId);
+        log.info("planId:{}, subscribePlan:{}", planId, subscribePlan.toString());
+        if (subscribePlan != null) {
+            // 添加套餐数据
+            comprehensiveStatistics.setPlanAppointKeywordNum(subscribePlan.getSpecifyKeyword());
+            comprehensiveStatistics.setPlanLongTailKeywordNum(subscribePlan.getLongTailKeyword());
+            comprehensiveStatistics.setPlanOuterLinkNum(subscribePlan.getExternalLinks());
+            comprehensiveStatistics.setPlanArticleNum(subscribePlan.getArticle());
+        }
+        // 添加站点数据
+        // 1.获取站点指定词数量
+        QueryWrapper<SeoKeywords> seoKeywordsQueryWrapper = new QueryWrapper<>();
+        seoKeywordsQueryWrapper.eq("site_code", siteCode);
+        seoKeywordsQueryWrapper.eq("status", 1);
+        seoKeywordsQueryWrapper.eq("keyword_type", 1);
+        Long appointKeywordNum = seoKeywordsMapper.selectCount(seoKeywordsQueryWrapper);
+        // 2.获取站点关键词数量
+        QueryWrapper<SeoKeywords> seoKeywordsQueryWrapper1 = new QueryWrapper<>();
+        seoKeywordsQueryWrapper1.eq("site_code", siteCode);
+        seoKeywordsQueryWrapper1.eq("status", 1);
+        seoKeywordsQueryWrapper1.eq("keyword_type", 2);
+        Long longTailKeywordNum = seoKeywordsMapper.selectCount(seoKeywordsQueryWrapper1);
+        comprehensiveStatistics.setAppointKeywordNum(Math.toIntExact(appointKeywordNum));
+        comprehensiveStatistics.setLongTailKeywordNum(Math.toIntExact(longTailKeywordNum));
+        return comprehensiveStatistics;
+    }
+
+    @Override
+    public Map getRankInfo(String siteCode, String subscriptionId) throws ParseException {
+        List<SeoRankInfoVO> seoRankInfos = this.getSeoRankInfo(siteCode, subscriptionId);
+
+        Map map = new HashMap<>();
+        map.put("appointKeyword", seoRankInfos.get(0));
+        map.put("longTailKeyword", seoRankInfos.get(1));
+        return map;
+    }
+
+    /**
+     * 获取一个站点的关键词排名数量
+     *
+     * @param siteCode 站点code
+     * @return 关键词排名数量
+     */
+    @Override
+    public List<SeoRankInfoVO> getSeoRankInfo(String siteCode, String subscriptionId)
+            throws ParseException {
+        // 排名信息
+        SeoRankInfoVO appointRankInfo = new SeoRankInfoVO();
+        SeoRankInfoVO longTailRankInfo = new SeoRankInfoVO();
+        // 1-10
+        Map<String, Integer> first = dealKeywordsRankData(siteCode, subscriptionId, 1, 10);
+        if (first.containsKey("appoint")) {
+            appointRankInfo.setFirstNum(first.get("appoint"));
+        }
+        if (first.containsKey("longTail")) {
+            longTailRankInfo.setFirstNum(first.get("longTail"));
+        }
+
+        // 11-30
+        Map<String, Integer> second = dealKeywordsRankData(siteCode, subscriptionId, 11, 30);
+        if (second.containsKey("appoint")) {
+            appointRankInfo.setSecondNum(second.get("appoint"));
+        }
+        if (second.containsKey("longTail")) {
+            longTailRankInfo.setSecondNum(second.get("longTail"));
+        }
+        // 31-100
+        Map<String, Integer> third = dealKeywordsRankData(siteCode, subscriptionId, 31, 100);
+        if (third.containsKey("appoint")) {
+            appointRankInfo.setThirdType(third.get("appoint"));
+        }
+        if (third.containsKey("longTail")) {
+            longTailRankInfo.setThirdType(third.get("longTail"));
+        }
+        List<SeoRankInfoVO> list = new ArrayList<>();
+        list.add(appointRankInfo);
+        list.add(longTailRankInfo);
+        return list;
+    }
+
+    private Map<String, Integer> dealKeywordsRankData(
+            String siteCode, String subscriptionId, Integer rankStart, Integer rankEnd) {
+        Map<String, Integer> KeywordsRankMap = new HashMap<>();
+
+        // 指定词
+        List<String> appointDateList =
+                seoKeywordsService.getKeywordsLastSevenDays(siteCode, 1, subscriptionId);
+        String appointDatesStr = "";
+        if (ListUtil.notEmpty(appointDateList)) {
+            for (int i = 0; i < appointDateList.size(); i++) {
+                if (i == appointDateList.size() - 1) {
+                    appointDatesStr += "'" + appointDateList.get(i) + "'";
+                } else {
+                    appointDatesStr += "'" + appointDateList.get(i) + "'" + ",";
+                }
+            }
+            appointDatesStr = "(" + appointDatesStr + ")";
+
+            // 指定词
+            RankInfoVO appointRankInfo =
+                    seoKeywordsMapper.queryKeywordNumByRankRange(
+                            siteCode, subscriptionId, rankStart, rankEnd, appointDatesStr, "1");
+            if (appointRankInfo != null) {
+                KeywordsRankMap.put("appoint", appointRankInfo.getKeywordNum());
+            }
+        }
+
+        // 长尾词
+        List<String> dateList = seoKeywordsService.getKeywordsLastSevenDays(siteCode, 2, subscriptionId);
+        String dateStr = "";
+        if (ListUtil.notEmpty(dateList)) {
+            for (int i = 0; i < dateList.size(); i++) {
+                if (i == dateList.size() - 1) {
+                    dateStr += "'" + dateList.get(i) + "'";
+                } else {
+                    dateStr += "'" + dateList.get(i) + "'" + ",";
+                }
+            }
+            dateStr = "(" + dateStr + ")";
+            RankInfoVO longRankInfo =
+                    seoKeywordsMapper.queryKeywordNumByRankRange(
+                            siteCode, subscriptionId, rankStart, rankEnd, dateStr, "2");
+            if (longRankInfo != null) {
+                KeywordsRankMap.put("longTail", longRankInfo.getKeywordNum());
+            }
+        }
+        return KeywordsRankMap;
+    }
+
+    /**
+     * 给列表中的关键字增加排名信息
+     * 根据关键词类型
+     * @param keywordList 关键词列表
+     *
+     */
+    @Override
+    public void addRankInfoByIntervalTimeByType(List<SeoKeywords> keywordList, Integer keywordType, String siteCode, String subscriptionId, List<String> dateList) throws Exception {
+        if (ListUtil.isEmpty(keywordList)) {
+            return;
+        }
+
+        String datesStr = "";
+        if(ListUtil.isEmpty(dateList)){
+            return;
+        }else{
+            for(int i = 0; i < dateList.size(); i++){
+                if(i == dateList.size() - 1){
+                    datesStr +=  "'" + dateList.get(i) + "'";
+                }else{
+                    datesStr +=  "'" + dateList.get(i) + "'" + ",";
+                }
+            }
+            datesStr = "(" + datesStr + ")";
+        }
+        List<Integer> keywordIds = keywordList.stream().map(SeoKeywords::getId).collect(Collectors.toList());
+        List<SeoKeywordsSerp> serpList = seoKeywordsSerpService.list(new LambdaQueryWrapper<SeoKeywordsSerp>()
+                .in(SeoKeywordsSerp::getKeywordsId, keywordIds)
+                .in(SeoKeywordsSerp::getSeDate, dateList));
+        List<SeoKeywordsSerp> noRepeatSerpList = new ArrayList<>();
+        Map<String, Integer> serpMap = new HashMap<>();
+        if(ListUtil.notEmpty(serpList)){
+            for (SeoKeywordsSerp seoKeywordsSerp : serpList) {
+                seoKeywordsSerp.setKey(seoKeywordsSerp.getKeywordsId() + "_" + seoKeywordsSerp.getSeDate());
+                if(ListUtil.notEmpty(noRepeatSerpList)){
+                    List<SeoKeywordsSerp> have = noRepeatSerpList.stream().filter(o -> o.getKey().equals(seoKeywordsSerp.getKey())).collect(Collectors.toList());
+                    if(ListUtil.isEmpty(have)){
+                        serpMap.put(seoKeywordsSerp.getKey(),seoKeywordsSerp.getRankAbsolute());
+                        noRepeatSerpList.add(seoKeywordsSerp);
+                    }else{
+                        log.info("重复的数据:" + seoKeywordsSerp.getKey());
+                    }
+                }else {
+                    serpMap.put(seoKeywordsSerp.getKey(),seoKeywordsSerp.getRankAbsolute());
+                    noRepeatSerpList.add(seoKeywordsSerp);
+                }
+            }
+        }
+
+        for (SeoKeywords keyword : keywordList) {
+            Map<String, String> rankInfo = new LinkedHashMap<>();
+            for (String date : dateList) {
+
+                // 从Map中获取那天的排名信息
+                Integer rankAbsolute = serpMap.get(keyword.getId() + "_" + date);
+
+                if(rankAbsolute == null){
+                    rankAbsolute = 0;
+                }
+                rankInfo.put(date, rankAbsolute + "");
+            }
+
+            keyword.setRankInfo(rankInfo);
+            keyword.setIpAddress("223.112.18.226");
+
+            keyword.setDateList(dateList);
+            if(StringUtil.isEmpty(keyword.getSearchUrl())){
+                keyword.setSearchUrl("https://www.google.com/");
+            }
+        }
+    }
+
+    /**
+     * 自适应宽度(中文支持)
+     * @param sheet
+     * @param size 因为for循环从0开始,size值为 列数-1
+     */
+    @Override
+    public void setSizeColumn(Sheet sheet, int size) {
+        for (int columnNum = 0; columnNum <= size; columnNum++) {
+            int columnWidth = sheet.getColumnWidth(columnNum) / 256;
+            for (int rowNum = 0; rowNum <= sheet.getLastRowNum(); rowNum++) {
+                Row currentRow;
+                //当前行未被使用过
+                if (sheet.getRow(rowNum) == null) {
+                    currentRow = sheet.createRow(rowNum);
+                } else {
+                    currentRow = sheet.getRow(rowNum);
+                }
+
+                if (currentRow.getCell(columnNum) != null) {
+                    Cell currentCell = currentRow.getCell(columnNum);
+                    if (currentCell.getCellType() == CellType.STRING) {
+                        int length = currentCell.getStringCellValue().getBytes().length;
+                        if (columnWidth < length) {
+                            columnWidth = length;
+                        }
+                    }
                 }
             }
+            sheet.setColumnWidth(columnNum, columnWidth * 256);
         }
     }
 }

+ 13 - 5
jeecg-module-system/jeecg-system-biz/src/main/java/org/jeecg/modules/adweb/site/entity/AdwebSite.java

@@ -7,15 +7,20 @@ import com.baomidou.mybatisplus.annotation.IdType;
 import com.baomidou.mybatisplus.annotation.TableField;
 import com.baomidou.mybatisplus.annotation.TableId;
 import com.baomidou.mybatisplus.annotation.TableName;
-import lombok.Data;
 import com.fasterxml.jackson.annotation.JsonFormat;
-import org.springframework.format.annotation.DateTimeFormat;
-import org.jeecgframework.poi.excel.annotation.Excel;
-import org.jeecg.common.aspect.annotation.Dict;
+
 import io.swagger.v3.oas.annotations.media.Schema;
+
+import lombok.Data;
 import lombok.EqualsAndHashCode;
 import lombok.experimental.Accessors;
 
+import org.jeecg.common.aspect.annotation.Dict;
+import org.jeecgframework.poi.excel.annotation.Excel;
+import org.springframework.format.annotation.DateTimeFormat;
+
+import java.io.Serializable;
+
 /**
  * @Description: adweb站点配置表单
  * @Author: jeecg-boot
@@ -388,6 +393,9 @@ public class AdwebSite implements Serializable {
     @TableField(exist = false)
     private String planType;
 
+    /**
+     * 订阅ID
+     */
     @TableField(exist = false)
-    private String historyId;
+    private String subscriptionId;
 }