Pārlūkot izejas kodu

Merge branch 'master' into cpq-dev

chenlei1231 4 mēneši atpakaļ
vecāks
revīzija
ff34861f93

+ 15 - 0
jeecg-module-system/jeecg-system-biz/src/main/java/org/jeecg/modules/adweb/dmp/service/impl/GACountryReportServiceImpl.java

@@ -3,12 +3,16 @@ package org.jeecg.modules.adweb.dmp.service.impl;
 import com.baomidou.mybatisplus.extension.service.impl.ServiceImpl;
 
 import org.apache.commons.collections4.CollectionUtils;
+import org.apache.commons.lang3.StringUtils;
 import org.jeecg.modules.adweb.common.util.NumberUtil;
 import org.jeecg.modules.adweb.dmp.entity.GACountryReport;
 import org.jeecg.modules.adweb.dmp.mapper.GACountryReportMapper;
 import org.jeecg.modules.adweb.dmp.service.IGACountryReportService;
 import org.jeecg.modules.adweb.dmp.vo.report.CountryStatsVO;
+import org.jeecg.modules.redis.CacheConfig;
+import org.jeecg.modules.redis.TTLCacheManager;
 import org.springframework.beans.factory.annotation.Autowired;
+import org.springframework.cache.annotation.Cacheable;
 import org.springframework.stereotype.Service;
 
 import java.util.Collections;
@@ -25,6 +29,10 @@ public class GACountryReportServiceImpl extends ServiceImpl<GACountryReportMappe
     @Autowired private GACountryReportMapper gaCountryReportMapper;
 
     @Override
+    @Cacheable(
+            cacheManager = CacheConfig.TTL_CACHE_MANAGER,
+            cacheNames =
+                    "getCountryStats" + TTLCacheManager.TTL_SPLITTER + 60 * 30) // Redis TTL为半小时
     public List<CountryStatsVO> getCountryStats(String siteCode, Date start, Date end) {
         List<CountryStatsVO> countryStatsVOs =
                 gaCountryReportMapper.getCountryStats(siteCode, start, end);
@@ -37,6 +45,13 @@ public class GACountryReportServiceImpl extends ServiceImpl<GACountryReportMappe
 
         // 2. VO数据填充
         for (CountryStatsVO countryStatsVO : countryStatsVOs) {
+            // adweb_country表无记录时,如CountryStatsVO.country = (not set)
+            if (StringUtils.isBlank(countryStatsVO.getCountryName())
+                    || StringUtils.isBlank(countryStatsVO.getCountryCode())) {
+                countryStatsVO.setCountryName(countryStatsVO.getCountry());
+                countryStatsVO.setCountryCode(countryStatsVO.getCountry());
+            }
+
             countryStatsVO.setTotalUsersProportion(
                     NumberUtil.formatPercentage(
                             NumberUtil.safeDivide(countryStatsVO.getTotalUsers(), totalUsersSum),

+ 11 - 1
jeecg-module-system/jeecg-system-biz/src/main/java/org/jeecg/modules/adweb/dmp/service/impl/GADailyReportServiceImpl.java

@@ -16,7 +16,10 @@ import org.jeecg.modules.adweb.dmp.mapper.GADailyReportMapper;
 import org.jeecg.modules.adweb.dmp.service.IGADailyReportService;
 import org.jeecg.modules.adweb.enquiry.mapper.AdwebEnquiryMapper;
 import org.jeecg.modules.adweb.site.service.IAdwebSiteService;
+import org.jeecg.modules.redis.CacheConfig;
+import org.jeecg.modules.redis.TTLCacheManager;
 import org.springframework.beans.factory.annotation.Autowired;
+import org.springframework.cache.annotation.Cacheable;
 import org.springframework.stereotype.Service;
 
 import java.util.*;
@@ -40,6 +43,13 @@ public class GADailyReportServiceImpl extends ServiceImpl<GADailyReportMapper, G
      *
      * <p>填充每日询盘数据
      */
+    @Override
+    @Cacheable(
+            cacheManager = CacheConfig.TTL_CACHE_MANAGER,
+            cacheNames =
+                    "getDailyStatsWithinPeriod"
+                            + TTLCacheManager.TTL_SPLITTER
+                            + 60 * 30) // Redis TTL为半小时
     public List<DailyStatsVO> getDailyStatsWithinPeriod(String siteCode, Date start, Date end) {
         Map<String, DailyStatsVO> dailyStatsVOs = Maps.newHashMap();
 
@@ -102,6 +112,6 @@ public class GADailyReportServiceImpl extends ServiceImpl<GADailyReportMapper, G
         // 3. 根据日期排序并返回
         return dailyStatsVOs.values().stream()
                 .sorted(Comparator.comparing(DailyStatsVO::getDate)) // 排日期升序排序
-                .toList();
+                .collect(Collectors.toList()); // 使用toList()方法返回UnmodifiableList,导致Redis类型解析异常
     }
 }

+ 8 - 0
jeecg-module-system/jeecg-system-biz/src/main/java/org/jeecg/modules/adweb/dmp/service/impl/GAPagePathReportServiceImpl.java

@@ -14,7 +14,10 @@ import org.jeecg.modules.adweb.dmp.service.IGAPagePathReportService;
 import org.jeecg.modules.adweb.dmp.vo.report.PagePathStatsVO;
 import org.jeecg.modules.adweb.site.entity.AdwebSite;
 import org.jeecg.modules.adweb.site.service.IAdwebSiteService;
+import org.jeecg.modules.redis.CacheConfig;
+import org.jeecg.modules.redis.TTLCacheManager;
 import org.springframework.beans.factory.annotation.Autowired;
+import org.springframework.cache.annotation.Cacheable;
 import org.springframework.stereotype.Service;
 
 import java.util.Collections;
@@ -34,6 +37,11 @@ public class GAPagePathReportServiceImpl
 
     @Autowired private IAdwebSiteService adwebSiteService;
 
+    @Override
+    @Cacheable(
+            cacheManager = CacheConfig.TTL_CACHE_MANAGER,
+            cacheNames =
+                    "getPagePathStats" + TTLCacheManager.TTL_SPLITTER + 60 * 30) // Redis TTL为半小时
     public List<PagePathStatsVO> getPagePathStats(
             String siteCode, Date start, Date end, int limit) {
         List<PagePathStatsVO> pagePathStatsVOs =

+ 9 - 0
jeecg-module-system/jeecg-system-biz/src/main/java/org/jeecg/modules/adweb/dmp/service/impl/GASourceMediumReportServiceImpl.java

@@ -8,7 +8,10 @@ import org.jeecg.modules.adweb.dmp.entity.GASourceMediumReport;
 import org.jeecg.modules.adweb.dmp.mapper.GASourceMediumReportMapper;
 import org.jeecg.modules.adweb.dmp.service.IGASourceMediumReportService;
 import org.jeecg.modules.adweb.dmp.vo.report.SourceMediumStatsVO;
+import org.jeecg.modules.redis.CacheConfig;
+import org.jeecg.modules.redis.TTLCacheManager;
 import org.springframework.beans.factory.annotation.Autowired;
+import org.springframework.cache.annotation.Cacheable;
 import org.springframework.stereotype.Service;
 
 import java.util.Collections;
@@ -25,6 +28,12 @@ public class GASourceMediumReportServiceImpl
     @Autowired private GASourceMediumReportMapper gaSourceMediumReportMapper;
 
     @Override
+    @Cacheable(
+            cacheManager = CacheConfig.TTL_CACHE_MANAGER,
+            cacheNames =
+                    "getSourceMediumStats"
+                            + TTLCacheManager.TTL_SPLITTER
+                            + 60 * 30) // Redis TTL为半小时
     public List<SourceMediumStatsVO> getSourceMediumStats(String siteCode, Date start, Date end) {
         List<SourceMediumStatsVO> sourceMediumStatsVOs =
                 gaSourceMediumReportMapper.getSourceMediumStats(siteCode, start, end);

+ 1 - 1
jeecg-module-system/jeecg-system-biz/src/main/java/org/jeecg/modules/adweb/dmp/vo/report/SiteOverviewStatsVO.java

@@ -52,7 +52,7 @@ public class SiteOverviewStatsVO {
 
         @JsonIgnore private double avgSessionDuration;
 
-        @JsonIgnore private int pageViews;
+        private int pageViews;
 
         @JsonIgnore private double pageViewsPerSession;
 

+ 95 - 0
jeecg-module-system/jeecg-system-biz/src/main/java/org/jeecg/modules/redis/CacheConfig.java

@@ -0,0 +1,95 @@
+package org.jeecg.modules.redis;
+
+import static org.springframework.data.redis.serializer.RedisSerializationContext.SerializationPair;
+
+import com.fasterxml.jackson.annotation.JsonAutoDetect;
+import com.fasterxml.jackson.annotation.PropertyAccessor;
+import com.fasterxml.jackson.databind.MapperFeature;
+import com.fasterxml.jackson.databind.ObjectMapper;
+import com.fasterxml.jackson.databind.json.JsonMapper;
+import com.fasterxml.jackson.databind.jsontype.impl.LaissezFaireSubTypeValidator;
+
+import org.jeecg.common.modules.redis.config.RedisConfig;
+import org.springframework.beans.factory.annotation.Autowired;
+import org.springframework.cache.CacheManager;
+import org.springframework.cache.annotation.Cacheable;
+import org.springframework.cache.annotation.EnableCaching;
+import org.springframework.cache.interceptor.CacheAspectSupport;
+import org.springframework.context.annotation.Bean;
+import org.springframework.context.annotation.Configuration;
+import org.springframework.context.annotation.Primary;
+import org.springframework.data.redis.cache.CacheKeyPrefix;
+import org.springframework.data.redis.cache.RedisCacheConfiguration;
+import org.springframework.data.redis.cache.RedisCacheWriter;
+import org.springframework.data.redis.connection.RedisConnectionFactory;
+import org.springframework.data.redis.connection.lettuce.LettuceConnectionFactory;
+import org.springframework.data.redis.serializer.Jackson2JsonRedisSerializer;
+import org.springframework.data.redis.serializer.StringRedisSerializer;
+
+import java.time.Duration;
+
+/**
+ * Redis cache配置
+ *
+ * @author wfansh
+ */
+@Configuration
+@EnableCaching
+public class CacheConfig {
+
+    public static final String PRIMARY_CACHE_MANAGER = "primaryCacheManager";
+
+    public static final String TTL_CACHE_MANAGER = "ttlCacheManager";
+
+    private static final int DEFAULT_TTL_HOURS = 5;
+
+    /** Jeecg默认Redis cache配置 */
+    @Autowired private RedisConfig redisConfig;
+
+    /**
+     * Workaround - 重新注册Jeecg默认的{@link CacheManager},标识为Primary
+     *
+     * <p>当有多个{@link CacheManager} beans时,需标识出@Prinary
+     *
+     * <p>参考{@link RedisConfig#cacheManager(LettuceConnectionFactory)}方法
+     *
+     * <p>参考{@link CacheAspectSupport#setCacheManager(CacheManager)}方法
+     */
+    @Bean(name = PRIMARY_CACHE_MANAGER)
+    @Primary
+    public CacheManager primaryCacheManager(LettuceConnectionFactory connectionFactory) {
+        return redisConfig.cacheManager(connectionFactory);
+    }
+
+    /** 注册{@link CacheManager},支持在{@link Cacheable#cacheNames()}中设置Redis TTL */
+    @Bean(name = TTL_CACHE_MANAGER)
+    public CacheManager ttlCacheManager(RedisConnectionFactory connectionFactory) {
+        // 解决缓存对象转换异常问题
+        JsonMapper jsonMapper =
+                JsonMapper.builder()
+                        .visibility(PropertyAccessor.ALL, JsonAutoDetect.Visibility.ANY)
+                        .activateDefaultTyping(
+                                LaissezFaireSubTypeValidator.instance,
+                                ObjectMapper.DefaultTyping.NON_FINAL)
+                        .configure(MapperFeature.USE_ANNOTATIONS, false) // 忽略@JsonIgnore注解,缓存所有字段
+                        .build();
+
+        // 配置缓存默认TTL,以及键值序列化方式 - 解决乱码的问题
+        RedisCacheConfiguration cacheConfiguration =
+                RedisCacheConfiguration.defaultCacheConfig()
+                        .computePrefixWith(CacheKeyPrefix.simple())
+                        .entryTtl(Duration.ofHours(DEFAULT_TTL_HOURS))
+                        .serializeKeysWith(
+                                SerializationPair.fromSerializer(new StringRedisSerializer()))
+                        .serializeValuesWith(
+                                SerializationPair.fromSerializer(
+                                        new Jackson2JsonRedisSerializer(jsonMapper, Object.class)))
+                        .disableCachingNullValues();
+
+        // 非锁写入方式
+        RedisCacheWriter cacheWriter =
+                RedisCacheWriter.nonLockingRedisCacheWriter(connectionFactory);
+
+        return new TTLCacheManager(cacheWriter, cacheConfiguration);
+    }
+}

+ 45 - 0
jeecg-module-system/jeecg-system-biz/src/main/java/org/jeecg/modules/redis/TTLCacheManager.java

@@ -0,0 +1,45 @@
+package org.jeecg.modules.redis;
+
+import lombok.extern.slf4j.Slf4j;
+
+import org.apache.commons.lang3.StringUtils;
+import org.jeecg.modules.adweb.dmp.service.impl.GACountryReportServiceImpl;
+import org.springframework.cache.annotation.Cacheable;
+import org.springframework.data.redis.cache.*;
+
+import java.time.Duration;
+import java.util.Date;
+
+/**
+ * 支持在{@link Cacheable#cacheNames()}中设置Redis TTL,以#分隔,单位为秒
+ *
+ * <p>示例: @Cacheable(cacheManager = CacheConfig.TTL_CACHE_MANAGER, cacheNames = "test:demo#3600")
+ *
+ * <p>用法参考{@link GACountryReportServiceImpl#getCountryStats(String, Date, Date)}方法
+ *
+ * @author wfansh
+ */
+@Slf4j
+public class TTLCacheManager extends RedisCacheManager {
+
+    public static final char TTL_SPLITTER = '#';
+
+    public TTLCacheManager(
+            RedisCacheWriter cacheWriter, RedisCacheConfiguration cacheConfiguration) {
+        super(cacheWriter, cacheConfiguration);
+    }
+
+    @Override
+    protected RedisCache createRedisCache(String name, RedisCacheConfiguration cacheConfiguration) {
+        final int idx = StringUtils.lastIndexOf(name, TTL_SPLITTER);
+        if (idx > -1) {
+            long ttl = Long.parseLong(StringUtils.substring(name, idx + 1));
+            name = StringUtils.substring(name, 0, idx);
+
+            // 设置TTL
+            cacheConfiguration = cacheConfiguration.entryTtl(Duration.ofSeconds(ttl));
+        }
+
+        return super.createRedisCache(name, cacheConfiguration);
+    }
+}

+ 3 - 1
jeecg-module-system/jeecg-system-start/src/main/java/org/jeecg/JeecgSystemApplication.java

@@ -1,15 +1,17 @@
 package org.jeecg;
 
 import com.xkcoding.justauth.autoconfigure.JustAuthAutoConfiguration;
+
 import lombok.extern.slf4j.Slf4j;
+
 import org.jeecg.common.util.oConvertUtils;
 import org.springframework.boot.SpringApplication;
 import org.springframework.boot.autoconfigure.ImportAutoConfiguration;
 import org.springframework.boot.autoconfigure.SpringBootApplication;
 import org.springframework.boot.builder.SpringApplicationBuilder;
 import org.springframework.boot.web.servlet.support.SpringBootServletInitializer;
+import org.springframework.cache.annotation.EnableCaching;
 import org.springframework.context.ConfigurableApplicationContext;
-import org.springframework.context.annotation.Import;
 import org.springframework.core.env.Environment;
 
 import java.net.InetAddress;