mirror of
				https://github.com/dromara/RuoYi-Vue-Plus.git
				synced 2025-11-04 08:13:44 +08:00 
			
		
		
		
	!451 响应加密功能
* update 优化调整加解密判断逻辑, 避免 NPE ; * rollback 回滚错误提交, 保留加密组件开关 ; * add 新增注解 @ApiEncrypt 用于校验接口加解密 ; * add 新增 EncryptResponseBodyWrapper 加密响应参数包装类 ;
This commit is contained in:
		@@ -16,6 +16,7 @@ import org.dromara.common.core.domain.model.LoginBody;
 | 
			
		||||
import org.dromara.common.core.domain.model.RegisterBody;
 | 
			
		||||
import org.dromara.common.core.domain.model.SocialLoginBody;
 | 
			
		||||
import org.dromara.common.core.utils.*;
 | 
			
		||||
import org.dromara.common.encrypt.annotation.ApiEncrypt;
 | 
			
		||||
import org.dromara.common.json.utils.JsonUtils;
 | 
			
		||||
import org.dromara.common.satoken.utils.LoginHelper;
 | 
			
		||||
import org.dromara.common.social.config.properties.SocialLoginConfigProperties;
 | 
			
		||||
@@ -73,6 +74,7 @@ public class AuthController {
 | 
			
		||||
     * @param body 登录信息
 | 
			
		||||
     * @return 结果
 | 
			
		||||
     */
 | 
			
		||||
    @ApiEncrypt(response = false)
 | 
			
		||||
    @PostMapping("/login")
 | 
			
		||||
    public R<LoginVo> login(@Validated @RequestBody String body) {
 | 
			
		||||
        LoginBody loginBody = JsonUtils.parseObject(body, LoginBody.class);
 | 
			
		||||
 
 | 
			
		||||
@@ -171,8 +171,11 @@ api-decrypt:
 | 
			
		||||
  enabled: true
 | 
			
		||||
  # AES 加密头标识
 | 
			
		||||
  headerFlag: encrypt-key
 | 
			
		||||
  # 公私钥 非对称算法的公私钥 如:SM2,RSA 使用者请自行更换
 | 
			
		||||
  publicKey: MFwwDQYJKoZIhvcNAQEBBQADSwAwSAJBAKoR8mX0rGKLqzcWmOzbfj64K8ZIgOdHnzkXSOVOZbFu/TJhZ7rFAN+eaGkl3C4buccQd/EjEsj9ir7ijT7h96MCAwEAAQ==
 | 
			
		||||
  # 响应加密公钥 非对称算法的公私钥 如:SM2,RSA 使用者请自行更换
 | 
			
		||||
  # 对应前端解密私钥 MIIBVAIBADANBgkqhkiG9w0BAQEFAASCAT4wggE6AgEAAkEAmc3CuPiGL/LcIIm7zryCEIbl1SPzBkr75E2VMtxegyZ1lYRD+7TZGAPkvIsBcaMs6Nsy0L78n2qh+lIZMpLH8wIDAQABAkEAk82Mhz0tlv6IVCyIcw/s3f0E+WLmtPFyR9/WtV3Y5aaejUkU60JpX4m5xNR2VaqOLTZAYjW8Wy0aXr3zYIhhQQIhAMfqR9oFdYw1J9SsNc+CrhugAvKTi0+BF6VoL6psWhvbAiEAxPPNTmrkmrXwdm/pQQu3UOQmc2vCZ5tiKpW10CgJi8kCIFGkL6utxw93Ncj4exE/gPLvKcT+1Emnoox+O9kRXss5AiAMtYLJDaLEzPrAWcZeeSgSIzbL+ecokmFKSDDcRske6QIgSMkHedwND1olF8vlKsJUGK3BcdtM8w4Xq7BpSBwsloE=
 | 
			
		||||
  publicKey: MFwwDQYJKoZIhvcNAQEBBQADSwAwSAJBAJnNwrj4hi/y3CCJu868ghCG5dUj8wZK++RNlTLcXoMmdZWEQ/u02RgD5LyLAXGjLOjbMtC+/J9qofpSGTKSx/MCAwEAAQ==
 | 
			
		||||
  # 请求解密私钥 非对称算法的公私钥 如:SM2,RSA 使用者请自行更换
 | 
			
		||||
  # 对应前端加密公钥 MFwwDQYJKoZIhvcNAQEBBQADSwAwSAJBAKoR8mX0rGKLqzcWmOzbfj64K8ZIgOdHnzkXSOVOZbFu/TJhZ7rFAN+eaGkl3C4buccQd/EjEsj9ir7ijT7h96MCAwEAAQ==
 | 
			
		||||
  privateKey: MIIBVAIBADANBgkqhkiG9w0BAQEFAASCAT4wggE6AgEAAkEAqhHyZfSsYourNxaY7Nt+PrgrxkiA50efORdI5U5lsW79MmFnusUA355oaSXcLhu5xxB38SMSyP2KvuKNPuH3owIDAQABAkAfoiLyL+Z4lf4Myxk6xUDgLaWGximj20CUf+5BKKnlrK+Ed8gAkM0HqoTt2UZwA5E2MzS4EI2gjfQhz5X28uqxAiEA3wNFxfrCZlSZHb0gn2zDpWowcSxQAgiCstxGUoOqlW8CIQDDOerGKH5OmCJ4Z21v+F25WaHYPxCFMvwxpcw99EcvDQIgIdhDTIqD2jfYjPTY8Jj3EDGPbH2HHuffvflECt3Ek60CIQCFRlCkHpi7hthhYhovyloRYsM+IS9h/0BzlEAuO0ktMQIgSPT3aFAgJYwKpqRYKlLDVcflZFCKY7u3UP8iWi1Qw0Y=
 | 
			
		||||
 | 
			
		||||
springdoc:
 | 
			
		||||
 
 | 
			
		||||
@@ -37,6 +37,18 @@
 | 
			
		||||
            <artifactId>hutool-crypto</artifactId>
 | 
			
		||||
        </dependency>
 | 
			
		||||
 | 
			
		||||
        <!-- SpringBoot Web容器 -->
 | 
			
		||||
        <dependency>
 | 
			
		||||
            <groupId>org.springframework.boot</groupId>
 | 
			
		||||
            <artifactId>spring-boot-starter-web</artifactId>
 | 
			
		||||
            <exclusions>
 | 
			
		||||
                <exclusion>
 | 
			
		||||
                    <artifactId>spring-boot-starter-tomcat</artifactId>
 | 
			
		||||
                    <groupId>org.springframework.boot</groupId>
 | 
			
		||||
                </exclusion>
 | 
			
		||||
            </exclusions>
 | 
			
		||||
        </dependency>
 | 
			
		||||
 | 
			
		||||
    </dependencies>
 | 
			
		||||
 | 
			
		||||
</project>
 | 
			
		||||
 
 | 
			
		||||
@@ -0,0 +1,20 @@
 | 
			
		||||
package org.dromara.common.encrypt.annotation;
 | 
			
		||||
 | 
			
		||||
import java.lang.annotation.*;
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * 强制加密注解
 | 
			
		||||
 *
 | 
			
		||||
 * @author Michelle.Chung
 | 
			
		||||
 */
 | 
			
		||||
@Documented
 | 
			
		||||
@Target({ElementType.METHOD})
 | 
			
		||||
@Retention(RetentionPolicy.RUNTIME)
 | 
			
		||||
public @interface ApiEncrypt {
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * 响应加密忽略,默认加密,为 false 时不加密
 | 
			
		||||
     */
 | 
			
		||||
    boolean response() default true;
 | 
			
		||||
 | 
			
		||||
}
 | 
			
		||||
@@ -3,10 +3,19 @@ package org.dromara.common.encrypt.filter;
 | 
			
		||||
import cn.hutool.core.util.ObjectUtil;
 | 
			
		||||
import jakarta.servlet.*;
 | 
			
		||||
import jakarta.servlet.http.HttpServletRequest;
 | 
			
		||||
import jakarta.servlet.http.HttpServletResponse;
 | 
			
		||||
import org.dromara.common.core.constant.HttpStatus;
 | 
			
		||||
import org.dromara.common.core.exception.ServiceException;
 | 
			
		||||
import org.dromara.common.core.utils.SpringUtils;
 | 
			
		||||
import org.dromara.common.core.utils.StringUtils;
 | 
			
		||||
import org.dromara.common.encrypt.annotation.ApiEncrypt;
 | 
			
		||||
import org.dromara.common.encrypt.properties.ApiDecryptProperties;
 | 
			
		||||
import org.springframework.http.HttpMethod;
 | 
			
		||||
import org.springframework.http.MediaType;
 | 
			
		||||
import org.springframework.web.method.HandlerMethod;
 | 
			
		||||
import org.springframework.web.servlet.HandlerExceptionResolver;
 | 
			
		||||
import org.springframework.web.servlet.HandlerExecutionChain;
 | 
			
		||||
import org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerMapping;
 | 
			
		||||
 | 
			
		||||
import java.io.IOException;
 | 
			
		||||
 | 
			
		||||
@@ -25,8 +34,14 @@ public class CryptoFilter implements Filter {
 | 
			
		||||
 | 
			
		||||
    @Override
 | 
			
		||||
    public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException, ServletException {
 | 
			
		||||
        ServletRequest requestWrapper = null;
 | 
			
		||||
        HttpServletRequest servletRequest = (HttpServletRequest) request;
 | 
			
		||||
        HttpServletResponse servletResponse = (HttpServletResponse) response;
 | 
			
		||||
 | 
			
		||||
        boolean encryptFlag = false;
 | 
			
		||||
        ServletRequest requestWrapper = null;
 | 
			
		||||
        ServletResponse responseWrapper = null;
 | 
			
		||||
        EncryptResponseBodyWrapper responseBodyWrapper = null;
 | 
			
		||||
 | 
			
		||||
        // 是否为 json 请求
 | 
			
		||||
        if (StringUtils.startsWithIgnoreCase(request.getContentType(), MediaType.APPLICATION_JSON_VALUE)) {
 | 
			
		||||
            // 是否为 put 或者 post 请求
 | 
			
		||||
@@ -34,16 +49,68 @@ public class CryptoFilter implements Filter {
 | 
			
		||||
                // 是否存在加密标头
 | 
			
		||||
                String headerValue = servletRequest.getHeader(properties.getHeaderFlag());
 | 
			
		||||
                if (StringUtils.isNotBlank(headerValue)) {
 | 
			
		||||
                    requestWrapper = new DecryptRequestBodyWrapper(servletRequest, properties.getPublicKey(), properties.getPrivateKey(), properties.getHeaderFlag());
 | 
			
		||||
                    // 请求解密
 | 
			
		||||
                    requestWrapper = new DecryptRequestBodyWrapper(servletRequest, properties.getPrivateKey(), properties.getHeaderFlag());
 | 
			
		||||
                    // 获取加密注解
 | 
			
		||||
                    ApiEncrypt apiEncrypt = this.getApiEncryptAnnotation(servletRequest);
 | 
			
		||||
                    if (ObjectUtil.isNotNull(apiEncrypt)) {
 | 
			
		||||
                        // 响应加密标志
 | 
			
		||||
                        encryptFlag = apiEncrypt.response();
 | 
			
		||||
                    } else {
 | 
			
		||||
                        // 是否有注解,有就报错,没有放行
 | 
			
		||||
                        HandlerExceptionResolver exceptionResolver = SpringUtils.getBean("handlerExceptionResolver");
 | 
			
		||||
                        exceptionResolver.resolveException(
 | 
			
		||||
                            servletRequest, servletResponse, null,
 | 
			
		||||
                            new ServiceException("没有访问权限,请联系管理员授权", HttpStatus.FORBIDDEN));
 | 
			
		||||
                    }
 | 
			
		||||
                }
 | 
			
		||||
                // 判断是否响应加密
 | 
			
		||||
                if (encryptFlag) {
 | 
			
		||||
                    responseBodyWrapper = new EncryptResponseBodyWrapper(servletResponse);
 | 
			
		||||
                    responseWrapper = responseBodyWrapper;
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        chain.doFilter(ObjectUtil.defaultIfNull(requestWrapper, request), response);
 | 
			
		||||
        chain.doFilter(
 | 
			
		||||
            ObjectUtil.defaultIfNull(requestWrapper, request),
 | 
			
		||||
            ObjectUtil.defaultIfNull(responseWrapper, response));
 | 
			
		||||
 | 
			
		||||
        if (encryptFlag) {
 | 
			
		||||
            servletResponse.reset();
 | 
			
		||||
            // 对原始内容加密
 | 
			
		||||
            String encryptContent = responseBodyWrapper.getEncryptContent(
 | 
			
		||||
                servletResponse, properties.getPublicKey(), properties.getHeaderFlag());
 | 
			
		||||
            // 对加密后的内容写出
 | 
			
		||||
            servletResponse.getWriter().write(encryptContent);
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * 获取 ApiEncrypt 注解
 | 
			
		||||
     */
 | 
			
		||||
    private ApiEncrypt getApiEncryptAnnotation(HttpServletRequest servletRequest) {
 | 
			
		||||
        RequestMappingHandlerMapping handlerMapping = SpringUtils.getBean("requestMappingHandlerMapping", RequestMappingHandlerMapping.class);
 | 
			
		||||
        // 获取注解
 | 
			
		||||
        try {
 | 
			
		||||
            HandlerExecutionChain mappingHandler = handlerMapping.getHandler(servletRequest);
 | 
			
		||||
            System.out.println(mappingHandler);
 | 
			
		||||
            if (ObjectUtil.isNotNull(mappingHandler)) {
 | 
			
		||||
                Object handler = mappingHandler.getHandler();
 | 
			
		||||
                if (ObjectUtil.isNotNull(handler)) {
 | 
			
		||||
                    // 从handler获取注解
 | 
			
		||||
                    if (handler instanceof HandlerMethod handlerMethod) {
 | 
			
		||||
                        return handlerMethod.getMethodAnnotation(ApiEncrypt.class);
 | 
			
		||||
                    }
 | 
			
		||||
                }
 | 
			
		||||
            }
 | 
			
		||||
        } catch (Exception e) {
 | 
			
		||||
            throw new RuntimeException(e);
 | 
			
		||||
        }
 | 
			
		||||
        return null;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    @Override
 | 
			
		||||
    public void destroy() {
 | 
			
		||||
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
@@ -24,7 +24,7 @@ public class DecryptRequestBodyWrapper extends HttpServletRequestWrapper {
 | 
			
		||||
 | 
			
		||||
    private final byte[] body;
 | 
			
		||||
 | 
			
		||||
    public DecryptRequestBodyWrapper(HttpServletRequest request, String publicKey, String privateKey, String headerFlag) throws IOException {
 | 
			
		||||
    public DecryptRequestBodyWrapper(HttpServletRequest request, String privateKey, String headerFlag) throws IOException {
 | 
			
		||||
        super(request);
 | 
			
		||||
        // 获取 AES 密码 采用 RSA 加密
 | 
			
		||||
        String headerRsa = request.getHeader(headerFlag);
 | 
			
		||||
 
 | 
			
		||||
@@ -0,0 +1,121 @@
 | 
			
		||||
package org.dromara.common.encrypt.filter;
 | 
			
		||||
 | 
			
		||||
import cn.hutool.core.util.RandomUtil;
 | 
			
		||||
import jakarta.servlet.ServletOutputStream;
 | 
			
		||||
import jakarta.servlet.WriteListener;
 | 
			
		||||
import jakarta.servlet.http.HttpServletResponse;
 | 
			
		||||
import jakarta.servlet.http.HttpServletResponseWrapper;
 | 
			
		||||
import org.dromara.common.encrypt.utils.EncryptUtils;
 | 
			
		||||
 | 
			
		||||
import java.io.*;
 | 
			
		||||
import java.nio.charset.StandardCharsets;
 | 
			
		||||
 | 
			
		||||
/**
 | 
			
		||||
 * 加密响应参数包装类
 | 
			
		||||
 *
 | 
			
		||||
 * @author Michelle.Chung
 | 
			
		||||
 */
 | 
			
		||||
public class EncryptResponseBodyWrapper extends HttpServletResponseWrapper {
 | 
			
		||||
 | 
			
		||||
    private final ByteArrayOutputStream byteArrayOutputStream;
 | 
			
		||||
    private final ServletOutputStream servletOutputStream;
 | 
			
		||||
    private final PrintWriter printWriter;
 | 
			
		||||
 | 
			
		||||
    public EncryptResponseBodyWrapper(HttpServletResponse response) throws IOException {
 | 
			
		||||
        super(response);
 | 
			
		||||
        this.byteArrayOutputStream = new ByteArrayOutputStream();
 | 
			
		||||
        this.servletOutputStream = this.getOutputStream();
 | 
			
		||||
        this.printWriter = new PrintWriter(new OutputStreamWriter(byteArrayOutputStream));
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    @Override
 | 
			
		||||
    public PrintWriter getWriter() {
 | 
			
		||||
        return printWriter;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    @Override
 | 
			
		||||
    public void flushBuffer() throws IOException {
 | 
			
		||||
        if (servletOutputStream != null) {
 | 
			
		||||
            servletOutputStream.flush();
 | 
			
		||||
        }
 | 
			
		||||
        if (printWriter != null) {
 | 
			
		||||
            printWriter.flush();
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    @Override
 | 
			
		||||
    public void reset() {
 | 
			
		||||
        byteArrayOutputStream.reset();
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    public byte[] getResponseData() throws IOException {
 | 
			
		||||
        flushBuffer();
 | 
			
		||||
        return byteArrayOutputStream.toByteArray();
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    public String getContent() throws IOException {
 | 
			
		||||
        flushBuffer();
 | 
			
		||||
        return byteArrayOutputStream.toString();
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * 获取加密内容
 | 
			
		||||
     *
 | 
			
		||||
     * @param servletResponse response
 | 
			
		||||
     * @param publicKey       RSA公钥 (用于加密 AES 秘钥)
 | 
			
		||||
     * @param headerFlag      请求头标志
 | 
			
		||||
     * @return 加密内容
 | 
			
		||||
     * @throws IOException
 | 
			
		||||
     */
 | 
			
		||||
    public String getEncryptContent(HttpServletResponse servletResponse, String publicKey, String headerFlag) throws IOException {
 | 
			
		||||
        // 生成秘钥
 | 
			
		||||
        String aesPassword = RandomUtil.randomString(32);
 | 
			
		||||
        System.out.println("aesPassword = " + aesPassword);
 | 
			
		||||
        // 秘钥使用 Base64 编码
 | 
			
		||||
        String encryptAes = EncryptUtils.encryptByBase64(aesPassword);
 | 
			
		||||
        // Rsa 公钥加密 Base64 编码
 | 
			
		||||
        String encryptPassword = EncryptUtils.encryptByRsa(encryptAes, publicKey);
 | 
			
		||||
 | 
			
		||||
        // 设置响应头
 | 
			
		||||
        servletResponse.setHeader(headerFlag, encryptPassword);
 | 
			
		||||
        servletResponse.setHeader("Access-Control-Allow-Origin", "*");
 | 
			
		||||
        servletResponse.setHeader("Access-Control-Allow-Methods", "*");
 | 
			
		||||
        servletResponse.setCharacterEncoding(StandardCharsets.UTF_8.toString());
 | 
			
		||||
 | 
			
		||||
        // 获取原始内容
 | 
			
		||||
        String originalBody = this.getContent();
 | 
			
		||||
        // 对内容进行加密
 | 
			
		||||
        return EncryptUtils.encryptByAes(originalBody, aesPassword);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    @Override
 | 
			
		||||
    public ServletOutputStream getOutputStream() throws IOException {
 | 
			
		||||
        return new ServletOutputStream() {
 | 
			
		||||
            @Override
 | 
			
		||||
            public boolean isReady() {
 | 
			
		||||
                return false;
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
            @Override
 | 
			
		||||
            public void setWriteListener(WriteListener writeListener) {
 | 
			
		||||
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
            @Override
 | 
			
		||||
            public void write(int b) throws IOException {
 | 
			
		||||
                byteArrayOutputStream.write(b);
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
            @Override
 | 
			
		||||
            public void write(byte[] b) throws IOException {
 | 
			
		||||
                byteArrayOutputStream.write(b);
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
            @Override
 | 
			
		||||
            public void write(byte[] b, int off, int len) throws IOException {
 | 
			
		||||
                byteArrayOutputStream.write(b, off, len);
 | 
			
		||||
            }
 | 
			
		||||
        };
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
}
 | 
			
		||||
@@ -21,14 +21,14 @@ public class ApiDecryptProperties {
 | 
			
		||||
     */
 | 
			
		||||
    private String headerFlag;
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * 公钥
 | 
			
		||||
     * 响应加密公钥
 | 
			
		||||
     */
 | 
			
		||||
    private String publicKey;
 | 
			
		||||
 | 
			
		||||
    /**
 | 
			
		||||
     * 私钥
 | 
			
		||||
     * 请求解密私钥
 | 
			
		||||
     */
 | 
			
		||||
    private String privateKey;
 | 
			
		||||
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
		Reference in New Issue
	
	Block a user