xiejun
2024-11-01 80b6cbfc9c861469146318d0b3dd5f8b8b525b8a
Source/BladeX-Tool/blade-core-boot/src/main/java/org/springblade/core/boot/request/XssHttpServletRequestWrapper.java
¶Ô±ÈÐÂÎļþ
@@ -0,0 +1,175 @@
/*
 *      Copyright (c) 2018-2028, Chill Zhuang All rights reserved.
 *
 *  Redistribution and use in source and binary forms, with or without
 *  modification, are permitted provided that the following conditions are met:
 *
 *  Redistributions of source code must retain the above copyright notice,
 *  this list of conditions and the following disclaimer.
 *  Redistributions in binary form must reproduce the above copyright
 *  notice, this list of conditions and the following disclaimer in the
 *  documentation and/or other materials provided with the distribution.
 *  Neither the name of the dreamlu.net developer nor the names of its
 *  contributors may be used to endorse or promote products derived from
 *  this software without specific prior written permission.
 *  Author: Chill åº„骞 (smallchill@163.com)
 */
package org.springblade.core.boot.request;
import org.springblade.core.tool.utils.StringUtil;
import org.springblade.core.tool.utils.WebUtil;
import org.springframework.http.HttpHeaders;
import org.springframework.http.MediaType;
import javax.servlet.ReadListener;
import javax.servlet.ServletInputStream;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletRequestWrapper;
import java.io.BufferedReader;
import java.io.ByteArrayInputStream;
import java.io.IOException;
import java.io.InputStreamReader;
import java.util.LinkedHashMap;
import java.util.Map;
/**
 * XSS过滤
 *
 * @author Chill
 */
public class XssHttpServletRequestWrapper extends HttpServletRequestWrapper {
   /**
    * æ²¡è¢«åŒ…装过的HttpServletRequest(特殊场景,需要自己过滤)
    */
   private final HttpServletRequest orgRequest;
   /**
    * ç¼“存报文,支持多次读取流
    */
   private byte[] body;
   /**
    * html过滤
    */
   private final static XssHtmlFilter HTML_FILTER = new XssHtmlFilter();
   public XssHttpServletRequestWrapper(HttpServletRequest request) {
      super(request);
      orgRequest = request;
   }
   @Override
   public BufferedReader getReader() throws IOException {
      return new BufferedReader(new InputStreamReader(getInputStream()));
   }
   @Override
   public ServletInputStream getInputStream() throws IOException {
      if (super.getHeader(HttpHeaders.CONTENT_TYPE) == null) {
         return super.getInputStream();
      }
      if (super.getHeader(HttpHeaders.CONTENT_TYPE).startsWith(MediaType.MULTIPART_FORM_DATA_VALUE)) {
         return super.getInputStream();
      }
      if (body == null) {
         body = xssEncode(WebUtil.getRequestBody(super.getInputStream())).getBytes();
      }
      final ByteArrayInputStream byteArrayInputStream = new ByteArrayInputStream(body);
      return new ServletInputStream() {
         @Override
         public int read() {
            return byteArrayInputStream.read();
         }
         @Override
         public boolean isFinished() {
            return false;
         }
         @Override
         public boolean isReady() {
            return false;
         }
         @Override
         public void setReadListener(ReadListener readListener) {
         }
      };
   }
   @Override
   public String getParameter(String name) {
      String value = super.getParameter(xssEncode(name));
      if (StringUtil.isNotBlank(value)) {
         value = xssEncode(value);
      }
      return value;
   }
   @Override
   public String[] getParameterValues(String name) {
      String[] parameters = super.getParameterValues(name);
      if (parameters == null || parameters.length == 0) {
         return null;
      }
      for (int i = 0; i < parameters.length; i++) {
         parameters[i] = xssEncode(parameters[i]);
      }
      return parameters;
   }
   @Override
   public Map<String, String[]> getParameterMap() {
      Map<String, String[]> map = new LinkedHashMap<>();
      Map<String, String[]> parameters = super.getParameterMap();
      for (String key : parameters.keySet()) {
         String[] values = parameters.get(key);
         for (int i = 0; i < values.length; i++) {
            values[i] = xssEncode(values[i]);
         }
         map.put(key, values);
      }
      return map;
   }
   @Override
   public String getHeader(String name) {
      String value = super.getHeader(xssEncode(name));
      if (StringUtil.isNotBlank(value)) {
         value = xssEncode(value);
      }
      return value;
   }
   private String xssEncode(String input) {
      return HTML_FILTER.filter(input);
   }
   /**
    * èŽ·å–åˆå§‹request
    *
    * @return HttpServletRequest
    */
   public HttpServletRequest getOrgRequest() {
      return orgRequest;
   }
   /**
    * èŽ·å–åˆå§‹request
    *
    * @param request request
    * @return HttpServletRequest
    */
   public static HttpServletRequest getOrgRequest(HttpServletRequest request) {
      if (request instanceof XssHttpServletRequestWrapper) {
         return ((XssHttpServletRequestWrapper) request).getOrgRequest();
      }
      return request;
   }
}