# 01-01 HTTP协议

  • HTTP协议的主要特点
  • HTTP报文的组成部分
  • HTTP方法
  • POST和GET的区别
  • HTTP状态码
  • 什么是持久链接
  • 什么是管线化

# HTTP协议主要特点

  • 简单快速:每个资源每个url,统一资源符是固定的
  • 灵活:每个HTTP协议头部都有数据类型,就可以通过HTTP协议完成不同的数据类型传输
  • 无连接:连接一次就会断掉
  • 无状态:客户端和服务器端两种身份,无法区别上一次连接和这一次连接的身份(通过session可以实现身份识别)

# HTTP报文

  • HTTP请求报文:请求行、请求头、空行、请求体
    • 请求行:包含HTTP方法、页面地址、HTTP协议、以及版本
    • 请求头:Key-Value值告诉服务端我需要哪些内容
    • 空行:下一个是请求体,需要解析
  • HTTP相应报文:状态行、响应头、空行、响应体

# POST和GET的区别

  • GET在浏览器回退时是无害的,而POST会再次提交请求
  • GET产生的URL地址可以被收藏,而POST不可以
  • GET请求会被浏览器主动缓存,而POST不会,除非手动设置
  • GET请求只能进行URL编码,而POST支持多种编码方式
  • GET请求参数会被完整保留在浏览器历史记录里,而POST中的参数不会被保留
  • GET请求在URL中传送的参数是有长度限制的,而POST没有长度限制
  • 对参数的数据类型,GET只能请求ASCII字符,而POST没有限制
  • GET比POST更不安全,因为参数直接暴露在URL上,所以不能用来传敏感信息
  • GET参数通过URL传递,POST放在Request body中

# HTTP状态码

  • 1XX:指示信息-表示请求已接受,继续处理
  • 2XX:成功-表示请求已被成功接收
    • 200 OK :客户端请求成功
    • 206 Partial Content:客户发送一个带有Range头的GET请求,服务器完成了它 播放视频和音频
  • 3XX:重定向-要完成请求必须进行更进一步的操作
    • 301 Move Permanently:所请求的页面已经转移至新的URL
      • 302 Found:所请求的页面已经临时转移到新的URL
      • 304 Not Modified:客户端有缓冲的文档并发出一个条件性的请求,服务器告诉客户,原来缓冲的文档还可以继续使用
  • 4XX:客户端错误-请求有语法错误或请求无法实现
    • 400 Bad Request:客户端请求有语法错误,不能被服务器所理解
    • 401 Unauthorized:请求未经授权,这个状态代码必须和WWW-Authenticate报头域一起使用
    • 403 Forbidden:对被请求页面的访问被禁止
    • 404 Not Found:请求资源不存在
  • 5XX:服务错误-服务器未能实现合法的请求
    • 500 Internal Server Error:服务器发生不可预期的错误原来缓冲的文档还可以继续使用
    • 503 Server Unavailable:请求未完成,服务器临时过载或当机,一段事件后恢复正常

# HTTP 持久连接 (HTTP1.1)

  • HTTP协议采用“请求-应答”模式,使用普通模式,即非Keep-Alive模式时,每个请求/应答客户和服务器都要新建一个连接,完成之后立即断开连接(HTTP协议为无连接的协议)
  • 当使用Keep-Alive模式(又称持久连接,连接重用)时,Keep-Alive功能使客户端到服务器端的连接持续有效,当出现对服务器的后继请求时,Keep-Alive功能避免了建立或者重新建立连接

# 管线化

  • 在使用持久连接的情况下,某个连接上消息的传递类似于

  • 请求1->响应1->请求2->响应2->请求3->响应3

  • 管线化,变成了以下情况(把所有的请求都打包📦发到服务端,然后服务端所有响应也打包📦响应回来)

  • 请求1->请求2->请求3->响应1->响应2->响应3

  • 管线化机制通过持久连接完成,仅HTTP/1.1支持此技术

  • 只有GET和HEAD请求可以完成管线化,而POST则有所限制

  • 初次创建连接时不应启动管线化,因为对方(服务器)不一定支持HTTP/1.1版本的协议

  • 管线化不会影响响应到来的顺序

  • HTTP/1.1要求服务器端支持管线化,但并不要求服务器端也对响应进行管线化处理,只是要求对于管线化的请求不失败即可

# 01-02 前端通信

  • 什么是同源策略及限制
  • 前后端如何通信
  • 如何创建Ajax
  • 跨域通信的几种方式

# 同源策略及限制

  • 同源限制从一个源加载的文档或脚本如何于来自另一个源的资源进行交互。这是一个用于隔离潜在恶意文件的关键的安全机制。
    • Cookie、LocalStorage和IndexDB无法读取
    • DOM无法读取
    • AJAX不能发送请求

# 前后端如何通信

  • Ajax
  • WebSocket
  • CORS

# 如何创建AJAX

  • XMLHttpRequest对象的工作流程
  • 兼容性处理
  • 事件的触发条件
  • 事件的触发顺序
function ajax(url,fnSucc,fnFaild) {
    //1.创建ajax对象
    if(window.XMLHttpRequest){
        var iAjax = new XMLHttpRequest();
    }else{
        var iAjax = new ActiveXObject("Microsoft.XMLHTTP");
    }
    //2.连接服务器
    //open(方法、文件名、异步传输)
    iAjax.open('GET',url,true);
    //3.发送请求
    iAjax.send();
    //4.接受返回
    iAjax.onreadystatechange = function () {
        //iAjax.readyState 浏览器和服务器进行到哪一步了
        if(iAjax.readyState === 4){
            if(iAjax.status === 200 || iAjax.status === 304){
                fnSucc(iAjax.responseText);
            }
            else{
                if(fnFaild){
                    fnFaild(iAjax.status);
                }
                alert('失败'+iAjax.status);
            }
        }
    }
}
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

# 跨域通讯📞的几种方式

  • JSONP
  • script 标签异步加载
<script src="http://www.abc.com/?data=name&callback=jsonp" charset="utf-8"></script>
<script>
jsonp({
  data:{
   <!-- 全局函数接受回掉信息 -->
  }
})
</script>
1
2
3
4
5
6
7
8
var util = {};

/*[function 在页面注入脚本]*/
util.createScript = function(url,charset){
  var script = document.createElement('script');
  script.setAttribute('type','text/javascript');
  charset && script.setAttribute('charset',charset);
  script.async = true;
  return script;
}

/*[function jsonp]*/
util.jsonp =function(url,onsuccess,onerror,charset){
  var callbackName = util.getName('tt_player');
  window[callbackName] = function () {
    if(onsuccess && util.isFunction(onsuccess)){
      onsuccess(arguments[0]);
    }
  };
  var script = util.createScript(url + '&callback=' + callbackName,charset);
  script.onload = script.onreadystatechange = function () {
    if(!script.readyState || /loaded|complete/.test(script.readyState)){
      script.onload = script.onreadystatechange = null;
      //移除该script 的DOM 对象
      if(script.parentNode){
        script.parentNode.removeChild(script);
      }
      //删除函数或变量
      window[callbackName] = null;
    }
  };
  script.onerror = function(){
    if(onerror && util.isFunction(onerror)){
      onerror();
    }
  };
  document.getElementsByTagName('head')[0].appendChild(script);
}
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
  • Hash
  • 嵌入页面跨域
//利用hash,场景是当前页面A通过iframe或iframe嵌入了跨域的页面 B
//A中伪代码
var B = document.getElementsByTagName('iframe');
B.src = B.src + '#' +'data';
//在B中的伪代码如下
window.onhashchange = function(){
  var data = window.location.hash;
}
1
2
3
4
5
6
7
8
  • postMessage(HTML5新增)

  • A窗口下的

//postMessage
//窗口A(http://A.com)向跨域窗口(http://B.com)发送信息
window.postMessage('data','http://B.com');

1
2
3
4
  • B窗口下的
//窗口B中监听;
window.addEventListener('message',function(event) {
  console.log(event.origin);// http://A.com
  console.log(event.source);// A window
  console.log(event.data);// data!
},false);
1
2
3
4
5
6
  • WebSocket

WebSocket-参考资料 (opens new window)

//Websocket
var ws = new WebSocket('wss://echo.websocket.org');

ws.onopen = function(evt) {
  console.log('Connection open ...');
  ws.send('Hello WebSockets!');
};

ws.onmessage = function(evt) {
  console.log('Received Message:' + evt.data);
  ws.close();
}

ws.onclose = function(evt) {
  console.log('Connection closes.');
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
  • CORS支持跨域通讯的AJAX

CORS - fetch 跨域 (opens new window)

  • 浏览器会拦截http请求,如果发现这个请求是跨域的
// url (必须), options (可选)
fetch('/some/url',{
 method: 'get',
}).then(function(response) {

}).catch(function(err) {
  //出错了;等价于then的第二个参数,但这样更好用更直观
})
1
2
3
4
5
6
7
8

# 02-01 XSS介绍

  • Cross Site Scripting
  • XSS 跨站脚本攻击
  • 运行了来自别的网站的脚本!!!

# XSS攻击

  • 在URL中加入脚本loacalhost:8080/?from``<script>alert(1)</script>``Google
  • 会弹出1
  • 如果<script>alert(1)</script>改为<script src="http://code.jquery.com/jquery-3.2.1.min.js"></script>
  • 在浏览器的控制台中windows.jQuery,jQuery.fn.jquery得到jquery版本是3.2.1
  • 于是完成了一次跨站脚本攻击XSS

# XSS攻击原理

  • 程序+数据=结果
    • <div>#{content}</div>=>content:Hello world!=><div>Hello world</div>
    • <div>#{content}</div>=>content:<script>alert(1)</script>=><div><script>alert(1)</script></div>

# XSS攻击危害

  • 从以上弹框,弹个框能上天!!!

# Scripting

  • 获取页面数据

  • 获取Cookies

  • 劫持前端逻辑

  • 发送请求

  • 。。。

  • 偷取网站任意数据

  • 偷取用户资料

  • 偷取用户密码和登录态

  • 欺骗用户

  • 。。。

# 实际案例

“微博病毒”攻击事件回顾:

2011年6月28日晚,新浪微博出现了一次比较大的XSS攻击事件。大量用户自动发送诸如:“郭美美事件的一些未注意到的细节”,“建党大业中穿帮的地方”,“让女人心动的100句诗歌”,“3D肉团团高清普通话版种子”,“这是传说中的神仙眷侣啊”,“惊爆!范冰冰艳照真流出了”等等微博和私信,并自动关注一位名为hellosamy的用户。

事件的经过线索如下:

20:14,开始有大量带V的认证用户中招转发蠕虫

20:30,某网站中的病毒页面无法访问

20:32,新浪微博中hellosamy用户无法访问

21:02,新浪漏洞修补完毕

# 02-02 XSS攻击类型

  • 基本可以分为
    • 反射型
    • 存储型

# 反射型

  • URL参数直接注入
  • 反射型XSS攻击传播,需要攻击把带有脚本的URL传播给用户 http://loacalhost:8080/?from<script>alert(1)</script>Google
  • 一般都会看出URL有问题
  • 所以攻击者会对URL做出变换 利用短网址,可以通过dwz.cn进行转换

# 存储型攻击

  • 存储到DB后读取时注入
    • xss代码会保存到网站的数据中,比如保存到数据库,当其他用户在访问到一篇文章或一个评论时,这段代码会被从数据库中读取取出来,显示在用户页面上
  • XSS攻击注入点
    • HTML节点内容
    • HTML属性
    • JavaScript代码
    • 富文本

# 1. HTML节点内容

<div>
  #{content}
</div>
<div>
  <script>
  </script>
</div>
1
2
3
4
5
6
7

# 案例

  • URL输入1"onerror="alert(1)导致的问题
<img src="#{image}" alt="图片">
<img src="1" onerror="alert(1)"/>
1
2
  1. URL中请求图片时search传递id:http://loacalhost:8080/?imgid=1
<!-- 正常情况下搜索到id为1的图片 -->
<img src="/image/1" >
1
2
  1. 当我改变URL中的请求时:http://loacalhost:8080/?imgid=1" onerror="alert(1)
<!-- 改变后的情况 -->
<img src="/image/1" onerror="alert(1)" >
<!-- " onerror="alert(1)  -->
1
2
3
  • 这样src属性被提前关闭,出现了一个onerror属性,就会出现弹框!!!

# 2. JavaScript代码

<script>
var data = "#{data}";
var data = "hello";alert(1);"";
</script>
1
2
3
4
# 案例
  • 以XXX来登录,页面上会显示“欢迎Google登录”
<script>
var from = "Google"//此段代码是传递到页面上
</script>
1
2
3
  1. URL中由数据库获取传递id:http://loacalhost:8080/?from=Google
<!-- 正常情况下搜索到id为1的图片 -->
<script>
var from = "Google"//此段代码是传递到页面上
</script>
1
2
3
4
  1. 当我改变URL中的请求时:http://loacalhost:8080/?from=Googel";alert(1);"
<!-- 改变后的情况 -->
<script>
var from = "Google";alert(1);""//此段代码是传递到页面上
</script>
<!-- ";alert(1);"  -->
1
2
3
4
5
  • 这样from属性被关闭,中间会注入了alert,就会出现弹框!!!

# 3. 富文本

  • 富文本得保留HTML
  • HTML有XSS攻击风险

# 浏览器自带的XSS攻击拦截机制

  • 设置Header X-XSS-Protection
  • 此机制只适用于参数出现在HTML内容或属性才会去拦截
  • 只适用于反射型
  • 并不是所有浏览器都支持

# 02-03 HTML内容和属性转义

# HTML内容转义

  • 尖括号的替换<,>
<div>
#{content}
</div>
1
2
3
  • 转义< &lt; 和 > &gt
var escapeHtml = function(str) {
  if(!str) return '';
  str = str.replace(/</g,'&lt;');//替换成html实体
  str = str.replace(/>/g,'&gt;');
  return str;
}
1
2
3
4
5
6

# HTML属性转义

  • HTML属性
<img class="#{class}" />
<img class="hello" onload="alert(1)" />
1
2
  • "'进行转义
var = escapeHtmlProperty = function(str) {
  if(!str) return '';
  str = str.replace(/"/g,'&quto');
  str = str.replace(/'/g,'&39');
  str = str.replace(/ /g,'&32');
  return str;
}
1
2
3
4
5
6
7

# 转义总结

var escapeHtml = function(str) {
  if(!str) return '';
  str = str.replace(/</g,'&lt;');//替换成html实体
  str = str.replace(/>/g,'&gt;');
  str = str.replace(/"/g,'&quto');
  str = str.replace(/'/g,'&39');
  str = str.replace(/ /g,'&32'); //遵循书写规则,可以不做转义
  return str;
}
1
2
3
4
5
6
7
8
9

# 新的防御方法-CSP 内容安全策略

  • 内容安全策略(CSP)用于检测和减轻用于 Web 站点的特定类型的攻击,例如 XSS 和数据注入等。
  • 该安全策略的实现基于一个称作 Content-Security-Policy 的 HTTP 首部。
  • Content Security Policy

# 限制规则

  • child-src connect-src default-src
  • font-src frame-src img-src
  • manifest-src media-src object-src
  • script-src style-src worker-src
child-src
child-src:为 web workers和其他内嵌浏览器内容定义 合法的源,例如用<frame><iframe>加载到页面的内容。
connect-src
connect-src:限制能通过脚本接口加载的URL。
default-src
default-src:为其他取指令提供备用服务fetch directives.
font-src
font-src:限制通过@font-face加载的字体源。
frame-src
frame-src: 限制通过类似<frame><iframe> 标签加载的内嵌内容源。
img-src
img-src: 限制图片和图标源
manifest-src
manifest-src : 限制 application manifest 文件源。
media-src
media-src:限制通过<audio><video> 标签加载的媒体文件源。
object-src
object-src:限制通过  <object>, <embed><applet> 标签加载源。
script-src
script-src:限制javascript 源。
style-src
style-src:限制层叠样式表文件源。
worker-src
worker-src:限制Worker, SharedWorker, 或者 ServiceWorker脚本源。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
  • 指定哪些可信,哪些不可信
  • <host-source><scheme-source>'self'
  • 'unsafe-inline''ubsafe-eval''none' -XSS攻击重点:检测页面内容(信任规则)
  • 'nonce-<base64-value>' <hash-source>-一次性凭证 后台hash传递
  • 'strit-dynamic' 后续脚本的信任

# 事例

<!-- 在Content中设置规则 -->
<meta http-equiv="Content-Security-Policy" content="default-src https:">
1
2
  • 防止XSS攻击
<!-- 这样设置页面上的脚本会失效,但是外链引用的脚本会有效 -->
<meta http-equiv="Content-Security-Policy" content="default-src self">
<!-- 但是这样还会导致页面图片失效 因为没有设置相应的'img-src'规则 ,所以他只会信赖本域下的图片 -->
<!-- 为了更精准的操作放置XSS攻击,则采用script-src 这样图片不会受到影响 -->
<meta http-equiv="Content-Security-Policy" content="script-src self">
1
2
3
4
5
  • 但是页面上有需要的呢,那就需要设置新规则nonce+随即字符串
<!-- nonce+随机🔀字符串 -->
<meta http-equiv="Content-Security-Policy" content="script-src self nonce-1234">
1
2
<script type="text/javascript" nonce="1234">

</script>
1
2
3
  • 通过计算页面hash值也可以保证界面不受XSS攻击

# 03-01 前端CSRF攻击简介

  • Cross Site Request Forgy
  • 跨站请求伪造
  • 其他网站对本网站产生的影响

# 登录对新闻评论

  • 声明:这个网站是可以匿名评论和登录评论的,网站地址http://loacalhost
  • 先在网站进行登录(把userid存储到cookie)
  • 攻击方法
<script type="text/javascript">
  document.write(`<form method="post" name="commentForm" target="csrf" action="http://loacalhost">
  <input type="hidden" name="postId" value="13">
  <textarea name="content" >来自csrf</textarea>
  </form>`);
  var irame = document.createElement('iframe');
  iframe.name = 'csrf';
  iframe.style.display = 'none';
  document.body.appendChild(iframe);

  setTimeout(() => {
    document.querySelector('[name=commentForm]').submit();//执行在攻击页面该脚本
  }, 1000);
</script>
1
2
3
4
5
6
7
8
9
10
11
12
13
14

# 03-02 CSRF攻击原理和危害

  • 1.用户登录A网站 2.A网站确认身份 3.B网站页面向A网站发起请求(带A网站的身份)
  • B网站通过手段模拟用户,向A网站后台发送数据

# 03-03 CSRF防御 - samesite

  • 通过 referer、token 或者 验证码 来检测用户提交。

  • 尽量不要在页面的链接中暴露用户隐私信息。

  • 对于用户修改删除等操作最好都使用post 操作 。

  • 避免全站通用的cookie,严格设置cookie的域。

  • 禁止第三方网站带Cookies

  • Cookie新属性same-site属性

    • SameSite = Strict 任何网站不允许携带
    • SameSite = Lax 允许链接带上Cookie,但不允许ajax不可以带Cookie,form提交也不允许带上Cookie
  • 只有Chrome支持

# 03-06 CSRF防御-referer

  • 验证referer
  • 禁止来自第三方网站的请求

# Q&A

# XSS攻击-关于URL

  • 问: 如果对URL传值的时候做encodeURICompoment,在接收的时候decodeURLComponent,是不是这样也可以放置XSS攻击呢?
  • 答:不能,因为XSS攻击的重点不在于传输过程中有没有转义,而在于最后输出的时候有没有意义。如果我传输一段脚本,它直接显示出来,这是XSS攻击,但如果我做了某种encode,但是显示的时候又decode,那实际上相当于什么也没做。类似的SQL注入也是同样原理,只要在拼接SQL的时候用的原来的值,不管传输过程中有没有转义,都没有起到防御作用