XSS跨站脚本安全漏洞防护

文章目录

  • 1 跨站脚本
    • 1.1 存储型XSS
    • 1.2 反射型XSS
  • 2 、案例
    • 2.1 通过正则表达式替换跨站脚本
    • 2.2 构建请求的代理类,在构造方法中对请求中的内容进行分析
    • 2.3 构建响应的代理类
  • 3 测试
    • 3.1 在接口的body参数中添加一个脚本
    • 3.2 我们在数据库中给一条测试数据加入脚本

1 跨站脚本

1.1 存储型XSS

概念:存储型XSS是指应用程序通过Web请求获取不可信赖的数据,并且在未检验数据是否存在XSS代码的情况下,将其存入数据库。当程序下一次从数据库中获取该数据时,致使页面再次执行XSS代码。存储型XSS可以持续攻击用户,在用户提交了包含XSS代码的数据存储到数据库后,每当用户在浏览网页查询对应数据库中的数据时,那些包含XSS代码的数据就会在服务器解析并加载,当浏览器读到XSS代码后,会当做正常的HTML和JS解析并执行,于是发生存储型XSS攻击。
:下面JSP代码片段的功能是根据一个已知用户雇员ID(id)从数据库中查询出该用户的地址,并显示在JSP页面上。

<%
...
Statement stmt = conn.createStatement();
ResultSet rs = stmt.executeQuery("select * from users where id =" + id);
String address = null;
if (rs != null) {
rs.next();
address = rs.getString("address");
}
%>
家庭地址: <%= address %>

如果address的值是由用户提供的,且存入数据库时没有进行合理的校验,那么攻击者就可以利用上面的代码进行存储型XSS攻击。

修复建议:为了避免存储型XSS攻击,建议采用以下方式进行防御:

  1. 对从数据库或其它后端数据存储获取不可信赖的数据进行合理验证(如年龄只能是数字),对特殊字符(如<、>、'、"以及<script>、javascript等进行过滤。
  2. 根据数据将要置于HTML上下文中的不同位置(HTML标签、HTML属性、JavaScript脚本、CSS、URL),对所有不可信数据进行恰当的输出编码。
    例:采用OWASP ESAPI对数据输出HTML上下文中不同位置,编码方法如下。
//HTML encode
ESAPI.encoder().encodeForHTML(inputData);
//HTML attribute encode
ESAPI.encoder().encodeForHTMLAttribute(inputData);
//JavaScript encode
ESAPI.encoder().encodeForJavaScript(inputData);
//CSS encode
ESAPI.encoder().encodeForCSS(inputData);
//URL encode
ESAPI.encoder().encodeForURL(inputData);
  1. 设置HttpOnly属性,避免攻击者利用跨站脚本漏洞进行Cookie劫持攻击。在Java EE中,给Cookie添加HttpOnly的代码如下:
response.setHeader("Set-Cookie","cookiename=cookievalue; path=/; Domain=domainvaule; Max-age=seconds; HttpOnly");

1.2 反射型XSS

概念:反射型XSS是指应用程序通过Web请求获取不可信赖的数据,并在未检验数据是否存在恶意代码的情况下,将其发送给用户。反射型XSS一般可以由攻击者构造带有恶意代码参数的URL来实现,在构造的URL地址被打开后,其中包含的恶意代码参数被浏览器解析和执行。这种攻击的特点是非持久化,必须用户点击包含恶意代码参数的链接时才会触发。
:下面JSP代码片段的功能是从HTTP请求中读取输入的用户名(username)并显示到页面。

<%
String name= request.getParameter("username"); %>
...
姓名: <%= name%>

如果name里有包含恶意代码,那么Web浏览器就会像显示HTTP响应那样执行该代码,应用程序将受到反射型XSS攻击。

修复建议:为了避免反射型XSS攻击,建议采用以下方式进行防御:

  1. 对用户的输入进行合理验证(如年龄只能是数字),对特殊字符(如<、>、'、"以及<script>、javascript等进行过滤。
  2. 根据数据将要置于HTML上下文中的不同位置(HTML标签、HTML属性、JavaScript脚本、CSS、URL),对所有不可信数据进行恰当的输出编码。
    :采用OWASP ESAPI对数据输出HTML上下文中不同位置,编码方法如下。
//HTML encode
ESAPI.encoder().encodeForHTML(inputData);
//HTML attribute encode
ESAPI.encoder().encodeForHTMLAttribute(inputData);
//JavaScript encode
ESAPI.encoder().encodeForJavaScript(inputData);
//CSS encode
ESAPI.encoder().encodeForCSS(inputData);
//URL encode
ESAPI.encoder().encodeForURL(inputData);
  1. 设置HttpOnly属性,避免攻击者利用跨站脚本漏洞进行Cookie劫持攻击。在Java EE中,给Cookie添加HttpOnly的代码如下:
response.setHeader("Set-Cookie","cookiename=cookievalue; path=/; Domain=domainvaule; Max-age=seconds; HttpOnly");

2 、案例

2.1 通过正则表达式替换跨站脚本

import org.owasp.esapi.ESAPI;
import org.springframework.core.codec.EncodingException;import java.util.regex.Matcher;
import java.util.regex.Pattern;/*** 跨站脚本和存储型XSS攻击防御工具类* @author zyw* @date 16:52:11*/
public class XSSUtils {public static Pattern scriptPattern1 = Pattern.compile("<script>(.*?)</script>", Pattern.CASE_INSENSITIVE);public static Pattern scriptPattern2 = Pattern.compile("src[\r\n]*=[\r\n]*\\\'(.*?)\\\'", Pattern.CASE_INSENSITIVE | Pattern.MULTILINE | Pattern.DOTALL);public static Pattern scriptPattern3 = Pattern.compile("src[\r\n]*=[\r\n]*\\\"(.*?)\\\"", Pattern.CASE_INSENSITIVE | Pattern.MULTILINE | Pattern.DOTALL);public static Pattern scriptPattern4 = Pattern.compile("</script>", Pattern.CASE_INSENSITIVE);public static Pattern scriptPattern5 = Pattern.compile("<script(.*?)>", Pattern.CASE_INSENSITIVE | Pattern.MULTILINE | Pattern.DOTALL);public static Pattern scriptPattern6 = Pattern.compile("eval\\((.*?)\\)", Pattern.CASE_INSENSITIVE | Pattern.MULTILINE | Pattern.DOTALL);public static Pattern scriptPattern7 = Pattern.compile("e­xpression\\((.*?)\\)", Pattern.CASE_INSENSITIVE | Pattern.MULTILINE | Pattern.DOTALL);public static Pattern scriptPattern8 = Pattern.compile("javascript:", Pattern.CASE_INSENSITIVE);public static Pattern scriptPattern9 = Pattern.compile("vbscript:", Pattern.CASE_INSENSITIVE);public static Pattern scriptPattern10 = Pattern.compile("onload(.*?)=", Pattern.CASE_INSENSITIVE | Pattern.MULTILINE | Pattern.DOTALL);public static Pattern scriptPattern11 = Pattern.compile(".*<.*", Pattern.CASE_INSENSITIVE );public static Pattern scriptPattern12 = Pattern.compile("<.*>.*</.*>", Pattern.CASE_INSENSITIVE);public static String striptXSS(String value) {if (value != null) {//脚本匹配Matcher matcher = scriptPattern12.matcher(value);if (matcher.find()) {//截取匹配的部分String group = matcher.group();//采用OWASP ESAPI对数据输出HTML上下文中不同位置value = value.replace(group, owaspEsapi(group));}value = value.replaceAll("", "");value = scriptPattern1.matcher(value).replaceAll("");value = scriptPattern2.matcher(value).replaceAll("");value = scriptPattern3.matcher(value).replaceAll("");value = scriptPattern4.matcher(value).replaceAll("");value = scriptPattern5.matcher(value).replaceAll("");value = scriptPattern6.matcher(value).replaceAll("");value = scriptPattern7.matcher(value).replaceAll("");value = scriptPattern8.matcher(value).replaceAll("");value = scriptPattern9.matcher(value).replaceAll("");value = scriptPattern10.matcher(value).replaceAll("");value = scriptPattern11.matcher(value).replaceAll("");}return value;}/*** 采用OWASP ESAPI对数据输出HTML上下文中不同位置** @param data* @return* @throws EncodingException*/public static String owaspEsapi(String data){try {//HTML encodedata = ESAPI.encoder().encodeForHTML(data);//HTML attribute encodedata = ESAPI.encoder().encodeForHTMLAttribute(data);//JavaScript encodedata = ESAPI.encoder().encodeForJavaScript(data);//CSS encodedata = ESAPI.encoder().encodeForCSS(data);//URL encodedata = ESAPI.encoder().encodeForURL(data);}catch (EncodingException e){}finally {return data;}}}

2.2 构建请求的代理类,在构造方法中对请求中的内容进行分析

注:这里需要对传输参数为文件的接口做针对性的处理,同时接口中的参数形式也分很多种,需要

/*** 请求过滤业务实现类*/
@Slf4j
public class XssRequestWrappers extends HttpServletRequestWrapper {@Getter@Setterprivate String requestBodyStr;//用于将流保存下来private byte[] requestBody;private CommonsMultipartResolver multiparResolver = new CommonsMultipartResolver();public XssRequestWrappers(HttpServletRequest request) {super(request);String type = request.getHeader("Content-Type");//处理上传文件接口if (!StringUtils.isEmpty(type) && type.contains("multipart/form-data")) {MultipartHttpServletRequest multipartHttpServletRequest = multiparResolver.resolveMultipart(request);Map<String, String[]> stringMap = multipartHttpServletRequest.getParameterMap();if (!stringMap.isEmpty()) {for (String key : stringMap.keySet()) {String value = multipartHttpServletRequest.getParameter(key);XSSUtils.striptXSS(key);XSSUtils.striptXSS(value);}}super.setRequest(multipartHttpServletRequest);} else {//其他接口//获取所有参数键值对的map集合Map parameterMap = request.getParameterMap();Set<String> keystrSet = parameterMap.keySet();if (keystrSet.size() != 0) {String key = "";String value = "";for (String keystring : keystrSet) {key = XSSUtils.striptXSS(keystring);if (!key.equals(keystring)) {parameterMap.put(key, parameterMap.remove(keystring));}value = XSSUtils.striptXSS(String.valueOf(parameterMap.get(keystring)));if (value != null && !"".equals(value) && !value.equals(parameterMap.get(keystring))) {parameterMap.put(keystring, value);}}}//获取请求的bodytry {requestBody = StreamUtils.copyToByteArray(request.getInputStream());String requestBodyStr = XSSUtils.striptXSS(IOUtils.toString(requestBody, "utf-8"));requestBody = requestBodyStr.getBytes("UTF8");}catch (IOException e){}}}@Overridepublic ServletInputStream getInputStream() {final ByteArrayInputStream bais = new ByteArrayInputStream(requestBody);return new ServletInputStream() {@Overridepublic boolean isFinished() {return false;}@Overridepublic boolean isReady() {return false;}@Overridepublic void setReadListener(ReadListener listener) {}@Overridepublic int read() {return bais.read();}};}@Overridepublic BufferedReader getReader() {return new BufferedReader(new InputStreamReader(getInputStream()));}@Overridepublic String[] getParameterValues(String parameter) {String[] values = super.getParameterValues(parameter);if (values == null) {return null;}int count = values.length;String[] encodedValues = new String[count];for (int i = 0; i < count; i++) {encodedValues[i] = XSSUtils.striptXSS(values[i]);}return encodedValues;}@Overridepublic String getParameter(String parameter) {String value = super.getParameter(parameter);return XSSUtils.striptXSS(value);}@Overridepublic String getHeader(String name) {String value = super.getHeader(name);return XSSUtils.striptXSS(value);}@Overridepublic Map<String, String[]> getParameterMap() {Map<String, String[]> map1 = super.getParameterMap();Map<String, String[]> escapseMap = new HashMap<String, String[]>();Set<String> keys = map1.keySet();for (String key : keys) {String[] valArr = map1.get(key);if (valArr != null && valArr.length > 0) {String[] escapseValArr = new String[valArr.length];for (int i = 0; i < valArr.length; i++) {String escapseVal = XSSUtils.striptXSS(valArr[i]);escapseValArr[i] = escapseVal;}escapseMap.put(key, escapseValArr);}}return escapseMap;}}

2.3 构建响应的代理类


```java
import java.io.ByteArrayOutputStream;
import java.io.IOException;import javax.servlet.ServletOutputStream;
import javax.servlet.WriteListener;
import javax.servlet.http.HttpServletResponse;
import javax.servlet.http.HttpServletResponseWrapper;/*** 返回值输出代理类** @Title: ResponseWrapper* @Description:* @author zyw* @date 9:52:11*/
public class ResponseWrapper extends HttpServletResponseWrapper {private ByteArrayOutputStream buffer;private ServletOutputStream out;public ResponseWrapper(HttpServletResponse httpServletResponse){super(httpServletResponse);buffer = new ByteArrayOutputStream();out = new WrapperOutputStream(buffer);}@Overridepublic ServletOutputStream getOutputStream()throws IOException{return out;}@Overridepublic void flushBuffer()throws IOException{if (out != null){out.flush();}}public byte[] getContent()throws IOException{flushBuffer();return buffer.toByteArray();}class WrapperOutputStream extends ServletOutputStream{private ByteArrayOutputStream bos;public WrapperOutputStream(ByteArrayOutputStream bos){this.bos = bos;}@Overridepublic void write(int b)throws IOException{bos.write(b);}@Overridepublic boolean isReady(){// TODO Auto-generated method stubreturn false;}@Overridepublic void setWriteListener(WriteListener arg0){// TODO Auto-generated method stub}}
}

## 2.4 通过Filter过滤掉请求和响应中的跨站脚本```java
/*** 存储型XSS、跨站脚本过滤器** @author zyw* @since 2022-09-01 20:09:31*/
@Component
@WebFilter(filterName = "XSSFilter",/*** 通配符(*)表示对所有的web资源进行拦截*/urlPatterns = "/*"
)
public class XSSFilter implements Filter {@Overridepublic void init(FilterConfig arg0) throws ServletException {System.out.println("初始化过滤器!");}@Overridepublic void destroy() {System.out.println("销毁过滤器!");}@Overridepublic void doFilter(ServletRequest request, ServletResponse response, FilterChain chain)throws IOException, ServletException {PrintWriter writer = response.getWriter();//过滤请求XssRequestWrappers xssRequestWrappers = new XssRequestWrappers((HttpServletRequest) request);//过滤返回ResponseWrapper wrapperResponse = new ResponseWrapper((HttpServletResponse) response);//转换成代理类chain.doFilter(xssRequestWrappers, wrapperResponse);byte[] content = wrapperResponse.getContent();//获取返回值//判断是否有值if (content.length > 0) {String str = new String(content, "UTF-8");String ciphertext = null;try {//......根据需要处理返回值ciphertext = XSSUtils.striptXSS(str);} catch (Exception e) {e.printStackTrace();}//把返回值输出到客户端writer.append(ciphertext);writer.close();}}}

3 测试

3.1 在接口的body参数中添加一个脚本

在这里插入图片描述
可以看出已对其进行恰当的输出编码
在这里插入图片描述

3.2 我们在数据库中给一条测试数据加入脚本

在这里插入图片描述

可以看到已对其进行恰当的输出编码在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://xiahunao.cn/news/350855.html

如若内容造成侵权/违法违规/事实不符,请联系瞎胡闹网进行投诉反馈,一经查实,立即删除!

相关文章

offer收割机再现,接口测试常问面试题 (附答案),对标大厂面试...

目录&#xff1a;导读 前言一、Python编程入门到精通二、接口自动化项目实战三、Web自动化项目实战四、App自动化项目实战五、一线大厂简历六、测试开发DevOps体系七、常用自动化测试工具八、JMeter性能测试九、总结&#xff08;尾部小惊喜&#xff09; 前言 软件测试面试题&am…

【送书福利-第九期】《用ChatGPT与VBA一键搞定Excel》

大家好&#xff0c;我是洲洲&#xff0c;欢迎关注&#xff0c;一个爱听周杰伦的程序员。关注公众号【程序员洲洲】即可获得10G学习资料、面试笔记、大厂独家学习体系路线等…还可以加入技术交流群欢迎大家在CSDN后台私信我&#xff01; 本文目录 一、前言二、内容介绍三、作者介…

Ubuntu系统镜像下载,国内镜像站大全(山大/清华/阿里/浙大/中科大...)

装Ubuntu&#xff0c;是很多理工科同学入门的第一个挑战&#xff0c;首先我们就需要找到一个能用的iso镜像&#xff0c;根据你的网络环境的不同&#xff0c;不同的站点下载速度会不一样&#xff0c;下面列举一下几个比较好用的&#xff0c;都是来自Ubuntu官方推荐国内镜像站链接…

代码分析工具 自定义插件——检查any类型的代码

代码分析工具 自定义插件——检查any类型的代码 本文是基于掘金小册《前端依赖治理&#xff1a;代码分析工具开发实战》 转载于我的掘金博客&#xff0c;并非抄袭 gitHub仓库&#xff1a;https://github.com/huang-jaskld/code-analysis/tree/master 逻辑步骤 源代码过于庞大&a…

文华财经指标公式博易大师软件指标公式期货多空买卖指标图,文华加密破解二次加密

天:ma(CLOSE,25)MA(CLOSE,25)*20/100,COLORRED,LINETHICK2; 强势线:MA(CLOSE,25)MA(CLOSE,25)*6/100,COLORYELLOW,LINETHICK2; 天2:MA(CLOSE,18)MA(CLOSE,18)*20/100,COLORFFFFFF,LINETHICK1; stICKLINE(C>强势线,H,L,0,1),COLORFFFFFF; STICKLINE(C>强势线,C,O,2,1),COL…

外盘期货分仓软件(如智星系统,信管家)等功能

1、创建子账户&#xff0c;如图所示&#xff1a;管理--账户管理—开户&#xff0c;填写客户姓名&#xff0c;登陆密码&#xff0c;手机号&#xff0c;勾选账户风控。 2、子账户风控 a) 报警线&#xff1a;资金剩余达到保证金的 n%&#xff0c;系统报警&#xff1b; b) 强平线…

开发国信股票自动交易软件

大多免费的股票交易软件&#xff0c;不能自动盯盘&#xff0c;不能灵活简单地定制自动交易策略&#xff0c;而且最大的问题在于&#xff0c;不能上班时间看盘^_^。于是客户找到我们&#xff0c;希望能开发一个能解决以上痛点的股票自动交易软件。 我们通过逆向国信股票交易网页…

免费软件 --- 国信证券帐户历史成交统计器

很早之前就想写这个软件&#xff0c;这2天花费了点时间写出来了&#xff0c;测试后就发布了。当然了&#xff0c;本软件只针对国信证券而开发的&#xff0c;只支持统计国信证券交易软件的记录。如果是其他券商开户的&#xff0c;请不要来下载了。 软件简要说明&#xff1a;本软…

【大作业之爬虫实战+数据分析与可视化处理上】——案例——如桃花来

目的任务&#xff1a; 目的任务 爬取租房网对应元素&#xff1a; 租房名 每月的价格 室内规格构造 房间大小 交通信息 保存数据到excel表格中 做数据清洗 数据可视化呈现 设计要求&#xff1a; 设计要求&#xff1a; 以类的方式书写&#xff0c;简洁明了。能够促进学生…

记录 vue3 webpack 使用 iframe 遇到的坑

需求 我尝试用Vue3写一个自己的主页&#xff0c;把常用的功能集中到主页中&#xff0c;如下图 后发现一个好玩的东西&#xff0c;js实现的在网页底部出现鱼和波浪&#xff0c;如下图&#xff0c;就像想也放到自己的主页中&#xff0c;搜索后发现可以在Vue中用iframe标签直接引…

基于redis实现消息队列(更推荐使用专业的mq)

目录 利用redis实现消息队列&#xff08;基于list&#xff0c;点对点模型&#xff09;——lpush存放队列&#xff08;lpush 队列名 队列内容&#xff08;可一次存放多个内容&#xff0c;用空格隔开&#xff09;&#xff09; brpop取队列&#xff08;brpop 队列名 等待时间单位秒…

(windows)如何删除删不掉的文件

有时候删除时明明是管理员&#xff0c;却提示需要管理员权限 点击更改-然后再高级 立即查找&#xff0c;搜索当前用户 找到现在的用户名&#xff0c;确定 点击审核&#xff0c;添加--选择主体 高级—选择现在的计算机名——权限完全控制 7.所有对话框点击确定&#…

文件夹删不掉,显示有文件打开怎么办

1 打开任务管理器 2 选中性能 3 点开打开资源监视器 4 点进CPU 5 在输入框输入删除不掉的文件夹名字 6 再右击进程关掉就可

解决:文件名太长删不掉

今天删除一个无用文件夹时windows跳出如下弹窗&#xff1a; 解决方法&#xff1a; 1、在要删除的文件夹同级新建一个空文件夹&#xff08;如图&#xff1a;1是要删除的文件夹&#xff0c;也就是包含那个文件名过长的文件的文件夹&#xff0c;2是空文件夹&#xff09; 2、点下空…

win10系统遇到删不掉的文件夹怎么办

win10系统遇到删不掉的文件夹怎么办&#xff1a; 具体操作步骤如下&#xff1a; 1、打开“此电脑”-“查看”&#xff0c;在“文件扩展名”前打钩&#xff1b; 2、然后在桌面上新建一个txt文本文档&#xff1b; 3、双击打开新建的文本文档&#xff0c;复制粘贴下列的代码&…

删除电脑中删不掉的文件或文件夹

强制删除文件或文件夹 前言一、新建一个文本文件二、打开文本文件1.在文本文件中输入2.保存 前言 在一次拉取git文件&#xff0c;不知道做了什么错误的操作&#xff0c;是的文件夹在删除的时候&#xff0c;一直显示“当前文件不存在”&#xff0c;可是明明就存在我的桌面&…

遇见流氓软件怎么才能卸载,文件夹删不掉显示被占用(小鸟,海螺,原始传奇等),到底应该怎么办?

遇见流氓软件怎么才能卸载&#xff0c;文件夹删不掉显示被占用&#xff08;小鸟&#xff0c;海螺&#xff0c;原始传奇等&#xff09; 首先&#xff0c;描述一下我遇见流氓软件的过程&#xff0c;最开始我想下一个音乐播放器&#xff0c;选择了网易云音乐&#xff0c;在谷歌浏…

服务器上网站文件无法删除不了怎么办啊,Windows服务器上文件夹删不掉怎么办...

我们在用服务器的过程中有的时候会遇到这样的问题&#xff0c;那就是文件夹删除不掉。甚至有的时候需要重启服务器才能删除&#xff0c;甚至还遇到即便重启了服务器也删除不掉文件夹的情况。那么今天纵然云计算小编就带大家来学习一下怎么删除这些删除不掉的文件夹。 那么我们先…

如何删除tmp计算机桌面,temp文件删不掉怎么办

当我们在使用win10系统的时候&#xff0c;会产生系统临时文件&#xff0c;temp文件。如果一直不清理的话&#xff0c;太多的话就会占用系统磁盘空间影响电脑运行速度。但是有小伙伴发现自己电脑的temp文件删不掉不知道怎么办&#xff1f;下面小编就教下大家temp文件怎么删除。 …

空文件夹删不掉打不开,“该项目不存在请确认该项目位置“,“项目正在打开中无法删除“,“文件已损坏或者已经被移动删除“(多种方法图文详解,细节需要注意,以及可能遇到的问题)

1.事前概要-触发这种文件夹的情况 这个删不掉的文件或文件夹其实是Windows系统的祖传bug到目前为止依然没有修复,所以说我们需要通过特别的手段来处理它,听我慢慢讲他的缘由可能会对解决这个问题的帮助更多,会有几种方法,我都试过了的,我把最后一种成功的放在第一个讲,没有成功…