|
@@ -1,12 +1,17 @@
|
|
|
package org.jeecg.modules.adweb.dmp.service;
|
|
|
|
|
|
+import static org.jeecg.modules.adweb.dmp.vo.report.SiteOverviewStatsVO.DailyStatsVO;
|
|
|
+
|
|
|
import com.baomidou.mybatisplus.core.conditions.query.LambdaQueryWrapper;
|
|
|
import com.google.common.base.Strings;
|
|
|
+import com.google.common.collect.Maps;
|
|
|
+import com.google.common.collect.Sets;
|
|
|
|
|
|
import lombok.extern.slf4j.Slf4j;
|
|
|
|
|
|
import org.apache.commons.compress.utils.Lists;
|
|
|
import org.apache.commons.lang3.StringUtils;
|
|
|
+import org.apache.commons.lang3.tuple.ImmutableTriple;
|
|
|
import org.jeecg.modules.adweb.common.enums.CountryCode;
|
|
|
import org.jeecg.modules.adweb.common.util.AdwebRedisUtil;
|
|
|
import org.jeecg.modules.adweb.common.util.DateUtil;
|
|
@@ -19,10 +24,8 @@ import org.jeecg.modules.adweb.dmp.dto.google.analytics.report.ReportType;
|
|
|
import org.jeecg.modules.adweb.dmp.dto.google.analytics.report.data.CustomReportData;
|
|
|
import org.jeecg.modules.adweb.dmp.entity.GoogleGTM;
|
|
|
import org.jeecg.modules.adweb.dmp.service.google.GAReportService;
|
|
|
-import org.jeecg.modules.adweb.dmp.vo.report.CountryStatsVO;
|
|
|
-import org.jeecg.modules.adweb.dmp.vo.report.DeviceStatsVO;
|
|
|
-import org.jeecg.modules.adweb.dmp.vo.report.PagePathStatsVO;
|
|
|
-import org.jeecg.modules.adweb.dmp.vo.report.SourceMediumStatsVO;
|
|
|
+import org.jeecg.modules.adweb.dmp.vo.report.*;
|
|
|
+import org.jeecg.modules.adweb.enquiry.mapper.AdwebEnquiryMapper;
|
|
|
import org.jeecg.modules.adweb.site.entity.AdwebSite;
|
|
|
import org.jeecg.modules.adweb.site.service.IAdwebSiteService;
|
|
|
import org.jeecg.modules.redis.CacheConfig;
|
|
@@ -35,24 +38,133 @@ import java.util.*;
|
|
|
import java.util.stream.Collectors;
|
|
|
|
|
|
/**
|
|
|
- * 从DataBridge直接生成GA报表数据,不依赖DB查询 - http://data-bridge.v3.adwebcloud.com:9002/swagger-ui/index.html
|
|
|
+ * 从DataBridge实时拉取GA报表数据,非DB离线查询 - http://data-bridge.v3.adwebcloud.com:9002/swagger-ui/index.html
|
|
|
*
|
|
|
* @author wfansh
|
|
|
*/
|
|
|
@Service
|
|
|
@Slf4j
|
|
|
-public class DMPReportService {
|
|
|
+public class RealtimeReportService {
|
|
|
|
|
|
@Autowired private IAdwebSiteService adwebSiteService;
|
|
|
@Autowired private IGoogleGTMService googleGTMService;
|
|
|
+ @Autowired private AdwebEnquiryMapper adwebEnquiryMapper;
|
|
|
@Autowired private GAReportService gaReportService;
|
|
|
@Autowired private AdwebRedisUtil redisUtil;
|
|
|
|
|
|
+ /**
|
|
|
+ * 拉取Google Analytics - Daily报表
|
|
|
+ *
|
|
|
+ * <p>如数据库中某天数据缺失,需填充空数据
|
|
|
+ *
|
|
|
+ * <p>填充每日询盘数据
|
|
|
+ */
|
|
|
+ @Cacheable(
|
|
|
+ cacheManager = CacheConfig.TTL_CACHE_MANAGER,
|
|
|
+ cacheNames =
|
|
|
+ "dmp:realtime:getDailyStats"
|
|
|
+ + TTLCacheManager.TTL_SPLITTER
|
|
|
+ + 60 * 10) // Redis TTL为10分钟
|
|
|
+ public List<SiteOverviewStatsVO.DailyStatsVO> getDailyStats(
|
|
|
+ String siteCode, Date start, Date end) {
|
|
|
+ GoogleGTM googleGTM = this.getGTMAccount(siteCode);
|
|
|
+ if (StringUtils.isBlank(googleGTM.getGaPropertyId())) {
|
|
|
+ log.info("Google Analytics帐号未配置, siteCode = {}", siteCode);
|
|
|
+ return Collections.EMPTY_LIST;
|
|
|
+ }
|
|
|
+
|
|
|
+ // 0. 起止日期为空时,设置默认区间 - start = 网站发布日期 & end = 今天
|
|
|
+ if (start == null || end == null) {
|
|
|
+ AdwebSite adwebSite = adwebSiteService.getSiteByCode(siteCode);
|
|
|
+ start =
|
|
|
+ DateUtil.getTodayZeroTime(
|
|
|
+ Optional.ofNullable(adwebSite.getIssueTime())
|
|
|
+ .orElse(adwebSite.getCtime()));
|
|
|
+ end = DateUtil.getTodayZeroTime(new Date());
|
|
|
+ }
|
|
|
+
|
|
|
+ // 1. 构建GA报表请求参数
|
|
|
+ GAReportRequestDTO gaReportRequest =
|
|
|
+ this.buildGAReportRequest(
|
|
|
+ googleGTM.getGaPropertyId(),
|
|
|
+ start,
|
|
|
+ end,
|
|
|
+ List.of(
|
|
|
+ ReportConstant.METRIC_TOTAL_USERS,
|
|
|
+ ReportConstant.METRIC_SESSIONS,
|
|
|
+ ReportConstant.METRIC_BOUNCE_RATE,
|
|
|
+ ReportConstant.METRIC_AVG_SESSION_DURATION,
|
|
|
+ ReportConstant.METRIC_SCREEN_PAGE_VIEWS,
|
|
|
+ ReportConstant.METRIC_SCREEN_PAGE_VIEWS_PER_SESSION),
|
|
|
+ List.of(ReportConstant.DIMENSION_DATE),
|
|
|
+ OrderByType.DIMENSIONS,
|
|
|
+ ReportConstant.DIMENSION_DATE,
|
|
|
+ false);
|
|
|
+
|
|
|
+ // 2. 请求API接口
|
|
|
+ List<CustomReportData> reportDataList =
|
|
|
+ gaReportService.runGAReport(gaReportRequest, CustomReportData.class);
|
|
|
+
|
|
|
+ // 3. 转化为VOs
|
|
|
+ Map<String, DailyStatsVO> dailyStatsVOs = Maps.newHashMap();
|
|
|
+ for (CustomReportData reportData : reportDataList) {
|
|
|
+ DailyStatsVO dailyStatsVO = new DailyStatsVO();
|
|
|
+ dailyStatsVO.setDate(reportData.get(ReportConstant.DIMENSION_DATE));
|
|
|
+ dailyStatsVO.setTotalUsers(
|
|
|
+ Integer.parseInt(reportData.get(ReportConstant.METRIC_TOTAL_USERS)));
|
|
|
+ dailyStatsVO.setSessions(
|
|
|
+ Integer.parseInt(reportData.get(ReportConstant.METRIC_SESSIONS)));
|
|
|
+ dailyStatsVO.setBounceRate(
|
|
|
+ Double.parseDouble(reportData.get(ReportConstant.METRIC_BOUNCE_RATE)));
|
|
|
+ dailyStatsVO.setAvgSessionDuration(
|
|
|
+ Double.parseDouble(reportData.get(ReportConstant.METRIC_AVG_SESSION_DURATION)));
|
|
|
+ dailyStatsVO.setPageViews(
|
|
|
+ Integer.parseInt(reportData.get(ReportConstant.METRIC_SCREEN_PAGE_VIEWS)));
|
|
|
+ dailyStatsVO.setPageViewsPerSession(
|
|
|
+ Double.parseDouble(
|
|
|
+ reportData.get(ReportConstant.METRIC_SCREEN_PAGE_VIEWS_PER_SESSION)));
|
|
|
+
|
|
|
+ dailyStatsVOs.put(dailyStatsVO.getDate(), dailyStatsVO);
|
|
|
+ }
|
|
|
+
|
|
|
+ // 4. 缺失日期补充
|
|
|
+ Set<String> absentDays =
|
|
|
+ Sets.difference(
|
|
|
+ DateUtil.getAllDaysBetween(start, end).stream()
|
|
|
+ .map(date -> DateUtil.formatDate(date, DateUtil.DATE_FORMAT))
|
|
|
+ .collect(Collectors.toSet()),
|
|
|
+ dailyStatsVOs.keySet());
|
|
|
+ absentDays.forEach(
|
|
|
+ date -> {
|
|
|
+ DailyStatsVO dailyStatsVO = new DailyStatsVO();
|
|
|
+ dailyStatsVO.setDate(date); // 其它数值字段设置为默认值
|
|
|
+ dailyStatsVOs.put(date, dailyStatsVO);
|
|
|
+ });
|
|
|
+
|
|
|
+ // 5. 询盘数据补充
|
|
|
+ List<ImmutableTriple<String, Long, Long>> enquiryDailyCounts =
|
|
|
+ adwebEnquiryMapper.getEnquiryDailyCounts(siteCode, start, end);
|
|
|
+ for (ImmutableTriple<String, Long, Long> enquiryDailyCount : enquiryDailyCounts) {
|
|
|
+ if (!dailyStatsVOs.containsKey(enquiryDailyCount.getLeft())) {
|
|
|
+ continue;
|
|
|
+ }
|
|
|
+
|
|
|
+ DailyStatsVO dailyStatsVO = dailyStatsVOs.get(enquiryDailyCount.getLeft());
|
|
|
+ dailyStatsVO.setEnquires(enquiryDailyCount.getMiddle().intValue());
|
|
|
+ dailyStatsVO.setUnreadEnquires(enquiryDailyCount.getRight().intValue());
|
|
|
+ }
|
|
|
+
|
|
|
+ // 6. 根据日期排序并返回
|
|
|
+ return dailyStatsVOs.values().stream()
|
|
|
+ .sorted(Comparator.comparing(DailyStatsVO::getDate)) // 排日期升序排序
|
|
|
+ .collect(Collectors.toList()); // 使用toList()方法返回UnmodifiableList,导致Redis类型解析异常
|
|
|
+ }
|
|
|
+
|
|
|
/** 拉取Google Analytics - Country报表 */
|
|
|
@Cacheable(
|
|
|
cacheManager = CacheConfig.TTL_CACHE_MANAGER,
|
|
|
cacheNames =
|
|
|
- "dmp:getCountryStats"
|
|
|
+ "dmp:realtime:getCountryStats"
|
|
|
+ TTLCacheManager.TTL_SPLITTER
|
|
|
+ 60 * 10) // Redis TTL为10分钟
|
|
|
public List<CountryStatsVO> getCountryStats(String siteCode, Date start, Date end) {
|
|
@@ -112,7 +224,7 @@ public class DMPReportService {
|
|
|
@Cacheable(
|
|
|
cacheManager = CacheConfig.TTL_CACHE_MANAGER,
|
|
|
cacheNames =
|
|
|
- "dmp:getSourceMediumStats"
|
|
|
+ "dmp:realtime:getSourceMediumStats"
|
|
|
+ TTLCacheManager.TTL_SPLITTER
|
|
|
+ 60 * 10) // Redis TTL为10分钟
|
|
|
public List<SourceMediumStatsVO> getSourceMediumStats(String siteCode, Date start, Date end) {
|
|
@@ -201,7 +313,7 @@ public class DMPReportService {
|
|
|
@Cacheable(
|
|
|
cacheManager = CacheConfig.TTL_CACHE_MANAGER,
|
|
|
cacheNames =
|
|
|
- "dmp:getPagePathStats"
|
|
|
+ "dmp:realtime:getPagePathStats"
|
|
|
+ TTLCacheManager.TTL_SPLITTER
|
|
|
+ 60 * 10) // Redis TTL为10分钟
|
|
|
public List<PagePathStatsVO> getPagePathStats(
|
|
@@ -273,7 +385,9 @@ public class DMPReportService {
|
|
|
@Cacheable(
|
|
|
cacheManager = CacheConfig.TTL_CACHE_MANAGER,
|
|
|
cacheNames =
|
|
|
- "dmp:getDeviceStats" + TTLCacheManager.TTL_SPLITTER + 60 * 10) // Redis TTL为10分钟
|
|
|
+ "dmp:realtime:getDeviceStats"
|
|
|
+ + TTLCacheManager.TTL_SPLITTER
|
|
|
+ + 60 * 10) // Redis TTL为10分钟
|
|
|
public List<DeviceStatsVO> getDeviceStats(String siteCode, Date start, Date end) {
|
|
|
GoogleGTM googleGTM = this.getGTMAccount(siteCode);
|
|
|
if (StringUtils.isBlank(googleGTM.getGaPropertyId())) {
|