From a4c9dc1e68d79dee0e6d165090fd913cc846c697 Mon Sep 17 00:00:00 2001 From: click33 <2393584716@qq.com> Date: Fri, 21 Feb 2025 02:32:57 +0800 Subject: [PATCH] =?UTF-8?q?feat:=20=E6=96=B0=E5=A2=9E=20SaTokenPlugin=20?= =?UTF-8?q?=E6=8E=A5=E5=8F=A3=EF=BC=8C=E9=87=8D=E6=9E=84=E6=8F=92=E4=BB=B6?= =?UTF-8?q?=E4=BD=93=E7=B3=BB?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../dev33/satoken/plugin/SaTokenPlugin.java | 31 +++++++ .../satoken/plugin/SaTokenPluginLoader.java | 45 +++++++++ sa-token-demo/sa-token-demo-test/pom.xml | 9 +- sa-token-plugin/pom.xml | 2 + sa-token-plugin/sa-token-jackson/pom.xml | 41 +++++++++ .../dao}/SaJsonTemplateForJackson.java | 49 ++++++++-- .../plugin/impl/SaTokenPluginForJackson.java | 35 +++++++ .../cn.dev33.satoken.plugin.SaTokenPlugin | 1 + .../satoken/dao/SaTokenDaoRedisJackson.java | 91 +------------------ .../cn.dev33.satoken.plugin.SaTokenPlugin | 0 .../dev33/satoken/spring/SaBeanRegister.java | 22 ----- 11 files changed, 209 insertions(+), 117 deletions(-) create mode 100644 sa-token-core/src/main/java/cn/dev33/satoken/plugin/SaTokenPlugin.java create mode 100644 sa-token-core/src/main/java/cn/dev33/satoken/plugin/SaTokenPluginLoader.java create mode 100644 sa-token-plugin/sa-token-jackson/pom.xml rename {sa-token-starter/sa-token-spring-boot-autoconfig/src/main/java/cn/dev33/satoken/spring/json => sa-token-plugin/sa-token-jackson/src/main/java/cn/dev33/satoken/dao}/SaJsonTemplateForJackson.java (59%) create mode 100644 sa-token-plugin/sa-token-jackson/src/main/java/cn/dev33/satoken/plugin/impl/SaTokenPluginForJackson.java create mode 100644 sa-token-plugin/sa-token-jackson/src/main/resources/META-INF/services/cn.dev33.satoken.plugin.SaTokenPlugin create mode 100644 sa-token-plugin/sa-token-redis-jackson/src/main/resources/META-INF/services/cn.dev33.satoken.plugin.SaTokenPlugin diff --git a/sa-token-core/src/main/java/cn/dev33/satoken/plugin/SaTokenPlugin.java b/sa-token-core/src/main/java/cn/dev33/satoken/plugin/SaTokenPlugin.java new file mode 100644 index 00000000..64efc44b --- /dev/null +++ b/sa-token-core/src/main/java/cn/dev33/satoken/plugin/SaTokenPlugin.java @@ -0,0 +1,31 @@ +/* + * Copyright 2020-2099 sa-token.cc + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ +package cn.dev33.satoken.plugin; + +/** + * Sa-Token 插件总接口 + * + * @author click33 + * @since 1.41.0 + */ +public interface SaTokenPlugin { + + /** + * 安装插件 + */ + void setup(); + +} diff --git a/sa-token-core/src/main/java/cn/dev33/satoken/plugin/SaTokenPluginLoader.java b/sa-token-core/src/main/java/cn/dev33/satoken/plugin/SaTokenPluginLoader.java new file mode 100644 index 00000000..72063a76 --- /dev/null +++ b/sa-token-core/src/main/java/cn/dev33/satoken/plugin/SaTokenPluginLoader.java @@ -0,0 +1,45 @@ +/* + * Copyright 2020-2099 sa-token.cc + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ +package cn.dev33.satoken.plugin; + +import java.util.ArrayList; +import java.util.List; +import java.util.ServiceLoader; + +/** + * Sa-Token 插件加载器,管理所有插件的加载 + * + * @author click33 + * @since 1.41.0 + */ +public class SaTokenPluginLoader { + + public static List pluginList; + + /** + * 初始化插件 + */ + public static void init() { + List list = new ArrayList<>(); + ServiceLoader plugins = ServiceLoader.load(SaTokenPlugin.class); + for (SaTokenPlugin plugin : plugins) { + plugin.setup(); + list.add(plugin); + } + pluginList = list; + } + +} diff --git a/sa-token-demo/sa-token-demo-test/pom.xml b/sa-token-demo/sa-token-demo-test/pom.xml index f3983dd0..bf6e1a6c 100644 --- a/sa-token-demo/sa-token-demo-test/pom.xml +++ b/sa-token-demo/sa-token-demo-test/pom.xml @@ -53,7 +53,14 @@ sa-token-redis ${sa-token.version} --> - + + + + cn.dev33 + sa-token-jackson + ${sa-token.version} + + cn.dev33 diff --git a/sa-token-plugin/pom.xml b/sa-token-plugin/pom.xml index a3684dd3..c1a4aab6 100644 --- a/sa-token-plugin/pom.xml +++ b/sa-token-plugin/pom.xml @@ -18,6 +18,8 @@ + sa-token-jackson + sa-token-redis sa-token-redis-jackson sa-token-redis-fastjson diff --git a/sa-token-plugin/sa-token-jackson/pom.xml b/sa-token-plugin/sa-token-jackson/pom.xml new file mode 100644 index 00000000..291293ae --- /dev/null +++ b/sa-token-plugin/sa-token-jackson/pom.xml @@ -0,0 +1,41 @@ + + + 4.0.0 + + + cn.dev33 + sa-token-plugin + ${revision} + ../pom.xml + + jar + + sa-token-jackson + sa-token-jackson + sa-token-jackson + + + + + cn.dev33 + sa-token-core + + + + com.fasterxml.jackson.core + jackson-databind + true + + + + com.fasterxml.jackson.datatype + jackson-datatype-jsr310 + true + + + + + + diff --git a/sa-token-starter/sa-token-spring-boot-autoconfig/src/main/java/cn/dev33/satoken/spring/json/SaJsonTemplateForJackson.java b/sa-token-plugin/sa-token-jackson/src/main/java/cn/dev33/satoken/dao/SaJsonTemplateForJackson.java similarity index 59% rename from sa-token-starter/sa-token-spring-boot-autoconfig/src/main/java/cn/dev33/satoken/spring/json/SaJsonTemplateForJackson.java rename to sa-token-plugin/sa-token-jackson/src/main/java/cn/dev33/satoken/dao/SaJsonTemplateForJackson.java index c5394fd0..7a3125ac 100644 --- a/sa-token-starter/sa-token-spring-boot-autoconfig/src/main/java/cn/dev33/satoken/spring/json/SaJsonTemplateForJackson.java +++ b/sa-token-plugin/sa-token-jackson/src/main/java/cn/dev33/satoken/dao/SaJsonTemplateForJackson.java @@ -13,29 +13,47 @@ * See the License for the specific language governing permissions and * limitations under the License. */ -package cn.dev33.satoken.spring.json; +package cn.dev33.satoken.dao; -import cn.dev33.satoken.error.SaSpringBootErrorCode; import cn.dev33.satoken.exception.SaJsonConvertException; import cn.dev33.satoken.json.SaJsonTemplate; import cn.dev33.satoken.util.SaFoxUtil; import com.fasterxml.jackson.annotation.JsonTypeInfo; import com.fasterxml.jackson.core.JsonProcessingException; +import com.fasterxml.jackson.databind.DeserializationFeature; import com.fasterxml.jackson.databind.ObjectMapper; import com.fasterxml.jackson.databind.SerializationFeature; import com.fasterxml.jackson.databind.jsontype.BasicPolymorphicTypeValidator; import com.fasterxml.jackson.databind.jsontype.PolymorphicTypeValidator; +import com.fasterxml.jackson.datatype.jsr310.JavaTimeModule; +import com.fasterxml.jackson.datatype.jsr310.deser.LocalDateDeserializer; +import com.fasterxml.jackson.datatype.jsr310.deser.LocalDateTimeDeserializer; +import com.fasterxml.jackson.datatype.jsr310.deser.LocalTimeDeserializer; +import com.fasterxml.jackson.datatype.jsr310.ser.LocalDateSerializer; +import com.fasterxml.jackson.datatype.jsr310.ser.LocalDateTimeSerializer; +import com.fasterxml.jackson.datatype.jsr310.ser.LocalTimeSerializer; +import java.time.LocalDate; +import java.time.LocalDateTime; +import java.time.LocalTime; +import java.time.format.DateTimeFormatter; import java.util.Map; /** - * JSON 转换器, Jackson 版实现 + * JSON 转换器, Jackson 版实现 * * @author click33 * @since 1.34.0 */ public class SaJsonTemplateForJackson implements SaJsonTemplate { + public static final String DATE_TIME_PATTERN = "yyyy-MM-dd HH:mm:ss"; + public static final String DATE_PATTERN = "yyyy-MM-dd"; + public static final String TIME_PATTERN = "HH:mm:ss"; + public static final DateTimeFormatter DATE_TIME_FORMATTER = DateTimeFormatter.ofPattern(DATE_TIME_PATTERN); + public static final DateTimeFormatter DATE_FORMATTER = DateTimeFormatter.ofPattern(DATE_PATTERN); + public static final DateTimeFormatter TIME_FORMATTER = DateTimeFormatter.ofPattern(TIME_PATTERN); + /** * 底层 Mapper 对象 */ @@ -63,6 +81,25 @@ public class SaJsonTemplateForJackson implements SaJsonTemplate { // 2、使空 bean 在序列化时也能记录类型信息,而不是只序列化成 {} objectMapper.configure(SerializationFeature.FAIL_ON_EMPTY_BEANS, false); + // 3、配置 [ 忽略未知字段 ] + this.objectMapper.configure(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES, false); + + // 4、配置 [ 时间类型转换 ] + JavaTimeModule timeModule = new JavaTimeModule(); + // LocalDateTime序列化与反序列化 + timeModule.addSerializer(new LocalDateTimeSerializer(DATE_TIME_FORMATTER)); + timeModule.addDeserializer(LocalDateTime.class, new LocalDateTimeDeserializer(DATE_TIME_FORMATTER)); + // LocalDate序列化与反序列化 + timeModule.addSerializer(new LocalDateSerializer(DATE_FORMATTER)); + timeModule.addDeserializer(LocalDate.class, new LocalDateDeserializer(DATE_FORMATTER)); + // LocalTime序列化与反序列化 + timeModule.addSerializer(new LocalTimeSerializer(TIME_FORMATTER)); + timeModule.addDeserializer(LocalTime.class, new LocalTimeDeserializer(TIME_FORMATTER)); + // + this.objectMapper.registerModule(timeModule); + + + } /** @@ -73,7 +110,7 @@ public class SaJsonTemplateForJackson implements SaJsonTemplate { try { return objectMapper.writeValueAsString(obj); } catch (JsonProcessingException e) { - throw new SaJsonConvertException(e).setCode(SaSpringBootErrorCode.CODE_20103); + throw new SaJsonConvertException(e); } } @@ -89,7 +126,7 @@ public class SaJsonTemplateForJackson implements SaJsonTemplate { Object value = objectMapper.readValue(jsonStr, Object.class); return value; } catch (JsonProcessingException e) { - throw new SaJsonConvertException(e).setCode(SaSpringBootErrorCode.CODE_20106); + throw new SaJsonConvertException(e); } } @@ -106,7 +143,7 @@ public class SaJsonTemplateForJackson implements SaJsonTemplate { Map map = objectMapper.readValue(jsonStr, Map.class); return map; } catch (JsonProcessingException e) { - throw new SaJsonConvertException(e).setCode(SaSpringBootErrorCode.CODE_20104); + throw new SaJsonConvertException(e); } } diff --git a/sa-token-plugin/sa-token-jackson/src/main/java/cn/dev33/satoken/plugin/impl/SaTokenPluginForJackson.java b/sa-token-plugin/sa-token-jackson/src/main/java/cn/dev33/satoken/plugin/impl/SaTokenPluginForJackson.java new file mode 100644 index 00000000..a189e8d3 --- /dev/null +++ b/sa-token-plugin/sa-token-jackson/src/main/java/cn/dev33/satoken/plugin/impl/SaTokenPluginForJackson.java @@ -0,0 +1,35 @@ +/* + * Copyright 2020-2099 sa-token.cc + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ +package cn.dev33.satoken.plugin.impl; + +import cn.dev33.satoken.SaManager; +import cn.dev33.satoken.dao.SaJsonTemplateForJackson; +import cn.dev33.satoken.plugin.SaTokenPlugin; + +/** + * SaToken 插件:JSON 转换器安装 + * + * @author click33 + * @since 1.41.0 + */ +public class SaTokenPluginForJackson implements SaTokenPlugin { + + @Override + public void setup() { + SaManager.setSaJsonTemplate(new SaJsonTemplateForJackson()); + } + +} \ No newline at end of file diff --git a/sa-token-plugin/sa-token-jackson/src/main/resources/META-INF/services/cn.dev33.satoken.plugin.SaTokenPlugin b/sa-token-plugin/sa-token-jackson/src/main/resources/META-INF/services/cn.dev33.satoken.plugin.SaTokenPlugin new file mode 100644 index 00000000..488ec78a --- /dev/null +++ b/sa-token-plugin/sa-token-jackson/src/main/resources/META-INF/services/cn.dev33.satoken.plugin.SaTokenPlugin @@ -0,0 +1 @@ +cn.dev33.satoken.plugin.impl.SaTokenPluginForJackson \ No newline at end of file diff --git a/sa-token-plugin/sa-token-redis-jackson/src/main/java/cn/dev33/satoken/dao/SaTokenDaoRedisJackson.java b/sa-token-plugin/sa-token-redis-jackson/src/main/java/cn/dev33/satoken/dao/SaTokenDaoRedisJackson.java index 60084954..642ce65c 100644 --- a/sa-token-plugin/sa-token-redis-jackson/src/main/java/cn/dev33/satoken/dao/SaTokenDaoRedisJackson.java +++ b/sa-token-plugin/sa-token-redis-jackson/src/main/java/cn/dev33/satoken/dao/SaTokenDaoRedisJackson.java @@ -17,28 +17,11 @@ package cn.dev33.satoken.dao; import cn.dev33.satoken.strategy.SaStrategy; import cn.dev33.satoken.util.SaFoxUtil; -import com.fasterxml.jackson.databind.DeserializationFeature; -import com.fasterxml.jackson.databind.ObjectMapper; -import com.fasterxml.jackson.datatype.jsr310.JavaTimeModule; -import com.fasterxml.jackson.datatype.jsr310.deser.LocalDateDeserializer; -import com.fasterxml.jackson.datatype.jsr310.deser.LocalDateTimeDeserializer; -import com.fasterxml.jackson.datatype.jsr310.deser.LocalTimeDeserializer; -import com.fasterxml.jackson.datatype.jsr310.ser.LocalDateSerializer; -import com.fasterxml.jackson.datatype.jsr310.ser.LocalDateTimeSerializer; -import com.fasterxml.jackson.datatype.jsr310.ser.LocalTimeSerializer; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.data.redis.connection.RedisConnectionFactory; -import org.springframework.data.redis.core.RedisTemplate; import org.springframework.data.redis.core.StringRedisTemplate; -import org.springframework.data.redis.serializer.GenericJackson2JsonRedisSerializer; -import org.springframework.data.redis.serializer.StringRedisSerializer; import org.springframework.stereotype.Component; -import java.lang.reflect.Field; -import java.time.LocalDate; -import java.time.LocalDateTime; -import java.time.LocalTime; -import java.time.format.DateTimeFormatter; import java.util.ArrayList; import java.util.List; import java.util.Set; @@ -53,35 +36,11 @@ import java.util.concurrent.TimeUnit; @Component public class SaTokenDaoRedisJackson implements SaTokenDao { - public static final String DATE_TIME_PATTERN = "yyyy-MM-dd HH:mm:ss"; - public static final String DATE_PATTERN = "yyyy-MM-dd"; - public static final String TIME_PATTERN = "HH:mm:ss"; - public static final DateTimeFormatter DATE_TIME_FORMATTER = DateTimeFormatter.ofPattern(DATE_TIME_PATTERN); - public static final DateTimeFormatter DATE_FORMATTER = DateTimeFormatter.ofPattern(DATE_PATTERN); - public static final DateTimeFormatter TIME_FORMATTER = DateTimeFormatter.ofPattern(TIME_PATTERN); - - /** - * ObjectMapper 对象 (以 public 作用域暴露出此对象,方便开发者二次更改配置) - * - *

例如: - *

-	 *      SaTokenDaoRedisJackson redisJackson = (SaTokenDaoRedisJackson) SaManager.getSaTokenDao();
-	 *      redisJackson.objectMapper.xxx = xxx;
-	 * 	
- *

- */ - public ObjectMapper objectMapper; - /** * String 读写专用 */ public StringRedisTemplate stringRedisTemplate; - /** - * Object 读写专用 - */ - public RedisTemplate objectRedisTemplate; - /** * 标记:是否已初始化成功 */ @@ -93,59 +52,15 @@ public class SaTokenDaoRedisJackson implements SaTokenDao { if(this.isInit) { return; } - - // 指定相应的序列化方案 - StringRedisSerializer keySerializer = new StringRedisSerializer(); - GenericJackson2JsonRedisSerializer valueSerializer = new GenericJackson2JsonRedisSerializer(); - // 通过反射获取Mapper对象, 增加一些配置, 增强兼容性 - try { - Field field = GenericJackson2JsonRedisSerializer.class.getDeclaredField("mapper"); - field.setAccessible(true); - this.objectMapper = (ObjectMapper) field.get(valueSerializer); - - // 配置[忽略未知字段] - this.objectMapper.configure(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES, false); - - // 配置[时间类型转换] - JavaTimeModule timeModule = new JavaTimeModule(); - - // LocalDateTime序列化与反序列化 - timeModule.addSerializer(new LocalDateTimeSerializer(DATE_TIME_FORMATTER)); - timeModule.addDeserializer(LocalDateTime.class, new LocalDateTimeDeserializer(DATE_TIME_FORMATTER)); - - // LocalDate序列化与反序列化 - timeModule.addSerializer(new LocalDateSerializer(DATE_FORMATTER)); - timeModule.addDeserializer(LocalDate.class, new LocalDateDeserializer(DATE_FORMATTER)); - - // LocalTime序列化与反序列化 - timeModule.addSerializer(new LocalTimeSerializer(TIME_FORMATTER)); - timeModule.addDeserializer(LocalTime.class, new LocalTimeDeserializer(TIME_FORMATTER)); - - this.objectMapper.registerModule(timeModule); - - // 重写 SaSession 生成策略 - SaStrategy.instance.createSession = (sessionId) -> new SaSessionForJacksonCustomized(sessionId); - } catch (Exception e) { - System.err.println(e.getMessage()); - } // 构建StringRedisTemplate StringRedisTemplate stringTemplate = new StringRedisTemplate(); stringTemplate.setConnectionFactory(connectionFactory); stringTemplate.afterPropertiesSet(); - - // 构建RedisTemplate - RedisTemplate template = new RedisTemplate<>(); - template.setConnectionFactory(connectionFactory); - template.setKeySerializer(keySerializer); - template.setHashKeySerializer(keySerializer); - template.setValueSerializer(valueSerializer); - template.setHashValueSerializer(valueSerializer); - template.afterPropertiesSet(); - - // 开始初始化相关组件 this.stringRedisTemplate = stringTemplate; - this.objectRedisTemplate = template; + + // 重写 SaSession 生成策略 + SaStrategy.instance.createSession = (sessionId) -> new SaSessionForJacksonCustomized(sessionId); // 打上标记,表示已经初始化成功,后续无需再重新初始化 this.isInit = true; diff --git a/sa-token-plugin/sa-token-redis-jackson/src/main/resources/META-INF/services/cn.dev33.satoken.plugin.SaTokenPlugin b/sa-token-plugin/sa-token-redis-jackson/src/main/resources/META-INF/services/cn.dev33.satoken.plugin.SaTokenPlugin new file mode 100644 index 00000000..e69de29b diff --git a/sa-token-starter/sa-token-spring-boot-autoconfig/src/main/java/cn/dev33/satoken/spring/SaBeanRegister.java b/sa-token-starter/sa-token-spring-boot-autoconfig/src/main/java/cn/dev33/satoken/spring/SaBeanRegister.java index 4a91bb59..ab8e1e91 100644 --- a/sa-token-starter/sa-token-spring-boot-autoconfig/src/main/java/cn/dev33/satoken/spring/SaBeanRegister.java +++ b/sa-token-starter/sa-token-spring-boot-autoconfig/src/main/java/cn/dev33/satoken/spring/SaBeanRegister.java @@ -16,10 +16,7 @@ package cn.dev33.satoken.spring; import cn.dev33.satoken.config.SaTokenConfig; -import cn.dev33.satoken.json.SaJsonTemplate; -import cn.dev33.satoken.json.SaJsonTemplateDefaultImpl; import cn.dev33.satoken.spring.context.path.ApplicationContextPathLoading; -import cn.dev33.satoken.spring.json.SaJsonTemplateForJackson; import org.springframework.boot.context.properties.ConfigurationProperties; import org.springframework.context.annotation.Bean; @@ -41,25 +38,6 @@ public class SaBeanRegister { public SaTokenConfig getSaTokenConfig() { return new SaTokenConfig(); } - - /** - * 获取 json 转换器 Bean (Jackson版) - * - * @return json 转换器 Bean (Jackson版) - */ - @Bean - public SaJsonTemplate getSaJsonTemplateForJackson() { - try { - // 部分开发者会在 springboot 项目中排除 jackson 依赖,所以这里做一个判断: - // 1、如果项目中存在 jackson 依赖,则使用 jackson 的 json 转换器 - // 2、如果项目中不存在 jackson 依赖,则使用默认的 json 转换器 - // to:防止因为 jackson 依赖问题导致项目无法启动 - Class.forName("com.fasterxml.jackson.databind.ObjectMapper"); - return new SaJsonTemplateForJackson(); - } catch (ClassNotFoundException e) { - return new SaJsonTemplateDefaultImpl(); - } - } /** * 应用上下文路径加载器