清理冗余的代码
							parent
							
								
									ea7ad4b1ca
								
							
						
					
					
						commit
						0cf1738c0a
					
				|  | @ -1,36 +0,0 @@ | |||
| <?xml version="1.0" encoding="UTF-8"?> | ||||
| <project xmlns="http://maven.apache.org/POM/4.0.0" | ||||
|          xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" | ||||
|          xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd"> | ||||
|     <parent> | ||||
|         <artifactId>onemall</artifactId> | ||||
|         <groupId>cn.iocoder.mall</groupId> | ||||
|         <version>1.0-SNAPSHOT</version> | ||||
|     </parent> | ||||
|     <modelVersion>4.0.0</modelVersion> | ||||
| 
 | ||||
|     <artifactId>mall-spring-boot-starter-cache</artifactId> | ||||
| 
 | ||||
|     <dependencies> | ||||
|         <dependency> | ||||
|             <groupId>org.springframework.boot</groupId> | ||||
|             <artifactId>spring-boot-starter-data-redis</artifactId> | ||||
|         </dependency> | ||||
| 
 | ||||
|         <dependency> | ||||
|             <groupId>org.redisson</groupId> | ||||
|             <artifactId>redisson</artifactId> | ||||
|             <version>3.10.6</version> | ||||
|         </dependency> | ||||
| 
 | ||||
|         <dependency> | ||||
|             <groupId>redis.clients</groupId> | ||||
|             <artifactId>jedis</artifactId> | ||||
|             <version>3.1.0</version> | ||||
|         </dependency> | ||||
| 
 | ||||
| 
 | ||||
|     </dependencies> | ||||
| 
 | ||||
| 
 | ||||
| </project> | ||||
|  | @ -1,79 +0,0 @@ | |||
| package cn.iocoder.mall.cache.config; | ||||
| 
 | ||||
| 
 | ||||
| import org.springframework.stereotype.Component; | ||||
| import org.springframework.util.StringUtils; | ||||
| import redis.clients.jedis.Jedis; | ||||
| import redis.clients.jedis.JedisSentinelPool; | ||||
| 
 | ||||
| import javax.annotation.PostConstruct; | ||||
| import javax.annotation.Resource; | ||||
| 
 | ||||
| 
 | ||||
| @Component | ||||
| public class JedisClient { | ||||
| 
 | ||||
|     @Resource | ||||
|     private static JedisSentinelPool jedisSentinelPool; | ||||
| 
 | ||||
|     public static String get(String key) { | ||||
|         Jedis jedis = null; | ||||
|         try { | ||||
|             jedis = jedisSentinelPool.getResource(); | ||||
|             return jedis.get(key); | ||||
|         } catch (Exception e) { | ||||
|         } finally { | ||||
|             if (jedis != null) { | ||||
|                 jedis.close(); | ||||
|             } | ||||
|         } | ||||
|         return ""; | ||||
|     } | ||||
| 
 | ||||
|     public static boolean set(String key, String value) { | ||||
|         Jedis jedis = null; | ||||
|         try { | ||||
|             jedis = jedisSentinelPool.getResource(); | ||||
|             String ret = jedis.set(key, value); | ||||
|             return "ok".equalsIgnoreCase(ret); | ||||
|         } catch (Exception e) { | ||||
|         } finally { | ||||
|             if (jedis != null) { | ||||
|                 jedis.close(); | ||||
|             } | ||||
|         } | ||||
|         return false; | ||||
|     } | ||||
| 
 | ||||
|     public static boolean set(String key, String value, int seconds) { | ||||
|         Jedis jedis = null; | ||||
|         try { | ||||
|             jedis = jedisSentinelPool.getResource(); | ||||
|             String ret = jedis.set(key, value); | ||||
|             jedis.expire(key, seconds); | ||||
|             return "ok".equalsIgnoreCase(ret); | ||||
|         } catch (Exception e) { | ||||
|         } finally { | ||||
|             if (jedis != null) { | ||||
|                 jedis.close(); | ||||
|             } | ||||
|         } | ||||
|         return false; | ||||
|     } | ||||
| 
 | ||||
|     public static boolean del(String key) { | ||||
|         Long removedSize = 0L; | ||||
|         Jedis jedis = null; | ||||
|         try { | ||||
|             jedis = jedisSentinelPool.getResource(); | ||||
|             removedSize = jedis.del(key); | ||||
|         } catch (Exception e) { | ||||
|         } finally { | ||||
|             if (jedis != null) { | ||||
|                 jedis.close(); | ||||
|             } | ||||
|         } | ||||
|         return removedSize > 0; | ||||
|     } | ||||
| 
 | ||||
| } | ||||
|  | @ -1,51 +0,0 @@ | |||
| package cn.iocoder.mall.cache.config; | ||||
| 
 | ||||
| 
 | ||||
| import org.redisson.Redisson; | ||||
| import org.redisson.config.Config; | ||||
| import org.redisson.config.ReadMode; | ||||
| import org.redisson.config.SentinelServersConfig; | ||||
| import org.springframework.beans.factory.annotation.Value; | ||||
| import org.springframework.context.annotation.Bean; | ||||
| import org.springframework.stereotype.Component; | ||||
| 
 | ||||
| 
 | ||||
| import javax.annotation.Resource; | ||||
| import java.util.ArrayList; | ||||
| import java.util.Arrays; | ||||
| import java.util.List; | ||||
| 
 | ||||
| 
 | ||||
| @Component | ||||
| public class RedissonClient { | ||||
| 
 | ||||
|     @Value("${spring.redis.database}") | ||||
|     private int database; | ||||
| 
 | ||||
|     @Value("${spring.redis.sentinel.master}") | ||||
|     private String master; | ||||
| 
 | ||||
|     @Value("${spring.redis.sentinel.nodes}") | ||||
|     private String nodes; | ||||
| 
 | ||||
|     /** | ||||
|      * 哨兵模式 redisson 客户端 | ||||
|      * @return | ||||
|      */ | ||||
|     @Bean | ||||
|     public org.redisson.api.RedissonClient redissonClient() { | ||||
|         Config config = new Config(); | ||||
|         List<String> nodes = Arrays.asList(this.nodes.split(",")); | ||||
|         List<String> newNodes = new ArrayList(nodes.size()); | ||||
|         nodes .forEach((index) -> newNodes.add( | ||||
|                 index.startsWith("redis://") ? index : "redis://" + index)); | ||||
| 
 | ||||
|         SentinelServersConfig serverConfig = config.useSentinelServers() | ||||
|                 .addSentinelAddress(newNodes.toArray(new String[3])) | ||||
|                 .setMasterName(this.master) | ||||
|                 .setReadMode(ReadMode.SLAVE) ; | ||||
| 
 | ||||
|         serverConfig.setDatabase(this.database); | ||||
|         return Redisson.create(config); | ||||
|     } | ||||
| } | ||||
|  | @ -1,165 +0,0 @@ | |||
| package cn.iocoder.mall.cache.config; | ||||
| 
 | ||||
| import com.fasterxml.jackson.annotation.JsonAutoDetect; | ||||
| import com.fasterxml.jackson.annotation.PropertyAccessor; | ||||
| import com.fasterxml.jackson.databind.ObjectMapper; | ||||
| import org.redisson.Redisson; | ||||
| import org.redisson.api.RedissonClient; | ||||
| import org.redisson.config.Config; | ||||
| import org.redisson.config.ReadMode; | ||||
| import org.redisson.config.SentinelServersConfig; | ||||
| import org.springframework.beans.factory.annotation.Autowired; | ||||
| import org.springframework.beans.factory.annotation.Value; | ||||
| import org.springframework.cache.CacheManager; | ||||
| import org.springframework.cache.annotation.CachingConfigurerSupport; | ||||
| import org.springframework.cache.annotation.EnableCaching; | ||||
| import org.springframework.cache.interceptor.KeyGenerator; | ||||
| import org.springframework.cache.interceptor.SimpleKeyGenerator; | ||||
| import org.springframework.context.annotation.Bean; | ||||
| import org.springframework.context.annotation.Configuration; | ||||
| import org.springframework.data.redis.cache.RedisCacheConfiguration; | ||||
| import org.springframework.data.redis.cache.RedisCacheManager; | ||||
| import org.springframework.data.redis.connection.lettuce.LettuceConnectionFactory; | ||||
| import org.springframework.data.redis.core.RedisTemplate; | ||||
| import org.springframework.data.redis.core.StringRedisTemplate; | ||||
| import org.springframework.data.redis.serializer.Jackson2JsonRedisSerializer; | ||||
| import org.springframework.data.redis.serializer.RedisSerializationContext; | ||||
| import org.springframework.data.redis.serializer.RedisSerializer; | ||||
| import org.springframework.data.redis.serializer.StringRedisSerializer; | ||||
| import redis.clients.jedis.Jedis; | ||||
| 
 | ||||
| import java.lang.reflect.Method; | ||||
| import java.time.Duration; | ||||
| import java.util.*; | ||||
| 
 | ||||
| import static com.fasterxml.jackson.databind.DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES; | ||||
| 
 | ||||
| @Configuration | ||||
| @EnableCaching | ||||
| public class SpringDataRedisConfig extends CachingConfigurerSupport { | ||||
| 
 | ||||
|     @Value("${spring.redis.database}") | ||||
|     private int database; | ||||
| 
 | ||||
|     @Value("${spring.redis.sentinel.master}") | ||||
|     private String master; | ||||
| 
 | ||||
|     @Value("${spring.redis.sentinel.nodes}") | ||||
|     private String nodes; | ||||
| 
 | ||||
|     private static RedisTemplate redisTemplate; | ||||
| 
 | ||||
|     static { | ||||
| 
 | ||||
|     } | ||||
| 
 | ||||
|     public static String get(String key) { | ||||
|         redisTemplate.opsForValue().get(key); | ||||
|         return ""; | ||||
|     } | ||||
| 
 | ||||
|     public static boolean set(String key, String value) { | ||||
|         redisTemplate.opsForValue().set(key,value); | ||||
|         return false; | ||||
|     } | ||||
| 
 | ||||
|     public static boolean set(String key, String value, int seconds) { | ||||
|         redisTemplate.opsForValue().set(key,value,seconds); | ||||
|         return false; | ||||
|     } | ||||
| 
 | ||||
| 
 | ||||
|     /** | ||||
|      * json序列化 | ||||
|      * @return | ||||
|      */ | ||||
|     @Bean | ||||
|     public RedisSerializer<Object> jackson2JsonRedisSerializer() { | ||||
|         //使用Jackson2JsonRedisSerializer来序列化和反序列化redis的value值
 | ||||
|         Jackson2JsonRedisSerializer serializer = new Jackson2JsonRedisSerializer(Object.class); | ||||
| 
 | ||||
|         ObjectMapper mapper = new ObjectMapper(); | ||||
|         mapper.setVisibility(PropertyAccessor.ALL, JsonAutoDetect.Visibility.ANY); | ||||
|         mapper.enableDefaultTyping(ObjectMapper.DefaultTyping.NON_FINAL); | ||||
|         mapper.configure(FAIL_ON_UNKNOWN_PROPERTIES, false); | ||||
|         serializer.setObjectMapper(mapper); | ||||
|         return serializer; | ||||
|     } | ||||
| 
 | ||||
|     @Bean | ||||
|     public RedisTemplate<String, Object> redisTemplate(LettuceConnectionFactory lettuceConnectionFactory) { | ||||
|         //StringRedisTemplate的构造方法中默认设置了stringSerializer
 | ||||
|         RedisTemplate<String, Object> template = new RedisTemplate<>(); | ||||
|         //set key serializer
 | ||||
|         StringRedisSerializer stringRedisSerializer = new StringRedisSerializer(); | ||||
|         template.setKeySerializer(stringRedisSerializer); | ||||
|         template.setHashKeySerializer(stringRedisSerializer); | ||||
| 
 | ||||
| 
 | ||||
|         //set value serializer
 | ||||
|         template.setDefaultSerializer(jackson2JsonRedisSerializer()); | ||||
| 
 | ||||
|         template.setConnectionFactory(lettuceConnectionFactory); | ||||
|         template.afterPropertiesSet(); | ||||
|         return template; | ||||
|     } | ||||
| 
 | ||||
|     @Bean | ||||
|     public StringRedisTemplate stringRedisTemplate(LettuceConnectionFactory lettuceConnectionFactory) { | ||||
|         StringRedisTemplate template = new StringRedisTemplate(); | ||||
|         template.setConnectionFactory(lettuceConnectionFactory); | ||||
|         return template; | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     @Bean | ||||
|     public KeyGenerator keyGenerator() { | ||||
|         return new SimpleKeyGenerator() { | ||||
| 
 | ||||
|             @Override | ||||
|             public Object generate(Object target, Method method, Object... params) { | ||||
|                 StringBuilder sb = new StringBuilder(); | ||||
|                 sb.append(target.getClass().getName()); | ||||
|                 sb.append(".").append(method.getName()); | ||||
| 
 | ||||
|                 StringBuilder paramsSb = new StringBuilder(); | ||||
|                 for (Object param : params) { | ||||
|                     // 如果不指定,默认生成包含到键值中
 | ||||
|                     if (param != null) { | ||||
|                         paramsSb.append(param.toString()); | ||||
|                     } | ||||
|                 } | ||||
| 
 | ||||
|                 if (paramsSb.length() > 0) { | ||||
|                     sb.append("_").append(paramsSb); | ||||
|                 } | ||||
|                 return sb.toString(); | ||||
|             } | ||||
| 
 | ||||
|         }; | ||||
| 
 | ||||
|     } | ||||
| 
 | ||||
|     private RedisCacheConfiguration getRedisCacheConfigurationWithTtl(Integer time, int timeType) { | ||||
| 
 | ||||
|         Duration duration = Duration.ofMillis(time); | ||||
|         if (timeType == 1){//时
 | ||||
|             duration = Duration.ofHours(time); | ||||
|         }else if (timeType == 2){//分
 | ||||
|             duration = Duration.ofMinutes(time); | ||||
|         }else if (timeType == 3){//秒
 | ||||
|             duration = Duration.ofSeconds(time); | ||||
|         } | ||||
|         RedisCacheConfiguration redisCacheConfiguration = RedisCacheConfiguration.defaultCacheConfig(); | ||||
|         redisCacheConfiguration = redisCacheConfiguration | ||||
|                 .serializeKeysWith(RedisSerializationContext.SerializationPair.fromSerializer(new StringRedisSerializer())) | ||||
|                 .serializeValuesWith(RedisSerializationContext.SerializationPair.fromSerializer(jackson2JsonRedisSerializer()) | ||||
| 
 | ||||
|         ) | ||||
|                 //超时时间
 | ||||
|         .entryTtl(duration); | ||||
| 
 | ||||
|         return redisCacheConfiguration; | ||||
|     } | ||||
| 
 | ||||
| } | ||||
|  | @ -1,17 +0,0 @@ | |||
| redis.pool.maxIdle=200 | ||||
| redis.pool.minIdle=10 | ||||
| #redis.pool.maxActive=600 | ||||
| redis.pool.maxTotal=1024 | ||||
| redis.pool.maxWaitMillis=3000 | ||||
| redis.pool.minEvictableIdleTimeMillis=300000 | ||||
| redis.pool.numTestsPerEvictionRun=1024 | ||||
| redis.pool.timeBetweenEvictionRunsMillis=30000 | ||||
| redis.pool.testOnBorrow=true | ||||
| redis.pool.testWhileIdle=true | ||||
| redis.pool.testOnReturn=true | ||||
| 
 | ||||
| #Redis 配置 | ||||
| spring.redis.sentinel.master= | ||||
| spring.redis.sentinel.nodes= | ||||
| spring.redis.database= | ||||
| 
 | ||||
|  | @ -1,47 +0,0 @@ | |||
| <?xml version="1.0" encoding="UTF-8"?> | ||||
| <project xmlns="http://maven.apache.org/POM/4.0.0" | ||||
|          xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" | ||||
|          xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd"> | ||||
|     <parent> | ||||
|         <artifactId>common</artifactId> | ||||
|         <groupId>cn.iocoder.mall</groupId> | ||||
|         <version>1.0-SNAPSHOT</version> | ||||
|     </parent> | ||||
|     <modelVersion>4.0.0</modelVersion> | ||||
| 
 | ||||
|     <artifactId>mall-spring-boot-starter-mybatis</artifactId> | ||||
| 
 | ||||
|     <dependencies> | ||||
|         <!-- 通用相关 --> | ||||
|         <dependency> | ||||
|             <groupId>cn.iocoder.mall</groupId> | ||||
|             <artifactId>common-framework</artifactId> | ||||
|         </dependency> | ||||
| 
 | ||||
|         <!-- DB 相关 --> | ||||
|         <dependency> | ||||
|             <groupId>org.mybatis</groupId> | ||||
|             <artifactId>mybatis</artifactId> | ||||
|         </dependency> | ||||
|         <dependency> | ||||
|             <groupId>org.mybatis.spring.boot</groupId> | ||||
|             <artifactId>mybatis-spring-boot-starter</artifactId> | ||||
|         </dependency> | ||||
|         <dependency> | ||||
|             <groupId>com.baomidou</groupId> | ||||
|             <artifactId>mybatis-plus-core</artifactId> | ||||
|         </dependency> | ||||
|         <dependency> | ||||
|             <groupId>com.baomidou</groupId> | ||||
|             <artifactId>mybatis-plus-boot-starter</artifactId> | ||||
|         </dependency> | ||||
| 
 | ||||
|         <!-- 工具相关 --> | ||||
|         <dependency> | ||||
|             <groupId>com.alibaba</groupId> | ||||
|             <artifactId>fastjson</artifactId> | ||||
|             <optional>true</optional> | ||||
|         </dependency> | ||||
|     </dependencies> | ||||
| 
 | ||||
| </project> | ||||
|  | @ -1,23 +0,0 @@ | |||
| package cn.iocoder.mall.mybatis.config; | ||||
| 
 | ||||
| import cn.iocoder.mall.mybatis.core.injector.CustomSqlInject; | ||||
| import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; | ||||
| import org.springframework.context.annotation.Bean; | ||||
| import com.baomidou.mybatisplus.core.injector.ISqlInjector; | ||||
| 
 | ||||
| /** | ||||
|  * @author Hccake 2020/8/3 | ||||
|  * @version 1.0 | ||||
|  */ | ||||
| public class MybatisPlusAutoConfiguration { | ||||
| 
 | ||||
|     /** | ||||
|      * 自定义方法扩展注入器 | ||||
|      * @return ISqlInjector CustomSqlInject | ||||
|      */ | ||||
|     @Bean | ||||
|     @ConditionalOnMissingBean(ISqlInjector.class) | ||||
|     public ISqlInjector sqlInjector(){ | ||||
|         return new CustomSqlInject(); | ||||
|     } | ||||
| } | ||||
|  | @ -1,45 +0,0 @@ | |||
| package cn.iocoder.mall.mybatis.core.dataobject; | ||||
| 
 | ||||
| import java.io.Serializable; | ||||
| import java.util.Date; | ||||
| 
 | ||||
| /** | ||||
|  * 基础实体对象 | ||||
|  */ | ||||
| public class BaseDO implements Serializable { | ||||
| 
 | ||||
|     /** | ||||
|      * 创建时间 | ||||
|      */ | ||||
|     private Date createTime; | ||||
|     /** | ||||
|      * 最后更新时间 | ||||
|      */ | ||||
|     private Date updateTime; | ||||
| 
 | ||||
|     @Override | ||||
|     public String toString() { | ||||
|         return "BaseDO{" + | ||||
|                 "createTime=" + createTime + | ||||
|                 ", updateTime=" + updateTime + | ||||
|                 '}'; | ||||
|     } | ||||
| 
 | ||||
|     public Date getCreateTime() { | ||||
|         return createTime; | ||||
|     } | ||||
| 
 | ||||
|     public BaseDO setCreateTime(Date createTime) { | ||||
|         this.createTime = createTime; | ||||
|         return this; | ||||
|     } | ||||
| 
 | ||||
|     public Date getUpdateTime() { | ||||
|         return updateTime; | ||||
|     } | ||||
| 
 | ||||
|     public BaseDO setUpdateTime(Date updateTime) { | ||||
|         this.updateTime = updateTime; | ||||
|         return this; | ||||
|     } | ||||
| } | ||||
|  | @ -1,35 +0,0 @@ | |||
| package cn.iocoder.mall.mybatis.core.dataobject; | ||||
| 
 | ||||
| import com.baomidou.mybatisplus.annotation.TableLogic; | ||||
| 
 | ||||
| /** | ||||
|  * extends BaseDO 扩展 delete 操作 | ||||
|  * | ||||
|  * @author Sin | ||||
|  * @time 2019-03-22 22:03 | ||||
|  */ | ||||
| public class DeletableDO extends BaseDO { | ||||
| 
 | ||||
|     /** | ||||
|      * 是否删除 | ||||
|      */ | ||||
|     @TableLogic | ||||
|     private Integer deleted; | ||||
| 
 | ||||
|     @Override | ||||
|     public String toString() { | ||||
|         return "DeletableDO{" + | ||||
|                 "deleted=" + deleted + | ||||
|                 '}'; | ||||
|     } | ||||
| 
 | ||||
|     public Integer getDeleted() { | ||||
|         return deleted; | ||||
|     } | ||||
| 
 | ||||
|     public DeletableDO setDeleted(Integer deleted) { | ||||
|         this.deleted = deleted; | ||||
|         return this; | ||||
|     } | ||||
| 
 | ||||
| } | ||||
|  | @ -1,40 +0,0 @@ | |||
| package cn.iocoder.mall.mybatis.core.enums; | ||||
| 
 | ||||
| /** | ||||
|  * @author Hccake 2020/8/3 | ||||
|  * @version 1.0 | ||||
|  */ | ||||
| public enum CustomSqlMethodEnum { | ||||
|     /** | ||||
|      * 批量插入 | ||||
|      */ | ||||
|     INSERT_BATCH("insertByBatch", | ||||
|             "批量插入数据", | ||||
|             "<script>\n" | ||||
|                     + "INSERT INTO %s %s VALUES \n" | ||||
|                     + "<foreach collection=\"collection\"  item=\"item\" separator=\",\"> %s\n </foreach>\n" | ||||
|                     + "</script>"); | ||||
| 
 | ||||
|     private final String method; | ||||
|     private final String desc; | ||||
|     private final String sql; | ||||
| 
 | ||||
|     CustomSqlMethodEnum(String method, String desc, String sql) { | ||||
|         this.method = method; | ||||
|         this.desc = desc; | ||||
|         this.sql = sql; | ||||
|     } | ||||
| 
 | ||||
|     public String getMethod() { | ||||
|         return method; | ||||
|     } | ||||
| 
 | ||||
|     public String getDesc() { | ||||
|         return desc; | ||||
|     } | ||||
| 
 | ||||
|     public String getSql() { | ||||
|         return sql; | ||||
|     } | ||||
| 
 | ||||
| } | ||||
|  | @ -1,37 +0,0 @@ | |||
| package cn.iocoder.mall.mybatis.core.enums; | ||||
| 
 | ||||
| import cn.iocoder.mall.mybatis.core.dataobject.DeletableDO; | ||||
| 
 | ||||
| /** | ||||
|  * {@link DeletableDO#getDeleted()} delete 状态 | ||||
|  * | ||||
|  * @author Sin | ||||
|  * @time 2019-03-22 21:15 | ||||
|  */ | ||||
| public enum DeletedStatusEnum { | ||||
| 
 | ||||
|     DELETED_NO(0, "正常(未删除)"), | ||||
|     DELETED_YES(1, "删除"); | ||||
| 
 | ||||
|     /** | ||||
|      * 状态值 | ||||
|      */ | ||||
|     private Integer value; | ||||
|     /** | ||||
|      * 状态名 | ||||
|      */ | ||||
|     private String name; | ||||
| 
 | ||||
|     DeletedStatusEnum(Integer value, String name) { | ||||
|         this.value = value; | ||||
|         this.name = name; | ||||
|     } | ||||
| 
 | ||||
|     public Integer getValue() { | ||||
|         return value; | ||||
|     } | ||||
| 
 | ||||
|     public String getName() { | ||||
|         return name; | ||||
|     } | ||||
| } | ||||
|  | @ -1,23 +0,0 @@ | |||
| package cn.iocoder.mall.mybatis.core.injector; | ||||
| 
 | ||||
| import cn.iocoder.mall.mybatis.core.injector.method.InsertByBatch; | ||||
| import com.baomidou.mybatisplus.core.injector.AbstractMethod; | ||||
| import com.baomidou.mybatisplus.core.injector.DefaultSqlInjector; | ||||
| 
 | ||||
| import java.util.List; | ||||
| 
 | ||||
| /** | ||||
|  * 自定义 Sql 注入器,继承 MybatisPlus 提供的默认注入器 | ||||
|  * @author Hccake 2020/8/3 | ||||
|  * @version 1.0 | ||||
|  */ | ||||
| public class CustomSqlInject extends DefaultSqlInjector { | ||||
| 
 | ||||
|     @Override | ||||
|     public List<AbstractMethod> getMethodList(Class<?> mapperClass) { | ||||
|         List<AbstractMethod> methodList = super.getMethodList(mapperClass); | ||||
|         methodList.add(new InsertByBatch()); | ||||
|         return methodList; | ||||
|     } | ||||
| 
 | ||||
| } | ||||
|  | @ -1,82 +0,0 @@ | |||
| package cn.iocoder.mall.mybatis.core.injector.method; | ||||
| 
 | ||||
| import cn.iocoder.mall.mybatis.core.enums.CustomSqlMethodEnum; | ||||
| import com.baomidou.mybatisplus.annotation.IdType; | ||||
| import com.baomidou.mybatisplus.core.injector.AbstractMethod; | ||||
| import com.baomidou.mybatisplus.core.metadata.TableFieldInfo; | ||||
| import com.baomidou.mybatisplus.core.metadata.TableInfo; | ||||
| import com.baomidou.mybatisplus.core.metadata.TableInfoHelper; | ||||
| import com.baomidou.mybatisplus.core.toolkit.StringUtils; | ||||
| import com.baomidou.mybatisplus.core.toolkit.sql.SqlScriptUtils; | ||||
| import org.apache.ibatis.executor.keygen.Jdbc3KeyGenerator; | ||||
| import org.apache.ibatis.executor.keygen.KeyGenerator; | ||||
| import org.apache.ibatis.executor.keygen.NoKeyGenerator; | ||||
| import org.apache.ibatis.mapping.MappedStatement; | ||||
| import org.apache.ibatis.mapping.SqlSource; | ||||
| 
 | ||||
| import java.util.List; | ||||
| 
 | ||||
| /** | ||||
|  * 批量插入 | ||||
|  * @author Hccake 2020/8/3 | ||||
|  * @version 1.0 | ||||
|  */ | ||||
| @SuppressWarnings("all") | ||||
| public class InsertByBatch extends AbstractMethod { | ||||
| 
 | ||||
|     private final static String ITEM = "item"; | ||||
| 
 | ||||
|     @Override | ||||
|     public MappedStatement injectMappedStatement(Class<?> mapperClass, Class<?> modelClass, TableInfo tableInfo) { | ||||
|         KeyGenerator keyGenerator = new NoKeyGenerator(); | ||||
| 
 | ||||
|         CustomSqlMethodEnum sqlMethod = CustomSqlMethodEnum.INSERT_BATCH; | ||||
| 
 | ||||
|         // ==== 拼接 sql 模板 ==============
 | ||||
|         StringBuilder columnScriptBuilder = new StringBuilder(LEFT_BRACKET); | ||||
|         StringBuilder valuesScriptBuilder = new StringBuilder(LEFT_BRACKET); | ||||
|         // 主键拼接
 | ||||
|         if (tableInfo.havePK()) { | ||||
|             columnScriptBuilder.append(tableInfo.getKeyColumn()).append(COMMA); | ||||
|             valuesScriptBuilder.append(SqlScriptUtils.safeParam(ITEM + DOT + tableInfo.getKeyProperty())).append(COMMA); | ||||
|         } | ||||
|         // 普通字段拼接
 | ||||
|         // PS:如有需要可在此实现插入字段过滤
 | ||||
|         List<TableFieldInfo> fieldList = tableInfo.getFieldList(); | ||||
|         for (TableFieldInfo fieldInfo : fieldList) { | ||||
|             columnScriptBuilder.append(fieldInfo.getColumn()).append(COMMA); | ||||
|             valuesScriptBuilder.append(SqlScriptUtils.safeParam(ITEM + DOT + fieldInfo.getProperty())).append(COMMA); | ||||
|         } | ||||
|         // 替换多余的逗号为括号
 | ||||
|         columnScriptBuilder.setCharAt(columnScriptBuilder.length() - 1, ')'); | ||||
|         valuesScriptBuilder.setCharAt(valuesScriptBuilder.length() - 1, ')'); | ||||
|         // sql 模板占位符替换
 | ||||
|         String columnScript = columnScriptBuilder.toString(); | ||||
|         String valuesScript = valuesScriptBuilder.toString(); | ||||
|         String sql = String.format(sqlMethod.getSql(), tableInfo.getTableName(), columnScript, valuesScript); | ||||
| 
 | ||||
| 
 | ||||
|         // === mybatis 主键逻辑处理:主键生成策略,以及主键回填=======
 | ||||
|         String keyColumn = null; | ||||
|         String keyProperty = null; | ||||
|         // 表包含主键处理逻辑,如果不包含主键当普通字段处理
 | ||||
|         if (StringUtils.isNotBlank(tableInfo.getKeyProperty())) { | ||||
|             if (tableInfo.getIdType() == IdType.AUTO) { | ||||
|                 /** 自增主键 */ | ||||
|                 keyGenerator = new Jdbc3KeyGenerator(); | ||||
|                 keyProperty = tableInfo.getKeyProperty(); | ||||
|                 keyColumn = tableInfo.getKeyColumn(); | ||||
|             } else { | ||||
|                 if (null != tableInfo.getKeySequence()) { | ||||
|                     keyGenerator = TableInfoHelper.genKeyGenerator(sqlMethod.getMethod(), tableInfo, builderAssistant); | ||||
|                     keyProperty = tableInfo.getKeyProperty(); | ||||
|                     keyColumn = tableInfo.getKeyColumn(); | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
| 
 | ||||
|         // 模板写入
 | ||||
|         SqlSource sqlSource = languageDriver.createSqlSource(configuration, sql, modelClass); | ||||
|         return this.addInsertMappedStatement(mapperClass, modelClass, sqlMethod.getMethod(), sqlSource, keyGenerator, keyProperty, keyColumn); | ||||
|     } | ||||
| } | ||||
|  | @ -1,21 +0,0 @@ | |||
| package cn.iocoder.mall.mybatis.core.mapper; | ||||
| 
 | ||||
| import com.baomidou.mybatisplus.core.mapper.BaseMapper; | ||||
| import org.apache.ibatis.annotations.Param; | ||||
| 
 | ||||
| import java.util.Collection; | ||||
| 
 | ||||
| /** | ||||
|  * Mapper 层基类 | ||||
|  * @author Hccake 2020/8/3 | ||||
|  * @version 1.0 | ||||
|  */ | ||||
| public interface CommonMapper<T> extends BaseMapper<T> { | ||||
| 
 | ||||
|     /** | ||||
|      * 批量插入 | ||||
|      * @param collection 批量插入数据 | ||||
|      * @return ignore | ||||
|      */ | ||||
|     int insertByBatch(@Param("collection") Collection<T> collection); | ||||
| } | ||||
|  | @ -1,93 +0,0 @@ | |||
| package cn.iocoder.mall.mybatis.core.query; | ||||
| 
 | ||||
| import com.baomidou.mybatisplus.core.conditions.query.QueryWrapper; | ||||
| import com.baomidou.mybatisplus.core.toolkit.ArrayUtils; | ||||
| import com.baomidou.mybatisplus.core.toolkit.CollectionUtils; | ||||
| import org.springframework.util.StringUtils; | ||||
| 
 | ||||
| import java.util.Collection; | ||||
| 
 | ||||
| /** | ||||
|  * 拓展 MyBatis Plus QueryWrapper 类,主要增加如下功能: | ||||
|  * | ||||
|  * 1. 拼接条件的方法,增加 xxxIfPresent 方法,用于判断值不存在的时候,不要拼接到条件中。 | ||||
|  * | ||||
|  * @param <T> 数据类型 | ||||
|  */ | ||||
| public class QueryWrapperX<T> extends QueryWrapper<T> { | ||||
| 
 | ||||
|     public QueryWrapperX<T> likeIfPresent(String column, String val) { | ||||
|         if (StringUtils.hasText(val)) { | ||||
|             return (QueryWrapperX<T>) super.like(column, val); | ||||
|         } | ||||
|         return this; | ||||
|     } | ||||
| 
 | ||||
|     public QueryWrapperX<T> inIfPresent(String column, Collection<?> values) { | ||||
|         if (!CollectionUtils.isEmpty(values)) { | ||||
|             return (QueryWrapperX<T>) super.in(column, values); | ||||
|         } | ||||
|         return this; | ||||
|     } | ||||
| 
 | ||||
|     public QueryWrapperX<T> inIfPresent(String column, Object... values) { | ||||
|         if (!ArrayUtils.isEmpty(values)) { | ||||
|             return (QueryWrapperX<T>) super.in(column, values); | ||||
|         } | ||||
|         return this; | ||||
|     } | ||||
| 
 | ||||
|     public QueryWrapperX<T> eqIfPresent(String column, Object val) { | ||||
|         if (val != null) { | ||||
|             return (QueryWrapperX<T>) super.eq(column, val); | ||||
|         } | ||||
|         return this; | ||||
|     } | ||||
| 
 | ||||
|     public QueryWrapperX<T> gtIfPresent(String column, Object val) { | ||||
|         if (val != null) { | ||||
|             return (QueryWrapperX<T>) super.gt(column, val); | ||||
|         } | ||||
|         return this; | ||||
|     } | ||||
| 
 | ||||
|     public QueryWrapperX<T> betweenIfPresent(String column, Object val1, Object val2) { | ||||
|         if (val1 != null && val2 != null) { | ||||
|             return (QueryWrapperX<T>) super.between(column, val1, val2); | ||||
|         } | ||||
|         if (val1 != null) { | ||||
|             return (QueryWrapperX<T>) ge(column, val1); | ||||
|         } | ||||
|         if (val2 != null) { | ||||
|             return (QueryWrapperX<T>) le(column, val2); | ||||
|         } | ||||
|         return this; | ||||
|     } | ||||
| 
 | ||||
|     // ========== 重写父类方法,方便链式调用 ==========
 | ||||
| 
 | ||||
|     @Override | ||||
|     public QueryWrapperX<T> eq(boolean condition, String column, Object val) { | ||||
|         super.eq(condition, column, val); | ||||
|         return this; | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public QueryWrapperX<T> eq(String column, Object val) { | ||||
|         super.eq(column, val); | ||||
|         return this; | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public QueryWrapperX<T> orderByDesc(String column) { | ||||
|         super.orderByDesc(true, column); | ||||
|         return this; | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public QueryWrapperX<T> last(String lastSql) { | ||||
|         super.last(lastSql); | ||||
|         return this; | ||||
|     } | ||||
| 
 | ||||
| } | ||||
|  | @ -1,70 +0,0 @@ | |||
| package cn.iocoder.mall.mybatis.core.type; | ||||
| 
 | ||||
| import com.alibaba.fastjson.JSON; | ||||
| import org.apache.ibatis.type.BaseTypeHandler; | ||||
| import org.apache.ibatis.type.JdbcType; | ||||
| 
 | ||||
| import java.sql.CallableStatement; | ||||
| import java.sql.PreparedStatement; | ||||
| import java.sql.ResultSet; | ||||
| import java.sql.SQLException; | ||||
| 
 | ||||
| /** | ||||
|  * TODO 芋艿 | ||||
|  * | ||||
|  * 参考 https://www.cnblogs.com/waterystone/p/5547254.html
 | ||||
|  * | ||||
|  * 后续,补充下注释和测试类,以及文章。 | ||||
|  * | ||||
|  * @param <T> | ||||
|  */ | ||||
| public class JSONTypeHandler<T extends Object> extends BaseTypeHandler<T> { | ||||
| 
 | ||||
|     private Class<T> clazz; | ||||
| 
 | ||||
|     public JSONTypeHandler(Class<T> clazz) { | ||||
|         if (clazz == null) throw new IllegalArgumentException("Type argument cannot be null"); | ||||
|         this.clazz = clazz; | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public void setNonNullParameter(PreparedStatement ps, int i, T parameter, JdbcType jdbcType) throws SQLException { | ||||
|         ps.setString(i, this.toJson(parameter)); | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public T getNullableResult(ResultSet rs, String columnName) throws SQLException { | ||||
|         return this.toObject(rs.getString(columnName), clazz); | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public T getNullableResult(ResultSet rs, int columnIndex) throws SQLException { | ||||
|         return this.toObject(rs.getString(columnIndex), clazz); | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public T getNullableResult(CallableStatement cs, int columnIndex) throws SQLException { | ||||
|         return this.toObject(cs.getString(columnIndex), clazz); | ||||
|     } | ||||
| 
 | ||||
|     private String toJson(T object) { | ||||
|         try { | ||||
|             return JSON.toJSONString(object); | ||||
|         } catch (Exception e) { | ||||
|             throw new RuntimeException(e); | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     private T toObject(String content, Class<?> clazz) { | ||||
|         if (content != null && !content.isEmpty()) { | ||||
|             try { | ||||
|                 return (T) JSON.parseObject(content, clazz); | ||||
|             } catch (Exception e) { | ||||
|                 throw new RuntimeException(e); | ||||
|             } | ||||
|         } else { | ||||
|             return null; | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
| } | ||||
|  | @ -1,35 +0,0 @@ | |||
| package cn.iocoder.mall.mybatis.core.util; | ||||
| 
 | ||||
| import cn.iocoder.common.framework.util.CollectionUtils; | ||||
| import cn.iocoder.common.framework.vo.PageParam; | ||||
| import cn.iocoder.common.framework.vo.SortingField; | ||||
| import com.baomidou.mybatisplus.core.metadata.OrderItem; | ||||
| import com.baomidou.mybatisplus.extension.plugins.pagination.Page; | ||||
| 
 | ||||
| import java.util.Collection; | ||||
| import java.util.stream.Collectors; | ||||
| 
 | ||||
| /** | ||||
|  * 分页工具里 | ||||
|  * | ||||
|  * 目前主要用于 {@link Page} 的构建 | ||||
|  */ | ||||
| public class PageUtil { | ||||
| 
 | ||||
|     public static <T> Page<T> build(PageParam pageParam) { | ||||
|         return build(pageParam, null); | ||||
|     } | ||||
| 
 | ||||
|     public static <T> Page<T> build(PageParam pageParam, Collection<SortingField> sortingFields) { | ||||
|         // 页码 + 数量
 | ||||
|         Page<T> page = new Page<>(pageParam.getPageNo(), pageParam.getPageSize()); | ||||
|         // 排序字段
 | ||||
|         if (!CollectionUtils.isEmpty(sortingFields)) { | ||||
|             page.addOrder(sortingFields.stream().map(sortingField -> SortingField.ORDER_ASC.equals(sortingField.getOrder()) ? | ||||
|                     OrderItem.asc(sortingField.getField()) : OrderItem.desc(sortingField.getField())) | ||||
|                     .collect(Collectors.toList())); | ||||
|         } | ||||
|         return page; | ||||
|     } | ||||
| 
 | ||||
| } | ||||
|  | @ -1,2 +0,0 @@ | |||
| org.springframework.boot.autoconfigure.EnableAutoConfiguration=\ | ||||
|   cn.iocoder.mall.mybatis.config.MybatisPlusAutoConfiguration | ||||
|  | @ -1,21 +0,0 @@ | |||
| <?xml version="1.0" encoding="UTF-8"?> | ||||
| <project xmlns="http://maven.apache.org/POM/4.0.0" | ||||
|          xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" | ||||
|          xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd"> | ||||
|     <parent> | ||||
|         <artifactId>common</artifactId> | ||||
|         <groupId>cn.iocoder.mall</groupId> | ||||
|         <version>1.0-SNAPSHOT</version> | ||||
|     </parent> | ||||
|     <modelVersion>4.0.0</modelVersion> | ||||
| 
 | ||||
|     <artifactId>mall-spring-boot-starter-redis</artifactId> | ||||
| 
 | ||||
|     <dependencies> | ||||
|         <dependency> | ||||
|             <groupId>org.redisson</groupId> | ||||
|             <artifactId>redisson-spring-boot-starter</artifactId> | ||||
|         </dependency> | ||||
|     </dependencies> | ||||
| 
 | ||||
| </project> | ||||
|  | @ -1,71 +0,0 @@ | |||
| package cn.iocoder.mall.redis.core; | ||||
| 
 | ||||
| import java.time.Duration; | ||||
| 
 | ||||
| /** | ||||
|  * Redis Key 定义类 | ||||
|  */ | ||||
| public class RedisKeyDefine { | ||||
| 
 | ||||
|     public enum KeyTypeEnum { | ||||
| 
 | ||||
|         STRING, | ||||
|         LIST, | ||||
|         HASH, | ||||
|         SET, | ||||
|         ZSET, | ||||
|         STREAM, | ||||
|         PUBSUB; | ||||
| 
 | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * 过期时间 - 永不过期 | ||||
|      */ | ||||
|     public static final Duration TIMEOUT_FOREVER = null; | ||||
| 
 | ||||
|     /** | ||||
|      * Key 模板 | ||||
|      */ | ||||
|     private final String keyTemplate; | ||||
|     /** | ||||
|      * Key 类型的枚举 | ||||
|      */ | ||||
|     private final KeyTypeEnum keyType; | ||||
|     /** | ||||
|      * Value 类型 | ||||
|      * | ||||
|      * 如果是使用分布式锁,设置为 {@link java.util.concurrent.locks.Lock} 类型 | ||||
|      */ | ||||
|     private final Class<?> valueType; | ||||
|     /** | ||||
|      * 过期时间 | ||||
|      * | ||||
|      * 为空时,表示永不过期 {@link #TIMEOUT_FOREVER} | ||||
|      */ | ||||
|     private final Duration timeout; | ||||
| 
 | ||||
|     public RedisKeyDefine(String keyTemplate, KeyTypeEnum keyType, Class<?> valueType, Duration timeout) { | ||||
|         this.keyTemplate = keyTemplate; | ||||
|         this.keyType = keyType; | ||||
|         this.valueType = valueType; | ||||
|         this.timeout = timeout; | ||||
|     } | ||||
| 
 | ||||
|     public String getKeyTemplate() { | ||||
|         return keyTemplate; | ||||
|     } | ||||
| 
 | ||||
|     public KeyTypeEnum getKeyType() { | ||||
|         return keyType; | ||||
|     } | ||||
| 
 | ||||
|     public Class<?> getValueType() { | ||||
|         return valueType; | ||||
|     } | ||||
| 
 | ||||
|     public Duration getTimeout() { | ||||
|         return timeout; | ||||
|     } | ||||
| 
 | ||||
| } | ||||
|  | @ -1,22 +0,0 @@ | |||
| <?xml version="1.0" encoding="UTF-8"?> | ||||
| <project xmlns="http://maven.apache.org/POM/4.0.0" | ||||
|          xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" | ||||
|          xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd"> | ||||
|     <parent> | ||||
|         <artifactId>common</artifactId> | ||||
|         <groupId>cn.iocoder.mall</groupId> | ||||
|         <version>1.0-SNAPSHOT</version> | ||||
|     </parent> | ||||
|     <modelVersion>4.0.0</modelVersion> | ||||
| 
 | ||||
|     <artifactId>mall-spring-boot-starter-rocketmq</artifactId> | ||||
| 
 | ||||
|     <dependencies> | ||||
|         <!-- MQ 相关 --> | ||||
|         <dependency> | ||||
|             <groupId>org.apache.rocketmq</groupId> | ||||
|             <artifactId>rocketmq-spring-boot-starter</artifactId> | ||||
|         </dependency> | ||||
|     </dependencies> | ||||
| 
 | ||||
| </project> | ||||
|  | @ -1,50 +0,0 @@ | |||
| <?xml version="1.0" encoding="UTF-8"?> | ||||
| <project xmlns="http://maven.apache.org/POM/4.0.0" | ||||
|          xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" | ||||
|          xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd"> | ||||
|     <parent> | ||||
|         <artifactId>common</artifactId> | ||||
|         <groupId>cn.iocoder.mall</groupId> | ||||
|         <version>1.0-SNAPSHOT</version> | ||||
|     </parent> | ||||
|     <modelVersion>4.0.0</modelVersion> | ||||
| 
 | ||||
|     <artifactId>mall-spring-boot-starter-web</artifactId> | ||||
| 
 | ||||
|     <dependencies> | ||||
|         <!-- Mall 相关 --> | ||||
|         <dependency> | ||||
|             <groupId>cn.iocoder.mall</groupId> | ||||
|             <artifactId>system-service-api</artifactId> | ||||
|             <version>1.0-SNAPSHOT</version> | ||||
|             <optional>true</optional> | ||||
|         </dependency> | ||||
| 
 | ||||
|         <!-- Spring 核心 --> | ||||
|         <dependency> | ||||
|             <groupId>org.springframework.boot</groupId> | ||||
|             <artifactId>spring-boot-configuration-processor</artifactId> | ||||
|             <optional>true</optional> | ||||
|         </dependency> | ||||
| 
 | ||||
|         <!-- Web 相关 --> | ||||
|         <dependency> | ||||
|             <groupId>org.springframework.boot</groupId> | ||||
|             <artifactId>spring-boot-starter-web</artifactId> | ||||
|         </dependency> | ||||
| 
 | ||||
|         <!-- RPC 相关 --> | ||||
|         <dependency> | ||||
|             <groupId>org.apache.dubbo</groupId> | ||||
|             <artifactId>dubbo</artifactId> | ||||
|             <optional>true</optional> | ||||
|         </dependency> | ||||
| 
 | ||||
|         <!-- 工具相关 --> | ||||
|         <dependency> | ||||
|             <groupId>com.alibaba</groupId> | ||||
|             <artifactId>fastjson</artifactId> | ||||
|         </dependency> | ||||
|     </dependencies> | ||||
| 
 | ||||
| </project> | ||||
|  | @ -1,96 +0,0 @@ | |||
| package cn.iocoder.mall.web.config; | ||||
| 
 | ||||
| import cn.iocoder.mall.web.core.handler.GlobalExceptionHandler; | ||||
| import cn.iocoder.mall.web.core.handler.GlobalResponseBodyHandler; | ||||
| import cn.iocoder.mall.web.core.interceptor.AccessLogInterceptor; | ||||
| import cn.iocoder.mall.web.core.servlet.CorsFilter; | ||||
| import com.alibaba.fastjson.serializer.SerializerFeature; | ||||
| import com.alibaba.fastjson.support.config.FastJsonConfig; | ||||
| import com.alibaba.fastjson.support.spring.FastJsonHttpMessageConverter; | ||||
| import org.slf4j.Logger; | ||||
| import org.slf4j.LoggerFactory; | ||||
| import org.springframework.beans.factory.NoSuchBeanDefinitionException; | ||||
| import org.springframework.boot.autoconfigure.condition.ConditionalOnClass; | ||||
| import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; | ||||
| import org.springframework.boot.autoconfigure.condition.ConditionalOnWebApplication; | ||||
| import org.springframework.boot.web.servlet.FilterRegistrationBean; | ||||
| import org.springframework.context.annotation.Bean; | ||||
| import org.springframework.context.annotation.Configuration; | ||||
| import org.springframework.http.MediaType; | ||||
| import org.springframework.http.converter.HttpMessageConverter; | ||||
| import org.springframework.web.servlet.config.annotation.InterceptorRegistry; | ||||
| import org.springframework.web.servlet.config.annotation.WebMvcConfigurer; | ||||
| 
 | ||||
| import java.nio.charset.Charset; | ||||
| import java.util.Collections; | ||||
| import java.util.List; | ||||
| 
 | ||||
| @Configuration | ||||
| @ConditionalOnWebApplication(type = ConditionalOnWebApplication.Type.SERVLET) | ||||
| public class CommonWebAutoConfiguration implements WebMvcConfigurer { | ||||
| 
 | ||||
|     private Logger logger = LoggerFactory.getLogger(getClass()); | ||||
| 
 | ||||
|     // ========== 全局处理器 ==========
 | ||||
| 
 | ||||
|     @Bean | ||||
|     @ConditionalOnMissingBean(GlobalResponseBodyHandler.class) | ||||
|     public GlobalResponseBodyHandler globalResponseBodyHandler() { | ||||
|         return new GlobalResponseBodyHandler(); | ||||
|     } | ||||
| 
 | ||||
|     @Bean | ||||
|     @ConditionalOnMissingBean(GlobalExceptionHandler.class) | ||||
|     public GlobalExceptionHandler globalExceptionHandler() { | ||||
|         return new GlobalExceptionHandler(); | ||||
|     } | ||||
| 
 | ||||
|     // ========== 拦截器相关 ==========
 | ||||
| 
 | ||||
|     @Bean | ||||
|     @ConditionalOnClass(name = {"cn.iocoder.mall.systemservice.rpc.systemlog.SystemExceptionLogFeign", "org.apache.dubbo.config.annotation.Reference"}) | ||||
|     @ConditionalOnMissingBean(AccessLogInterceptor.class) | ||||
|     public AccessLogInterceptor accessLogInterceptor() { | ||||
|         return new AccessLogInterceptor(); | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public void addInterceptors(InterceptorRegistry registry) { | ||||
|         try { | ||||
|             registry.addInterceptor(this.accessLogInterceptor()); | ||||
|             logger.info("[addInterceptors][加载 AccessLogInterceptor 拦截器完成]"); | ||||
|         } catch (NoSuchBeanDefinitionException e) { | ||||
|             logger.warn("[addInterceptors][无法获取 AccessLogInterceptor 拦截器,因此不启动 AccessLog 的记录]"); | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     // ========== 过滤器相关 ==========
 | ||||
| 
 | ||||
|     @Bean | ||||
|     @ConditionalOnMissingBean | ||||
|     public FilterRegistrationBean<CorsFilter> corsFilter() { | ||||
|         FilterRegistrationBean<CorsFilter> registrationBean = new FilterRegistrationBean<>(); | ||||
|         registrationBean.setFilter(new CorsFilter()); | ||||
|         registrationBean.addUrlPatterns("/*"); | ||||
|         return registrationBean; | ||||
|     } | ||||
| 
 | ||||
|     // ========== MessageConverter 相关 ==========
 | ||||
| 
 | ||||
|     @Override | ||||
|     public void configureMessageConverters(List<HttpMessageConverter<?>> converters) { | ||||
|         // 创建 FastJsonHttpMessageConverter 对象
 | ||||
|         FastJsonHttpMessageConverter fastJsonHttpMessageConverter = new FastJsonHttpMessageConverter(); | ||||
|         // 自定义 FastJson 配置
 | ||||
|         FastJsonConfig fastJsonConfig = new FastJsonConfig(); | ||||
|         fastJsonConfig.setCharset(Charset.defaultCharset()); // 设置字符集
 | ||||
|         fastJsonConfig.setSerializerFeatures(SerializerFeature.DisableCircularReferenceDetect, // 剔除循环引用
 | ||||
|                 SerializerFeature.WriteNonStringKeyAsString); // 解决 Integer 作为 Key 时,转换为 String 类型,避免浏览器报错
 | ||||
|         fastJsonHttpMessageConverter.setFastJsonConfig(fastJsonConfig); | ||||
|         // 设置支持的 MediaType
 | ||||
|         fastJsonHttpMessageConverter.setSupportedMediaTypes(Collections.singletonList(MediaType.APPLICATION_JSON)); | ||||
|         // 添加到 converters 中
 | ||||
|         converters.add(0, fastJsonHttpMessageConverter); // 注意,添加到最开头,放在 MappingJackson2XmlHttpMessageConverter 前面
 | ||||
|     } | ||||
| 
 | ||||
| } | ||||
|  | @ -1,44 +0,0 @@ | |||
| package cn.iocoder.mall.web.core.constant; | ||||
| 
 | ||||
| public interface CommonMallConstants { | ||||
| 
 | ||||
|     // 全局请求路径枚举类,用于定义不同用户类型的根请求路径
 | ||||
|     /** | ||||
|      * 根路径 - 用户 | ||||
|      */ | ||||
|     @Deprecated | ||||
|     String ROOT_PATH_USER = "/users"; | ||||
|     /** | ||||
|      * 根路径 - 管理员 | ||||
|      */ | ||||
|     @Deprecated | ||||
|     String ROOT_PATH_ADMIN = "/admins"; | ||||
| 
 | ||||
|     // HTTP Request Attr
 | ||||
|     /** | ||||
|      * HTTP Request Attr - 用户编号 | ||||
|      * | ||||
|      * 考虑到 mall-spring-boot-starter-web 不依赖 mall-spring-boot-starter-security,但是又希望拿到认证过的用户编号, | ||||
|      * 因此通过 Request 的 Attribute 进行共享 | ||||
|      */ | ||||
|     String REQUEST_ATTR_USER_ID_KEY = "mall_user_id"; | ||||
|     /** | ||||
|      * HTTP Request Attr - 用户类型 | ||||
|      * | ||||
|      * 作用同 {@link #REQUEST_ATTR_USER_ID_KEY} | ||||
|      */ | ||||
|     String REQUEST_ATTR_USER_TYPE_KEY = "mall_user_type"; | ||||
| 
 | ||||
|     /** | ||||
|      * HTTP Request Attr - Controller 执行返回 | ||||
|      * | ||||
|      * 通过该 Request 的 Attribute,获取到请求执行结果,从而在访问日志中,记录是否成功。 | ||||
|      */ | ||||
|     String REQUEST_ATTR_COMMON_RESULT = "mall_common_result"; | ||||
|     /** | ||||
|      * HTTP Request Attr - 访问开始时间 | ||||
|      */ | ||||
|     String REQUEST_ATTR_ACCESS_START_TIME = "mall_access_start_time"; | ||||
| 
 | ||||
| 
 | ||||
| } | ||||
|  | @ -1,253 +0,0 @@ | |||
| package cn.iocoder.mall.web.core.handler; | ||||
| 
 | ||||
| import cn.iocoder.common.framework.exception.GlobalException; | ||||
| import cn.iocoder.common.framework.exception.ServiceException; | ||||
| import cn.iocoder.common.framework.exception.enums.GlobalErrorCodeConstants; | ||||
| import cn.iocoder.common.framework.util.ExceptionUtil; | ||||
| import cn.iocoder.common.framework.util.HttpUtil; | ||||
| import cn.iocoder.common.framework.util.MallUtils; | ||||
| import cn.iocoder.common.framework.vo.CommonResult; | ||||
| import cn.iocoder.mall.systemservice.rpc.systemlog.SystemExceptionLogFeign; | ||||
| import cn.iocoder.mall.systemservice.rpc.systemlog.dto.SystemExceptionLogCreateDTO; | ||||
| import cn.iocoder.mall.web.core.util.CommonWebUtil; | ||||
| import com.alibaba.fastjson.JSON; | ||||
| import org.apache.commons.lang3.exception.ExceptionUtils; | ||||
| import org.slf4j.Logger; | ||||
| import org.slf4j.LoggerFactory; | ||||
| import org.springframework.beans.factory.annotation.Autowired; | ||||
| import org.springframework.beans.factory.annotation.Value; | ||||
| import org.springframework.scheduling.annotation.Async; | ||||
| import org.springframework.util.Assert; | ||||
| import org.springframework.validation.BindException; | ||||
| import org.springframework.validation.FieldError; | ||||
| import org.springframework.web.HttpRequestMethodNotSupportedException; | ||||
| import org.springframework.web.bind.MethodArgumentNotValidException; | ||||
| import org.springframework.web.bind.MissingServletRequestParameterException; | ||||
| import org.springframework.web.bind.annotation.ExceptionHandler; | ||||
| import org.springframework.web.bind.annotation.RestControllerAdvice; | ||||
| import org.springframework.web.method.annotation.MethodArgumentTypeMismatchException; | ||||
| import org.springframework.web.servlet.NoHandlerFoundException; | ||||
| 
 | ||||
| import javax.servlet.http.HttpServletRequest; | ||||
| import javax.validation.ConstraintViolation; | ||||
| import javax.validation.ConstraintViolationException; | ||||
| import javax.validation.ValidationException; | ||||
| import java.util.Date; | ||||
| 
 | ||||
| import static cn.iocoder.common.framework.exception.enums.GlobalErrorCodeConstants.BAD_REQUEST; | ||||
| import static cn.iocoder.common.framework.exception.enums.GlobalErrorCodeConstants.INTERNAL_SERVER_ERROR; | ||||
| 
 | ||||
| /** | ||||
|  * 全局异常处理器,将 Exception 翻译成 CommonResult + 对应的异常编号 | ||||
|  */ | ||||
| @RestControllerAdvice | ||||
| public class GlobalExceptionHandler { | ||||
| 
 | ||||
|     // TODO 芋艿,应该还有其它的异常,需要进行翻译
 | ||||
| 
 | ||||
| //    /**
 | ||||
| //     * 异常总数 Metrics
 | ||||
| //     */
 | ||||
| //    private static final Counter EXCEPTION_COUNTER = Metrics.counter("mall.exception.total");
 | ||||
| 
 | ||||
|     private Logger logger = LoggerFactory.getLogger(getClass()); | ||||
| 
 | ||||
|     @Value("${spring.application.name}") | ||||
|     private String applicationName; | ||||
| 
 | ||||
|     // TODO 目前存在一个问题,如果未引入 system-rpc-api 依赖,GlobalExceptionHandler 会报类不存在。未来封装出 Repository 解决该问题
 | ||||
| 
 | ||||
|     @Autowired | ||||
|     private SystemExceptionLogFeign systemExceptionLogFeign; | ||||
|     /** | ||||
|      * 处理 SpringMVC 请求参数缺失 | ||||
|      * | ||||
|      * 例如说,接口上设置了 @RequestParam("xx") 参数,结果并未传递 xx 参数 | ||||
|      */ | ||||
|     @ExceptionHandler(value = MissingServletRequestParameterException.class) | ||||
|     public CommonResult missingServletRequestParameterExceptionHandler(MissingServletRequestParameterException ex) { | ||||
|         logger.warn("[missingServletRequestParameterExceptionHandler]", ex); | ||||
|         return CommonResult.error(BAD_REQUEST.getCode(), String.format("请求参数缺失:%s", ex.getParameterName())) | ||||
|                 .setDetailMessage(ExceptionUtil.getRootCauseMessage(ex)); | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * 处理 SpringMVC 请求参数类型错误 | ||||
|      * | ||||
|      * 例如说,接口上设置了 @RequestParam("xx") 参数为 Integer,结果传递 xx 参数类型为 String | ||||
|      */ | ||||
|     @ExceptionHandler(MethodArgumentTypeMismatchException.class) | ||||
|     public CommonResult methodArgumentTypeMismatchExceptionHandler(MethodArgumentTypeMismatchException ex) { | ||||
|         logger.warn("[missingServletRequestParameterExceptionHandler]", ex); | ||||
|         return CommonResult.error(BAD_REQUEST.getCode(), String.format("请求参数类型错误:%s", ex.getMessage())) | ||||
|                 .setDetailMessage(ExceptionUtil.getRootCauseMessage(ex)); | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * 处理 SpringMVC 参数校验不正确 | ||||
|      */ | ||||
|     @ExceptionHandler(MethodArgumentNotValidException.class) | ||||
|     public CommonResult methodArgumentNotValidExceptionExceptionHandler(MethodArgumentNotValidException ex) { | ||||
|         logger.warn("[methodArgumentNotValidExceptionExceptionHandler]", ex); | ||||
|         FieldError fieldError = ex.getBindingResult().getFieldError(); | ||||
|         assert fieldError != null; // 断言,避免告警
 | ||||
|         return CommonResult.error(BAD_REQUEST.getCode(), String.format("请求参数不正确:%s", fieldError.getDefaultMessage())) | ||||
|                 .setDetailMessage(ExceptionUtil.getRootCauseMessage(ex)); | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * 处理 SpringMVC 参数绑定不正确,本质上也是通过 Validator 校验 | ||||
|      */ | ||||
|     @ExceptionHandler(BindException.class) | ||||
|     public CommonResult bindExceptionHandler(BindException ex) { | ||||
|         logger.warn("[handleBindException]", ex); | ||||
|         FieldError fieldError = ex.getFieldError(); | ||||
|         assert fieldError != null; // 断言,避免告警
 | ||||
|         return CommonResult.error(BAD_REQUEST.getCode(), String.format("请求参数不正确:%s", fieldError.getDefaultMessage())) | ||||
|                 .setDetailMessage(ExceptionUtil.getRootCauseMessage(ex)); | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * 处理 Validator 校验不通过产生的异常 | ||||
|      */ | ||||
|     @ExceptionHandler(value = ConstraintViolationException.class) | ||||
|     public CommonResult constraintViolationExceptionHandler(ConstraintViolationException ex) { | ||||
|         logger.warn("[constraintViolationExceptionHandler]", ex); | ||||
|         ConstraintViolation<?> constraintViolation = ex.getConstraintViolations().iterator().next(); | ||||
|         return CommonResult.error(BAD_REQUEST.getCode(), String.format("请求参数不正确:%s", constraintViolation.getMessage())) | ||||
|                 .setDetailMessage(ExceptionUtil.getRootCauseMessage(ex)); | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * 处理 SpringMVC 请求地址不存在 | ||||
|      * | ||||
|      * 注意,它需要设置如下两个配置项: | ||||
|      * 1. spring.mvc.throw-exception-if-no-handler-found 为 true | ||||
|      * 2. spring.mvc.static-path-pattern 为 /statics/** | ||||
|      */ | ||||
|     @ExceptionHandler(NoHandlerFoundException.class) | ||||
|     public CommonResult noHandlerFoundExceptionHandler(NoHandlerFoundException ex) { | ||||
|         logger.warn("[noHandlerFoundExceptionHandler]", ex); | ||||
|         return CommonResult.error(GlobalErrorCodeConstants.NOT_FOUND.getCode(), String.format("请求地址不存在:%s", ex.getRequestURL())) | ||||
|                 .setDetailMessage(ExceptionUtil.getRootCauseMessage(ex)); | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * 处理 SpringMVC 请求方法不正确 | ||||
|      * | ||||
|      * 例如说,A 接口的方法为 GET 方式,结果请求方法为 POST 方式,导致不匹配 | ||||
|      */ | ||||
|     @ExceptionHandler(HttpRequestMethodNotSupportedException.class) | ||||
|     public CommonResult httpRequestMethodNotSupportedExceptionHandler(HttpRequestMethodNotSupportedException ex) { | ||||
|         logger.warn("[httpRequestMethodNotSupportedExceptionHandler]", ex); | ||||
|         return CommonResult.error(GlobalErrorCodeConstants.METHOD_NOT_ALLOWED.getCode(), String.format("请求方法不正确:%s", ex.getMessage())) | ||||
|                 .setDetailMessage(ExceptionUtil.getRootCauseMessage(ex)); | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * 处理业务异常 ServiceException | ||||
|      * | ||||
|      * 例如说,商品库存不足,用户手机号已存在。 | ||||
|      */ | ||||
|     @ExceptionHandler(value = ServiceException.class) | ||||
|     public CommonResult serviceExceptionHandler(ServiceException ex) { | ||||
|         logger.info("[serviceExceptionHandler]", ex); | ||||
|         return CommonResult.error(ex.getCode(), ex.getMessage()); | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * 处理全局异常 ServiceException | ||||
|      * | ||||
|      * 例如说,Dubbo 请求超时,调用的 Dubbo 服务系统异常 | ||||
|      */ | ||||
|     @ExceptionHandler(value = GlobalException.class) | ||||
|     public CommonResult globalExceptionHandler(HttpServletRequest req, GlobalException ex) { | ||||
|         // 系统异常时,才打印异常日志
 | ||||
|         if (INTERNAL_SERVER_ERROR.getCode().equals(ex.getCode())) { | ||||
|             // 插入异常日志
 | ||||
|             this.createExceptionLog(req, ex); | ||||
|         // 普通全局异常,打印 info 日志即可
 | ||||
|         } else { | ||||
|             logger.info("[globalExceptionHandler]", ex); | ||||
|         } | ||||
|         // 返回 ERROR CommonResult
 | ||||
|         return CommonResult.error(ex); | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * 处理 Dubbo Consumer 本地参数校验时,抛出的 ValidationException 异常 | ||||
|      */ | ||||
|     @ExceptionHandler(value = ValidationException.class) | ||||
|     public CommonResult validationException(ValidationException ex) { | ||||
|         logger.warn("[constraintViolationExceptionHandler]", ex); | ||||
|         // 无法拼接明细的错误信息,因为 Dubbo Consumer 抛出 ValidationException 异常时,是直接的字符串信息,且人类不可读
 | ||||
|         return CommonResult.error(BAD_REQUEST.getCode(), "请求参数不正确") | ||||
|                 .setDetailMessage(ExceptionUtil.getRootCauseMessage(ex)); | ||||
|     } | ||||
| 
 | ||||
|     /** | ||||
|      * 处理系统异常,兜底处理所有的一切 | ||||
|      */ | ||||
|     @ExceptionHandler(value = Exception.class) | ||||
|     public CommonResult defaultExceptionHandler(HttpServletRequest req, Throwable ex) { | ||||
|         logger.error("[defaultExceptionHandler]", ex); | ||||
|         // 插入异常日志
 | ||||
|         this.createExceptionLog(req, ex); | ||||
|         // 返回 ERROR CommonResult
 | ||||
|         return CommonResult.error(INTERNAL_SERVER_ERROR.getCode(), INTERNAL_SERVER_ERROR.getMessage()) | ||||
|                 .setDetailMessage(ExceptionUtil.getRootCauseMessage(ex)); | ||||
|     } | ||||
| 
 | ||||
|     public void createExceptionLog(HttpServletRequest req, Throwable e) { | ||||
|         // 插入异常日志
 | ||||
|         SystemExceptionLogCreateDTO exceptionLog = new SystemExceptionLogCreateDTO(); | ||||
|         try { | ||||
|             // 增加异常计数 metrics TODO 暂时去掉
 | ||||
| //            EXCEPTION_COUNTER.increment();
 | ||||
|             // 初始化 exceptionLog
 | ||||
|             initExceptionLog(exceptionLog, req, e); | ||||
|             // 执行插入 exceptionLog
 | ||||
|             createExceptionLog(exceptionLog); | ||||
|         } catch (Throwable th) { | ||||
|             logger.error("[createExceptionLog][插入访问日志({}) 发生异常({})", JSON.toJSONString(exceptionLog), ExceptionUtils.getRootCauseMessage(th)); | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     // TODO 优化点:后续可以增加事件
 | ||||
|     @Async | ||||
|     public void createExceptionLog(SystemExceptionLogCreateDTO exceptionLog) { | ||||
|         try { | ||||
|             systemExceptionLogFeign.createSystemExceptionLog(exceptionLog); | ||||
|         } catch (Throwable th) { | ||||
|             logger.error("[addAccessLog][插入异常日志({}) 发生异常({})", JSON.toJSONString(exceptionLog), ExceptionUtils.getRootCauseMessage(th)); | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     private void initExceptionLog(SystemExceptionLogCreateDTO exceptionLog, HttpServletRequest request, Throwable e) { | ||||
|         // 设置账号编号
 | ||||
|         exceptionLog.setUserId(CommonWebUtil.getUserId(request)); | ||||
|         exceptionLog.setUserType(CommonWebUtil.getUserType(request)); | ||||
|         // 设置异常字段
 | ||||
|         exceptionLog.setExceptionName(e.getClass().getName()); | ||||
|         exceptionLog.setExceptionMessage(ExceptionUtil.getMessage(e)); | ||||
|         exceptionLog.setExceptionRootCauseMessage(ExceptionUtil.getRootCauseMessage(e)); | ||||
|         exceptionLog.setExceptionStackTrace(ExceptionUtil.getStackTrace(e)); | ||||
|         StackTraceElement[] stackTraceElements = e.getStackTrace(); | ||||
|         Assert.notEmpty(stackTraceElements, "异常 stackTraceElements 不能为空"); | ||||
|         StackTraceElement stackTraceElement = stackTraceElements[0]; | ||||
|         exceptionLog.setExceptionClassName(stackTraceElement.getClassName()); | ||||
|         exceptionLog.setExceptionFileName(stackTraceElement.getFileName()); | ||||
|         exceptionLog.setExceptionMethodName(stackTraceElement.getMethodName()); | ||||
|         exceptionLog.setExceptionLineNumber(stackTraceElement.getLineNumber()); | ||||
|         // 设置其它字段
 | ||||
|         exceptionLog.setTraceId(MallUtils.getTraceId()) | ||||
|                 .setApplicationName(applicationName) | ||||
|                 .setUri(request.getRequestURI()) // TODO 提升:如果想要优化,可以使用 Swagger 的 @ApiOperation 注解。
 | ||||
|                 .setQueryString(HttpUtil.buildQueryString(request)) | ||||
|                 .setMethod(request.getMethod()) | ||||
|                 .setUserAgent(HttpUtil.getUserAgent(request)) | ||||
|                 .setIp(HttpUtil.getIp(request)) | ||||
|                 .setExceptionTime(new Date()); | ||||
|     } | ||||
| 
 | ||||
| } | ||||
|  | @ -1,45 +0,0 @@ | |||
| package cn.iocoder.mall.web.core.handler; | ||||
| 
 | ||||
| import cn.iocoder.common.framework.vo.CommonResult; | ||||
| import cn.iocoder.mall.web.core.util.CommonWebUtil; | ||||
| import org.springframework.core.MethodParameter; | ||||
| import org.springframework.http.MediaType; | ||||
| import org.springframework.http.server.ServerHttpRequest; | ||||
| import org.springframework.http.server.ServerHttpResponse; | ||||
| import org.springframework.http.server.ServletServerHttpRequest; | ||||
| import org.springframework.web.bind.annotation.ControllerAdvice; | ||||
| import org.springframework.web.servlet.mvc.method.annotation.ResponseBodyAdvice; | ||||
| 
 | ||||
| /** | ||||
|  * 全局响应结果(ResponseBody)处理器 | ||||
|  * | ||||
|  * 不同于在网上看到的很多文章,会选择自动将 Controller 返回结果包上 {@link CommonResult}, | ||||
|  * 在 onemall 中,是 Controller 在返回时,主动自己包上 {@link CommonResult}。 | ||||
|  * 原因是,GlobalResponseBodyHandler 本质上是 AOP,它不应该改变 Controller 返回的数据结构 | ||||
|  * | ||||
|  * 目前,GlobalResponseBodyHandler 的主要作用是,记录 Controller 的返回结果, | ||||
|  * 方便 {@link cn.iocoder.mall.web.core.interceptor.AccessLogInterceptor} 记录访问日志 | ||||
|  */ | ||||
| @ControllerAdvice | ||||
| public class GlobalResponseBodyHandler implements ResponseBodyAdvice { | ||||
| 
 | ||||
|     @Override | ||||
|     @SuppressWarnings("NullableProblems") // 避免 IDEA 警告
 | ||||
|     public boolean supports(MethodParameter returnType, Class converterType) { | ||||
|         if (returnType.getMethod() == null) { | ||||
|             return false; | ||||
|         } | ||||
|         // 只拦截返回结果为 CommonResult 类型
 | ||||
|         return returnType.getMethod().getReturnType() == CommonResult.class; | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     @SuppressWarnings("NullableProblems") // 避免 IDEA 警告
 | ||||
|     public Object beforeBodyWrite(Object body, MethodParameter returnType, MediaType selectedContentType, Class selectedConverterType, | ||||
|                                   ServerHttpRequest request, ServerHttpResponse response) { | ||||
|         // 记录 Controller 结果
 | ||||
|         CommonWebUtil.setCommonResult(((ServletServerHttpRequest) request).getServletRequest(), (CommonResult) body); | ||||
|         return body; | ||||
|     } | ||||
| 
 | ||||
| } | ||||
|  | @ -1,90 +0,0 @@ | |||
| package cn.iocoder.mall.web.core.interceptor; | ||||
| 
 | ||||
| import cn.iocoder.common.framework.util.HttpUtil; | ||||
| import cn.iocoder.common.framework.util.MallUtils; | ||||
| import cn.iocoder.common.framework.vo.CommonResult; | ||||
| import cn.iocoder.mall.systemservice.rpc.systemlog.SystemAccessLogFeign; | ||||
| import cn.iocoder.mall.systemservice.rpc.systemlog.dto.SystemAccessLogCreateDTO; | ||||
| import cn.iocoder.mall.web.core.util.CommonWebUtil; | ||||
| import com.alibaba.fastjson.JSON; | ||||
| import org.apache.commons.lang3.exception.ExceptionUtils; | ||||
| import org.slf4j.Logger; | ||||
| import org.slf4j.LoggerFactory; | ||||
| import org.springframework.beans.factory.annotation.Autowired; | ||||
| import org.springframework.beans.factory.annotation.Value; | ||||
| import org.springframework.scheduling.annotation.Async; | ||||
| import org.springframework.web.servlet.handler.HandlerInterceptorAdapter; | ||||
| 
 | ||||
| import javax.servlet.http.HttpServletRequest; | ||||
| import javax.servlet.http.HttpServletResponse; | ||||
| import java.util.Date; | ||||
| 
 | ||||
| /** | ||||
|  * 访问日志拦截器 | ||||
|  */ | ||||
| public class AccessLogInterceptor extends HandlerInterceptorAdapter { | ||||
| 
 | ||||
|     private Logger logger = LoggerFactory.getLogger(getClass()); | ||||
| 
 | ||||
| 
 | ||||
|     @Autowired | ||||
|     private SystemAccessLogFeign systemAccessLogFeign; | ||||
|     @Value("${spring.application.name}") | ||||
|     private String applicationName; | ||||
| 
 | ||||
|     @Override | ||||
|     public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) { | ||||
|         // 记录当前时间
 | ||||
|         CommonWebUtil.setAccessStartTime(request, new Date()); | ||||
|         return true; | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) { | ||||
|         SystemAccessLogCreateDTO accessLog = new SystemAccessLogCreateDTO(); | ||||
|         try { | ||||
|             // 初始化 accessLog
 | ||||
|             initAccessLog(accessLog, request); | ||||
|             // 执行插入 accessLog
 | ||||
|             addAccessLog(accessLog); | ||||
|             // TODO 提升:暂时不考虑 ELK 的方案。而是基于 MySQL 存储。如果访问日志比较多,需要定期归档。
 | ||||
|         } catch (Throwable th) { | ||||
|             logger.error("[afterCompletion][插入访问日志({}) 发生异常({})", JSON.toJSONString(accessLog), ExceptionUtils.getRootCauseMessage(th)); | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     private void initAccessLog(SystemAccessLogCreateDTO accessLog, HttpServletRequest request) { | ||||
|         // 设置账号编号
 | ||||
|         accessLog.setUserId(CommonWebUtil.getUserId(request)); | ||||
|         accessLog.setUserType(CommonWebUtil.getUserType(request)); | ||||
|         // 设置访问结果
 | ||||
|         CommonResult result = CommonWebUtil.getCommonResult(request); | ||||
|         if (result != null) { | ||||
|             accessLog.setErrorCode(result.getCode()).setErrorMessage(result.getMessage()); | ||||
|         } else { | ||||
|             // 在访问非 onemall 系统提供的 API 时,会存在没有 CommonResult 的情况。例如说,Swagger 提供的接口
 | ||||
|             accessLog.setErrorCode(0).setErrorMessage(""); | ||||
|         } | ||||
|         // 设置其它字段
 | ||||
|         accessLog.setTraceId(MallUtils.getTraceId()) | ||||
|                 .setApplicationName(applicationName) | ||||
|                 .setUri(request.getRequestURI()) // TODO 提升:如果想要优化,可以使用 Swagger 的 @ApiOperation 注解。
 | ||||
|                 .setQueryString(HttpUtil.buildQueryString(request)) | ||||
|                 .setMethod(request.getMethod()) | ||||
|                 .setUserAgent(HttpUtil.getUserAgent(request)) | ||||
|                 .setIp(HttpUtil.getIp(request)) | ||||
|                 .setStartTime(CommonWebUtil.getAccessStartTime(request)) | ||||
|                 .setResponseTime((int) (System.currentTimeMillis() - accessLog.getStartTime().getTime())); // 默认响应时间设为 0
 | ||||
|     } | ||||
| 
 | ||||
|     // TODO 优化点:后续可以增加事件
 | ||||
|     @Async // 异步入库
 | ||||
|     public void addAccessLog(SystemAccessLogCreateDTO accessLog) { | ||||
|         try { | ||||
|             systemAccessLogFeign.createSystemAccessLog(accessLog); | ||||
|         } catch (Throwable th) { | ||||
|             logger.error("[addAccessLog][插入访问日志({}) 发生异常({})", JSON.toJSONString(accessLog), ExceptionUtils.getRootCauseMessage(th)); | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
| } | ||||
|  | @ -1,40 +0,0 @@ | |||
| package cn.iocoder.mall.web.core.servlet; | ||||
| 
 | ||||
| import javax.servlet.*; | ||||
| import javax.servlet.http.HttpServletRequest; | ||||
| import javax.servlet.http.HttpServletResponse; | ||||
| import java.io.IOException; | ||||
| 
 | ||||
| /** | ||||
|  * Cors 过滤器 | ||||
|  * | ||||
|  * 未来使用 {@link org.springframework.web.filter.CorsFilter} 替换 | ||||
|  */ | ||||
| public class CorsFilter implements Filter { | ||||
| 
 | ||||
|     @Override | ||||
|     public void init(FilterConfig filterConfig) { } | ||||
| 
 | ||||
|     @Override | ||||
|     public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException, ServletException { | ||||
|         HttpServletResponse resp = (HttpServletResponse) response; | ||||
|         resp.setHeader("Access-Control-Allow-Origin", "*"); | ||||
|         resp.setHeader("Access-Control-Allow-Methods", "*"); | ||||
|         resp.setHeader("Access-Control-Allow-Headers", "*"); | ||||
|         resp.setHeader("Access-Control-Max-Age", "1800"); | ||||
|         // For HTTP OPTIONS verb/method reply with ACCEPTED status code -- per CORS handshake
 | ||||
|         // 例如说,vue axios 请求时,会自带该逻辑的
 | ||||
|         HttpServletRequest req = (HttpServletRequest) request; | ||||
|         if (req.getMethod().equals("OPTIONS")) { | ||||
|             resp.setStatus(HttpServletResponse.SC_ACCEPTED); | ||||
|             return; | ||||
|         } | ||||
|         // 如果是其它请求方法,则继续过滤器。
 | ||||
|         chain.doFilter(request, response); | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public void destroy() { | ||||
|     } | ||||
| 
 | ||||
| } | ||||
|  | @ -1,43 +0,0 @@ | |||
| package cn.iocoder.mall.web.core.util; | ||||
| 
 | ||||
| import cn.iocoder.common.framework.vo.CommonResult; | ||||
| import cn.iocoder.mall.web.core.constant.CommonMallConstants; | ||||
| 
 | ||||
| import javax.servlet.ServletRequest; | ||||
| import java.util.Date; | ||||
| 
 | ||||
| public class CommonWebUtil { | ||||
| 
 | ||||
|     public static Integer getUserId(ServletRequest request) { | ||||
|         return (Integer) request.getAttribute(CommonMallConstants.REQUEST_ATTR_USER_ID_KEY); | ||||
|     } | ||||
| 
 | ||||
|     public static void setUserId(ServletRequest request, Integer userId) { | ||||
|         request.setAttribute(CommonMallConstants.REQUEST_ATTR_USER_ID_KEY, userId); | ||||
|     } | ||||
| 
 | ||||
|     public static Integer getUserType(ServletRequest request) { | ||||
|         return (Integer) request.getAttribute(CommonMallConstants.REQUEST_ATTR_USER_TYPE_KEY); | ||||
|     } | ||||
| 
 | ||||
|     public static void setUserType(ServletRequest request, Integer userType) { | ||||
|         request.setAttribute(CommonMallConstants.REQUEST_ATTR_USER_TYPE_KEY, userType); | ||||
|     } | ||||
| 
 | ||||
|     public static CommonResult getCommonResult(ServletRequest request) { | ||||
|         return (CommonResult) request.getAttribute(CommonMallConstants.REQUEST_ATTR_COMMON_RESULT); | ||||
|     } | ||||
| 
 | ||||
|     public static void setCommonResult(ServletRequest request, CommonResult result) { | ||||
|         request.setAttribute(CommonMallConstants.REQUEST_ATTR_COMMON_RESULT, result); | ||||
|     } | ||||
| 
 | ||||
|     public static void setAccessStartTime(ServletRequest request, Date startTime) { | ||||
|         request.setAttribute(CommonMallConstants.REQUEST_ATTR_ACCESS_START_TIME, startTime); | ||||
|     } | ||||
| 
 | ||||
|     public static Date getAccessStartTime(ServletRequest request) { | ||||
|         return (Date) request.getAttribute(CommonMallConstants.REQUEST_ATTR_ACCESS_START_TIME); | ||||
|     } | ||||
| 
 | ||||
| } | ||||
|  | @ -1,2 +0,0 @@ | |||
| org.springframework.boot.autoconfigure.EnableAutoConfiguration=\ | ||||
|   cn.iocoder.mall.web.config.CommonWebAutoConfiguration | ||||
		Loading…
	
		Reference in New Issue
	
	 YunaiV
						YunaiV