Spring Boot中整合Jasypt 使用自定义注解+AOP实现敏感字段的加解密

Spring Boot中整合Jasypt 使用自定义注解+AOP实现敏感字段的加解密

码农世界 2024-06-08 后端 84 次浏览 0个评论

Spring Boot中整合Jasypt 使用自定义注解+AOP实现敏感字段的加解密

😄 19年之后由于某些原因断更了三年,23年重新扬帆起航,推出更多优质博文,希望大家多多支持~

🌷 古之立大事者,不惟有超世之才,亦必有坚忍不拔之志

🎐 个人CSND主页——Micro麦可乐的博客

🐥《Docker实操教程》专栏以最新的Centos版本为基础进行Docker实操教程,入门到实战

🌺《RabbitMQ》本专栏主要介绍使用JAVA开发RabbitMQ的系列教程,从基础知识到项目实战

🌸《设计模式》专栏以实际的生活场景为案例进行讲解,让大家对设计模式有一个更清晰的理解

💕《Jenkins实战》专栏主要介绍Jenkins+Docker+Git+Maven的实战教程,让你快速掌握项目CI/CD,是2024年最新的实战教程

如果文章能够给大家带来一定的帮助!欢迎关注、评论互动~

Spring Boot中整合Jasypt 使用自定义注解+AOP实现敏感字段的加解密

  • 前言
  • 开始接入
    • 步骤一:添加依赖
    • 步骤二:配置Jasypt
    • 步骤三:创建自定义注解
    • 步骤四:创建AOP切面
    • 步骤四:创建示例实体类
    • 步骤五:创建测试Controller
    • 步骤六:验证功能
    • 结语

      前言

      在博主前面一篇文章中,相信小伙伴对 Spring Boot 中整合 Jasypt 以及加解密的方法有了一定的了解,没看过的小伙伴可以访问 【Spring Boot整合Jasypt 库实现配置文件和数据库字段敏感数据的加解密】 一起探讨。

      本章节我们针对 Jasypt 来做一些升级的玩法,使用自定义注解 + AOP 来实现敏感字段的加解密。

      开始接入

      步骤一:添加依赖

      首先构建我们的 Spring Boot 项目, 引入相关依赖 Jasypt 和 Spring AOP 的依赖

      
          org.springframework.boot
          spring-boot-starter-web
      
      
          org.springframework.boot
          spring-boot-starter-aop
      
      
          com.github.ulisesbocchio
          jasypt-spring-boot-starter
          3.0.5
      
      

      步骤二:配置Jasypt

      这里博主复用了上一篇教程的配置,如果你希望更深入的了解 YML配置和各项配置的说明,可以访问

      【Spring Boot整合Jasypt 库实现配置文件和数据库字段敏感数据的加解密】

      import com.ulisesbocchio.jasyptspringboot.annotation.EnableEncryptableProperties;
      import org.jasypt.encryption.StringEncryptor;
      import org.jasypt.encryption.pbe.PooledPBEStringEncryptor;
      import org.jasypt.encryption.pbe.config.SimpleStringPBEConfig;
      import org.springframework.context.annotation.Bean;
      import org.springframework.context.annotation.Configuration;
      @Configuration
      @EnableEncryptableProperties
      public class StringEncryptorConfig {
          @Bean("jasyptStringEncryptor")
          public StringEncryptor stringEncryptor() {
              PooledPBEStringEncryptor encryptor = new PooledPBEStringEncryptor();
              SimpleStringPBEConfig config = new SimpleStringPBEConfig();
              config.setPassword("password");
              config.setAlgorithm("PBEWITHHMACSHA512ANDAES_256");
              config.setKeyObtentionIterations("1000");
              config.setPoolSize("1");
              config.setProviderName("SunJCE");
              config.setSaltGeneratorClassName("org.jasypt.salt.RandomSaltGenerator");
              config.setIvGeneratorClassName("org.jasypt.iv.RandomIvGenerator");
              config.setStringOutputType("base64");
              encryptor.setConfig(config);
              return encryptor;
          }
      }
      

      步骤三:创建自定义注解

      接下来,我们创建两个自定义注解,用于标记需要加解密的字段以及方法

      举个例子

      • 前端传递后端某些值需要加密入库 (需要方法注解是加密)
      • 后端返回前端某些值需要解密显示 (需要方法注解是解密)

      定义一个作用在字段的注解

      import java.lang.annotation.ElementType;
      import java.lang.annotation.Retention;
      import java.lang.annotation.RetentionPolicy;
      import java.lang.annotation.Target;
      @Target(ElementType.FIELD)
      @Retention(RetentionPolicy.RUNTIME)
      public @interface JasyptField {
      }
      

      定义一个作用在方法上的注解

      @Target(ElementType.METHOD)
      @Retention(RetentionPolicy.RUNTIME)
      public @interface JasyptMethod {
          String value() default "ENC"; //ENC:加密 DEC:解密
      }
      

      步骤四:创建AOP切面

      创建一个AOP切面,主要思路是找到方法上标注了 JasyptMethod 注解且定义枚举类型是加密还是解密,获取到对应参数 joinPoint.getArgs() 在进行加密或是获取返回对象解密,无论加密解密最后调用 proceed(Object[] args) 方法改变值

      需要注意处理的问题

      1、获取参数如果是字符串,直接加密字符串

      2、获取参数是对象,则通过反射获取对象字段上@JasyptField注解的字段进行加密;

      3、获取参数是集合,需要循环上一步骤操作

      4、解密返回对象 同样需要处理字符串 、对象 、集合操作

      注意看代码解释!注意看代码解释!注意看代码解释!

      import lombok.SneakyThrows;
      import lombok.extern.slf4j.Slf4j;
      import org.aspectj.lang.ProceedingJoinPoint;
      import org.aspectj.lang.annotation.Around;
      import org.aspectj.lang.annotation.Aspect;
      import org.aspectj.lang.annotation.Pointcut;
      import org.aspectj.lang.reflect.MethodSignature;
      import org.jasypt.encryption.StringEncryptor;
      import org.springframework.beans.factory.annotation.Qualifier;
      import org.springframework.stereotype.Component;
      import java.lang.reflect.Field;
      import java.util.List;
      @Aspect
      @Component
      @Slf4j
      public class JasyptAspect {
          //注入加密类
          private final StringEncryptor stringEncryptor;
          // jasyptStringEncryptor 配置类中定义的名称
          public JasyptAspect(@Qualifier("jasyptStringEncryptor") StringEncryptor stringEncryptor) {
              this.stringEncryptor = stringEncryptor;
          }
          @Pointcut("@annotation(JasyptMethod)")
          public void pointCut() {
          }
          @SneakyThrows
          @Around("pointCut())")
          public Object jasyptAround(ProceedingJoinPoint joinPoint) {
              Object proceed;
              //获取注解类
              JasyptMethod jasyptMethod = ((MethodSignature) joinPoint.getSignature()).getMethod().getAnnotation(JasyptMethod.class);
              //获取注解传递值
              String value = jasyptMethod.value();
              //获取参数
              Object[] args = joinPoint.getArgs();
              // 这里可以定义常量或枚举判断,博主就直接判断了
              if(value.equals("ENC")){
                  for(int i=0 ; i < args.length ; i++){
                      // 判断字符串还是对象
                      if(args[i] instanceof String) {
                          args[i] = stringEncryptor.encrypt(String.valueOf(args[i]));
                      }else {
                          //对象 还分集合还是单个对象
                          boolean isList = (args[i] instanceof List);
                          handlerArgs(args[i], value, isList);
                      }
                  }
                  proceed = joinPoint.proceed(args);
              }else{
                  proceed = joinPoint.proceed();
                  // 判断字符串还是对象
                  if(proceed instanceof String) {
                      proceed = stringEncryptor.decrypt(String.valueOf(proceed));
                  }else {
                      //对象 还分集合还是单个对象
                      boolean isList = (proceed instanceof List);
                      handlerArgs(proceed, value, isList);
                  }
              }
              return proceed;
          }
          /**
           * 处理对象加解密
           * @param obj 参数对象
           * @param value 加解密值
           * @param isList 是否集合
           */
          private void handlerArgs(Object obj , String value , boolean isList){
              if(isList){
                  List objs = (List)obj;
                  for(Object o : objs){
                      handlerFields(o, value);
                  }
              }else{
                  handlerFields(obj, value);
              }
          }
          /**
           * 抽取公共处理字段加解密方法
           * @param obj
           * @param value
           */
          private void handlerFields(Object obj , String value){
              Field[] fields = obj.getClass().getDeclaredFields();
              for(Field field : fields){
                  //判断是否存在注解
                  boolean hasJasyptField = field.isAnnotationPresent(JasyptField.class);
                  if (hasJasyptField) {
                      try {
                          field.setAccessible(true);
                          String plaintextValue = null;
                          plaintextValue = (String)field.get(obj);
                          String handlerValue;
                          if(value.equals("ENC")){
                              handlerValue = stringEncryptor.encrypt(plaintextValue); //处理加密
                          }
                          else{
                              handlerValue = stringEncryptor.decrypt(plaintextValue); //处理解密
                          }
                          field.set(obj, handlerValue);
                      } catch (IllegalAccessException e) {
                          throw new RuntimeException(e);
                      }
                  }
              }
          }
      }
       
      

      步骤四:创建示例实体类

      模拟一个User类,包含需要加密的字段,并使用 @JasyptField 注解标记

      import lombok.Data;
      @Data
      public class UserDto {
          @JasyptField
          private String phone;
          @JasyptField
          private String idCard;
          private int age;
      }
      

      步骤五:创建测试Controller

      创建一个 Controller ,用于处理用户请求,主要模拟保存单个对象、集合对象,以及返回单个对象、集合对象的操作

      import lombok.extern.slf4j.Slf4j;
      import org.springframework.web.bind.annotation.RequestBody;
      import org.springframework.web.bind.annotation.RequestMapping;
      import org.springframework.web.bind.annotation.RestController;
      import java.util.ArrayList;
      import java.util.List;
      @RestController
      @RequestMapping("/api")
      @Slf4j
      public class JasyptController {
          /**
           * 参数是字符串
           * @param text
           * @return
           */
          @RequestMapping("/param-text")
          @JasyptMethod
          public String isStringParam(String text){
              log.info("参数是字符串: {}" ,  text);
              return text;
          }
          /**
           * 参数是 单个对象
           * @param userDto
           * @return
           */
          @RequestMapping("/insert-user")
          @JasyptMethod
          public UserDto insertUser(@RequestBody UserDto userDto){
              log.info("参数是对象: {}" , userDto.toString());
              //TODO 操纵入库
              return userDto;
          }
          /**
           * 参数是 集合对象
           * @param userDtos
           * @return
           */
          @RequestMapping("/insert-users")
          @JasyptMethod
          public List insertUsers(@RequestBody List userDtos){
              log.info("参数是集合: {}", userDtos.toString());
              //TODO 操纵入库
              return userDtos;
          }
          
          /**
           * 返回是对象
           * @return
           */
          @RequestMapping("/get-user")
          @JasyptMethod("DEC")
          public UserDto getUser(){
              //模拟数据库取出
              UserDto userDto = new UserDto();
              userDto.setAge(10);
              userDto.setPhone("WyXyMRDDdvZEri1XcsPyMA/Pxv+f/N9ODU612IXi4HazSK5NicKK+zZJKolEz8bv");
              userDto.setIdCard("/KP3oTWB4pDRyyio54fJ+634pIS7VyVxltNACLG/gtDof4UDYTICMd+zsimbHGDJ0JwiubTLhHqMNxztyAU7zg==");
              return userDto;
          }
          /**
           * 返回是集合对象
           * @return
           */
          @RequestMapping("/get-users")
          @JasyptMethod("DEC")
          public List getUsers(){
              //模拟数据库取出
              UserDto userDto = new UserDto();
              userDto.setAge(10);
              userDto.setPhone("WyXyMRDDdvZEri1XcsPyMA/Pxv+f/N9ODU612IXi4HazSK5NicKK+zZJKolEz8bv");
              userDto.setIdCard("/KP3oTWB4pDRyyio54fJ+634pIS7VyVxltNACLG/gtDof4UDYTICMd+zsimbHGDJ0JwiubTLhHqMNxztyAU7zg==");
              UserDto userDto2 = new UserDto();
              userDto2.setAge(100);
              userDto2.setPhone("WyXyMRDDdvZEri1XcsPyMA/Pxv+f/N9ODU612IXi4HazSK5NicKK+zZJKolEz8bv");
              userDto2.setIdCard("/KP3oTWB4pDRyyio54fJ+634pIS7VyVxltNACLG/gtDof4UDYTICMd+zsimbHGDJ0JwiubTLhHqMNxztyAU7zg==");
              List userDtos = new ArrayList<>();
              userDtos.add(userDto);
              userDtos.add(userDto2);
              return userDtos;
          }
      }
      

      步骤六:验证功能

      运行 Spring Boot 应用程序,并发送请求到接口。观察请求和响应中的数据,确保密码字段已被加密

      加密参数是字符串

      Spring Boot中整合Jasypt 使用自定义注解+AOP实现敏感字段的加解密

      加密参数是对象

      Spring Boot中整合Jasypt 使用自定义注解+AOP实现敏感字段的加解密

      加密参数是集合

      Spring Boot中整合Jasypt 使用自定义注解+AOP实现敏感字段的加解密

      解密返回是对象

      Spring Boot中整合Jasypt 使用自定义注解+AOP实现敏感字段的加解密

      解密返回是集合

      Spring Boot中整合Jasypt 使用自定义注解+AOP实现敏感字段的加解密

      至此,我们所有测试均已通过,小伙伴们可以复制博主的代码进行测试,编写的代码结构如下(仅为了演示,所有类都放在一个包下

      Spring Boot中整合Jasypt 使用自定义注解+AOP实现敏感字段的加解密

      结语

      通过本文的步骤,我们成功地在Spring Boot项目中整合了Jasypt,并使用自定义注解结合AOP实现了敏感字段的自动加解密。这种方法不仅提高了代码的可读性和可维护性,还增强了数据的安全性。在实际项目中,您可以进一步扩展和优化这个示例(比如数据入库、数据查询等),以适应更多复杂的需求。

      希望本文对您有所帮助,如果您有任何疑问或建议,请随时留言讨论。如果觉得本文对你有所帮助,希望 一键三连 给博主一点点鼓励!


      Spring Boot中整合Jasypt 使用自定义注解+AOP实现敏感字段的加解密

      转载请注明来自码农世界,本文标题:《Spring Boot中整合Jasypt 使用自定义注解+AOP实现敏感字段的加解密》

      百度分享代码,如果开启HTTPS请参考李洋个人博客
      每一天,每一秒,你所做的决定都会改变你的人生!

      发表评论

      快捷回复:

      评论列表 (暂无评论,84人围观)参与讨论

      还没有评论,来说两句吧...

      Top