zoukankan      html  css  js  c++  java
  • Spring Boot 实现配置文件加解密原理

    Spring Boot 配置文件加解密原理就这么简单

    背景

    接上文《失踪人口回归,mybatis-plus 3.3.2 发布》[1] ,提供了一个非常实用的功能 「数据安全保护」 功能,不仅支持数据源的配置加密,对于 spring boot 全局的 yml /properties 文件均可实现敏感信息加密功能,在一定的程度上控制开发人员流动导致敏感信息泄露。

    // 数据源敏感信息加密

    spring:
      datasource:
        url: mpw:qRhvCwF4GOqjessEB3G+a5okP+uXXr96wcucn2Pev6BfaoEMZ1gVpPPhdDmjQqoM
        password: mpw:Hzy5iliJbwDHhjLs1L0j6w==
        username: mpw:Xb+EgsyuYRXw7U7sBJjBpA==
    

    // 数据源敏感信息加密

    spring:
      redis:
        password: mpw:Hzy5iliJbwDHhjLs1L0j6w==
    

    实现原理

    我们翻开 spring boot 官方文档,翻到 4.2.6 章节 Spring Boot 不提供对加密属性值的任何内置支持,但是提供修改 Spring 环境中包含的值所必需的扩展点 EnvironmentPostProcessor 允许在应用程序之前操作环境属性值

    mybatis-plus 的实现

    public class SafetyEncryptProcessor implements EnvironmentPostProcessor {
    
     @Override
     public void postProcessEnvironment(ConfigurableEnvironment environment, SpringApplication application) {
      //命令行中获取密钥
      String mpwKey = null;
    
      // 返回全部形式的配置源(环境变量、命令行参数、配置文件 ...)
      for (PropertySource<?> ps : environment.getPropertySources()) {
       // 判断是否需要含有加密密码,没有就直接跳过
       if (ps instanceof SimpleCommandLinePropertySource) {
        SimpleCommandLinePropertySource source = (SimpleCommandLinePropertySource) ps;
        mpwKey = source.getProperty("mpw.key");
        break;
       }
      }
    
      //处理加密内容(获取到原有配置,然后解密放到新的map 里面(key是原有key))
      HashMap<String, Object> map = new HashMap<>();
      for (PropertySource<?> ps : environment.getPropertySources()) {
       if (ps instanceof OriginTrackedMapPropertySource) {
        OriginTrackedMapPropertySource source = (OriginTrackedMapPropertySource) ps;
        for (String name : source.getPropertyNames()) {
         Object value = source.getProperty(name);
         if (value instanceof String) {
          String str = (String) value;
          if (str.startsWith("mpw:")) {
           map.put(name, AES.decrypt(str.substring(4), mpwKey));
          }
         }
        }
       }
      }
      // 将解密的数据放入环境变量,并处于第一优先级上 (这里一定要注意,覆盖其他配置)
      if (!map.isEmpty()) {
       environment.getPropertySources().addFirst(new MapPropertySource("custom-encrypt", map));
      }
     }
    }
    

    如何加载生效

    resources/META-INF/spring.factories 配置 SPI

    org.springframework.boot.env.EnvironmentPostProcessor=
      com.baomidou.mybatisplus.autoconfigure.SafetyEncryptProcessor
    

    扩展

    mybatis-plus 默认是读取启动参数,可以在此处可以根据自己需求修改为更安全的根密钥存储。

    读取环境变量

    System.getProperty("mpw.key")
    

    远程加载密码服务

    // 此处思路,参考 druid ConfigFilter
    public Properties loadConfig(String filePath) {
          Properties properties = new Properties();
    
          InputStream inStream = null;
          try {
              boolean xml = false;
              if (filePath.startsWith("file://")) {
                  filePath = filePath.substring("file://".length());
                  inStream = getFileAsStream(filePath);
                  xml = filePath.endsWith(".xml");
              } else if (filePath.startsWith("http://") || filePath.startsWith("https://")) {
                  URL url = new URL(filePath);
                  inStream = url.openStream();
                  xml = url.getPath().endsWith(".xml");
              } else if (filePath.startsWith("classpath:")) {
                  String resourcePath = filePath.substring("classpath:".length());
                  inStream = Thread.currentThread().getContextClassLoader().getResourceAsStream(resourcePath);
                  // 在classpath下应该也可以配置xml文件吧?
                  xml = resourcePath.endsWith(".xml");
              } else {
                  inStream = getFileAsStream(filePath);
                  xml = filePath.endsWith(".xml");
              }
    
              if (inStream == null) {
                  LOG.error("load config file error, file : " + filePath);
                  return null;
              }
    
              if (xml) {
                  properties.loadFromXML(inStream);
              } else {
                  properties.load(inStream);
              }
    
              return properties;
          } catch (Exception ex) {
              LOG.error("load config file error, file : " + filePath, ex);
              return null;
          } finally {
              JdbcUtils.close(inStream);
          }
      }
    

    总结

    • 配置文件加解密,是通过自定义扩展 EnvironmentPostProcessor 实现
    • 若项目中没有使用最新版本 mybatis-plus ,可以参考如上自己实现,不过我推荐 jasypt-spring-boot-starter[2] ,原理类似实现了一个 EnableEncryptablePropertySourcesPostProcessor ,但是支持的加密方式更多更成熟
    • 关于 jasypt 使用可以参考源码: https://gitee.com/log4j/pig

    项目推荐: Spring Cloud 、Spring Security OAuth2的RBAC权限管理系统 欢迎关注

  • 相关阅读:
    夺命雷公狗---ECSHOP---08---商品页的拇改成星星
    夺命雷公狗---ECSHOP---07---商品价格的遍历
    夺命雷公狗---ECSHOP---06---商品倒计时的实现
    WordPress博客密码忘记的解决方法
    夺命雷公狗---js_mv思路
    LAMP前一定要关闭防火墙
    夺命雷公狗---Smarty NO:25 缓存控制技术2(完结)
    夺命雷公狗---Smarty NO:24 缓存控制技术1
    夺命雷公狗---Smarty NO:23 常用方法
    夺命雷公狗---Smarty NO:22 常量—变量
  • 原文地址:https://www.cnblogs.com/leng-leng/p/13064042.html
Copyright © 2011-2022 走看看