通过HandlerInterceptor、HttpServletRequestWrapper、Filter实现接口前置参数处理后依然触发注解式参数校验(日志、参数加解密等)

1、因为正常HttpServletRequest只能获取一次body参数,无法进行修改,所以需要通过Wrapper(包装)自定义一个request

import javax.servlet.ReadListener;
import javax.servlet.ServletInputStream;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletRequestWrapper;
import java.io.*;

/**
 * getInputStream获取流 然后在流中获取数据 但是这个方法只能获取一次
 * 重写httpservletrequestwrapper把request保存下来
 * 用过滤器把保存的request填进去 就可以多次读取了
 */
public class RequestWrapper extends HttpServletRequestWrapper {

    private String body;

    public RequestWrapper(HttpServletRequest request) throws IOException {
        super(request);
        StringBuilder stringBuilder = new StringBuilder();
        BufferedReader bufferedReader = null;
        try {
            InputStream inputStream = request.getInputStream();
            if (inputStream != null) {
                bufferedReader = new BufferedReader(new InputStreamReader(inputStream));
                char[] charBuffer = new char[128];
                int bytesRead = -1;
                while ((bytesRead = bufferedReader.read(charBuffer)) > 0) {
                    stringBuilder.append(charBuffer, 0, bytesRead);
                }
            } else {
                stringBuilder.append("");
            }
        } catch (IOException ex) {
            throw ex;
        } finally {
            if (bufferedReader != null) {
                try {
                    bufferedReader.close();
                } catch (IOException ex) {
                    throw ex;
                }
            }
        }
        body = stringBuilder.toString();
    }

    @Override
    public ServletInputStream getInputStream() throws IOException {
        final ByteArrayInputStream byteArrayInputStream = new ByteArrayInputStream(body.getBytes());
        ServletInputStream servletInputStream = new ServletInputStream() {
            public boolean isFinished() {
                return false;
            }

            public boolean isReady() {
                return false;
            }

            public void setReadListener(ReadListener readListener) {
            }

            public int read() throws IOException {
                return byteArrayInputStream.read();
            }
        };
        return servletInputStream;

    }

    @Override
    public BufferedReader getReader() throws IOException {
        return new BufferedReader(new InputStreamReader(this.getInputStream()));
    }

    public String getBody() {
        return this.body;
    }

    // 赋值给body字段
    public void setBody(String body) {
        this.body = body;
    }
}

2、虽然自定义了request但是,请求不会直接使用自定义的Request,需要通过拦截器将所有的请求都转换成自定义的Request

public class WebFilter implements Filter {
    @Override
    public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain chain)
            throws IOException, ServletException {
        ServletRequest request = null;
        if (servletRequest instanceof HttpServletRequest) {
            request = new RequestWrapper((HttpServletRequest) servletRequest);
        }
        if (request == null) {
            chain.doFilter(servletRequest, servletResponse);
        } else {
            chain.doFilter(request, servletResponse);
        }
    }

    @Override
    public void destroy() {
    }
}

3、为WebFilter进行配置,此步骤可以省略,可以采用纯注解方式配置过滤器

@Configuration
public class FilterConfig {
    @Bean
    FilterRegistrationBean<WebFilter> myFilterFilterRegistrationBean() {
        FilterRegistrationBean<WebFilter> bean = new FilterRegistrationBean<>();
        bean.setFilter(new WebFilter());
        bean.setOrder(-1);
        bean.setUrlPatterns(Collections.singletonList("/*"));
        return bean;
    }
}

4、自定义注解,这里时为了实现加解密

@Retention(RetentionPolicy.RUNTIME)
@Target({ ElementType.METHOD, ElementType.TYPE })
public @interface DesDecrypt {
}

4、创建拦截器,对所有的标注了@DesDecrypt 的接口进行拦截,并使用自定义的request进行处理,将body参数取出,处理之后在注入回去

参数使用json字符串传递,使用data参数传递,接口中的参数依然正常使用自定义对象承接并且使用@Validated @RequestBody 注解,并出发参数校验

@Slf4j
@Component
public class DesDecryptInterceptor implements HandlerInterceptor {
    private final ObjectMapper objectMapper;

    public DesDecryptInterceptor(ObjectMapper objectMapper) {
        this.objectMapper = objectMapper;
    }

    @Override
    public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
        if (handler instanceof HandlerMethod) {
            HandlerMethod handlerMethod = (HandlerMethod) handler;
            DesDecrypt desDecryptAnnotation = handlerMethod.getMethodAnnotation(DesDecrypt.class);

            if (desDecryptAnnotation != null) {
                RequestWrapper requestWrapper = (RequestWrapper) request;
                String requestBody = requestWrapper.getBody();
                if (requestBody == null || requestBody.isEmpty()) {
                    throw new RuntimeException("请求体为空");
                }
                JSONObject jsonObject = JSON.parseObject(requestBody);
                if (!jsonObject.containsKey("data")) {
                    throw new RuntimeException("请求体中不存在data字段");
                }
                String data = jsonObject.getString("data");
                log.info("Decrypted request data: {}", data);
                requestWrapper.setBody(data);
            }
        }

        return true;
    }

    private String readRequestBody(HttpServletRequest request) throws IOException {
        byte[] buffer = new byte[request.getContentLength()];
        int bytesRead = request.getInputStream().read(buffer);
        return new String(buffer, 0, bytesRead, StandardCharsets.UTF_8);
    }
}

5、性能测试

5.1 创建一个测试接口

@DesDecrypt
    @PostMapping("/parameterPreProcess")
    public String parameterPreProcess(@Validated @RequestBody ControlFrontParam param) {
        log.info("param json: {}", JSON.toJSONString(param));
        return JSON.toJSONString(param);
    }

5.2 在使用DesDecrypt注解标记下进行访问(经过过滤器和拦截器)

{

    "data": "{\"name\": \"qR)\",\"age\": 51,\"idCard\": \"54829220331210989X\"}"

}

访问多次,Time在21~29ms之间,多数在21~25ms

5.3 在不使用DesDecrypt注解标记下进行访问

{
  "idCard": "54829220331210989X",
  "name": "zq",
  "age": 10
}

访问多次,Time在17~28ms之间,多数在18~23ms

*总体来说,经过过滤器和拦截器处理后,会影响响应时间,但是影响不大,对于带给整个项目在加解密方面的便捷性来说,是值得的;

通过这种方式实现的参数加密传输,只需要通过修改注解,即改变接口是否加密,对接口的结构、参数、以及参数校验都没有影响。

  • 4
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 1
    评论
如果在HandlerInterceptor修改了GET请求的参数,但在Controller获取到的参数依然修改前的值,可能是因为HandlerInterceptor没有正确地将修改后的请求对象传递给下一个处理器。 在HandlerInterceptor的preHandle方法,我们可以通过修改请求对象的方修改参数值。然后,我们需要使用FilterChain的doFilter方法将修改后的请求对象传递给下一个处理器(可以是下一个拦截器或Controller)。 以下是一个示例代码,演示了如何在HandlerInterceptor正确传递修改后的请求对象: ```java import org.springframework.web.servlet.HandlerInterceptor; import org.springframework.web.servlet.ModelAndView; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; public class ModifyGetParameterInterceptor implements HandlerInterceptor { @Override public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception { if (request.getMethod().equals("GET")) { // 获取原始参数值 String originalParamValue = request.getParameter("paramName"); // 修改参数值 String modifiedParamValue = originalParamValue + "_modified"; // 创建一个新的请求对象,并设置修改后的参数值 ModifiedRequestWrapper modifiedRequest = new ModifiedRequestWrapper(request); modifiedRequest.addParameter("paramName", modifiedParamValue); // 将修改后的请求对象传递给下一个处理器 return super.preHandle(modifiedRequest, response, handler); } return true; } @Override public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception { // 在postHandle方法不需要进行任何操作 } @Override public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception { // 在afterCompletion方法不需要进行任何操作 } } ``` 在上述示例代码,我们使用super.preHandle方法将修改后的请求对象传递给下一个处理器。这样,在Controller获取到的参数将会是修改后的值。 需要注意的是,如果有多个HandlerInterceptor,确保在每个HandlerInterceptor的preHandle方法都正确地传递修改后的请求对象。只有在最后一个HandlerInterceptor的preHandle方法调用super.preHandle方法才能将请求传递给Controller。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

起风了小猪仔

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值