跨域问题分析,及解决方案

为何有跨域问题

由于浏览器的同源策略,致使了跨域问题的出现。javascript

1. 什么是同源策略

同源策略限制了从同一个源加载的文档或脚本如何与来自另外一个源的资源进行交互。这是一个用于隔离潜在恶意文件的重要安全机制。html

2. 为何须要同源策略

出于安全缘由,浏览器限制从脚本内发起的跨源HTTP请求。 例如XMLHttpRequest遵循同源策略。 这意味着使用这些API的Web应用程序只能从加载应用程序的同一个域请求HTTP资源,除非使用CORS头文件或其余跨域方法。java

没有同源策略的两大安全隐患:node

  1. 针对接口请求 常见的场景为:cookie获取,CSRF攻击:(Cross-site request forgery)跨站请求伪造

One Day,当你兴致勃勃在某宝上准备双11的买买买,网购物车里面各类加,这个时候弹出了你爱豆的新闻,那必须得关注,看一看你家爱豆是否是也要介绍一下另外一半,因而就点击链接进去看了,你看的过程当中,也许这个网站暗地里就作了些什么不可描述的事情!json

好比说因为没有同源策略的限制,它向某宝发起了请求!由于你在登陆某宝时“服务端验证经过后会在响应头加入Set-Cookie字段,而后下次再发请求的时候,浏览器会自动将cookie附加在HTTP请求的头字段Cookie中”,这样一来,这个不法网站就至关于登陆了你的帐号,能够随心所欲了!canvas

  1. 针对Dom 假设一个场景须要用户先填写用户名密码等登陆信息进行身份验证,才能进行接下来的操做,当你输入用户名密码登陆成功后,你的帐号密码就被盗了,那酸爽~~~ 那这个钓鱼网站作了什么呢?
// HTML
    <iframe name="qq" src="www.qq.com"></iframe>
    // JS
    // 因为没有同源策略的限制,钓鱼网站能够直接拿到别的网站的Dom
    let iframe = window.frames['qq']
    let userInput = iframe.document.getElementById('帐号输入框'),
        passInput = iframe.document.getElementById('密码输入框');
    
    //dom都拿到了,帐号和密码不就是一件很容易的事情了么~~~
复制代码

所以同源策略确实能规避一些危险,不是说有了同源策略就安全,只是说同源策略是一种浏览器最基本的安全机制,毕竟能提升一点攻击的成本。其实没有刺不穿的盾,只是攻击的成本和攻击成功后得到的利益成不成正比。后端

3. 怎么判断URL是否同源

若是两个页面的协议端口(若是有指定)和域名都相同,则两个页面具备相同的源。api

同源的断定: 以http://www.example.com/dir/page.html为例,如下表格指出了不一样形式的连接是否与其同源:(缘由里未申明不一样的属性即说明其与例子里的原连接对应的属性相同)跨域

连接 结果 缘由
http://www.example.com/dir/page2.html 同协议同域名同端口
http://www.example.com/dir2/other.html 同协议同域名同端口
http://www.example.com:81/dir/other.html 端口不一样
https://www.example.com/dir/other.html 协议不一样端口不一样
http://en.example.com/dir/other.html 域名不一样
http://example.com/dir/other.html 域名不一样(要求精确匹配)
http://v2.www.example.com/dir/other.html 域名不一样(要求精确匹配)
http://www.example.com:80/dir/other.html 不肯定 取决于浏览器的实现方式

tips: 主域名与子域名的区别

主域名:由两个或两个以上的字母构成,中间由点号隔开,整个域名只有1个点号,惟一的 子域名:是在主域名之下的域名,域名内容会有多个点号浏览器

例如:https://www.baidu.com/ 协议:https:// 服务器名称:www 主域名:baidu.com 子域名(子域名包括服务器名称www + 主域名baidu.com):www.baidu.com

接口跨域的正确打开方式

目前经常使用的解决跨域问题的三种方式:

  • jsonp:只能发送GET请求
  • iframe + form
  • CORS:跨域资源共享(Cross-origin resource sharing);

1. JSONP

在HTML标签里,一些标签好比script、img这样的获取资源的标签是没有跨域限制的,利用这一点,咱们能够这样干:

<!DOCTYPE html>
<html>
  <head>
    <meta charset="utf-8">
  </head>
  <body>
    <script type='text/javascript'> // 后端返回直接执行的方法,至关于执行这个方法,因为后端把返回的数据放在方法的参数里,因此这里能拿到res。 window.jsonpCb = function (res) { console.log(res) } </script>
    <script src='http://localhost:9871/api/jsonp?msg=helloJsonp&cb=jsonpCb' type='text/javascript'></script>
  </body>
</html>

复制代码

由上可知JSNOP只能发送GET请求,不能发送POST,本质上经过script标签去加载资源就是GET

2. Iframe + Form

上代码

const requestPost = ({url, data}) => {
  // 首先建立一个用来发送数据的iframe.
  const iframe = document.createElement('iframe')
  iframe.name = 'iframePost'
  iframe.style.display = 'none'
  document.body.appendChild(iframe)
  const form = document.createElement('form')
  const node = document.createElement('input')
  // 注册iframe的load事件处理程序,若是你须要在响应返回时执行一些操做的话.
  iframe.addEventListener('load', function () {
    console.log('post success')
  })

  form.action = url
  // 在指定的iframe中执行form的action url
  form.target = iframe.name
  form.method = 'post'
  for (let name in data) {
    node.name = name
    node.value = data[name].toString()
    form.appendChild(node.cloneNode())
  }
  // 表单元素须要添加到主文档中.
  form.style.display = 'none'
  document.body.appendChild(form)
  form.submit()

  // 表单提交后,就能够删除这个表单,不影响下次的数据发送.
  document.body.removeChild(form)
}
// 使用方式
requestPost({
  url: 'http://localhost:9871/api/iframePost',
  data: {
    msg: 'helloIframePost'
  }
})
复制代码

3. CORS

CORS全称跨域资源共享(Cross-origin resource sharing),该机制容许Web应用服务器进行跨域访问控制,从而使跨域数据传输得以安全进行。浏览器支持在API容器中(例如XMLHttpRequest或Fetch)使用CORS,以下降跨域HTTP请求所带来的风险。CORS须要客户端和服务器同时支持,目前,全部浏览器都支持该机制(微企即采用这种方式)。

IE 10+ 提供了对规范的完整支持,但在较早版本(8 和 9)中,CORS 机制是借由 XDomainRequest 对象完成的。

CORS规范要求,对那些可能对服务器数据产生反作用的HTTP请求方法(特别是GET之外的HTTP请求,或者搭配某些 MIME 类型的POST请求),浏览器必须首先使用OPTIONS方法发起一个预检请求(preflight request),从而获知服务端是否容许该跨域请求。服务器确认容许以后,才发起实际的HTTP请求。在预检请求的返回中,服务器端也能够通知客户端,是否须要携带身份凭证(包括Cookies和HTTP认证相关数据)。

整个CORS通讯过程,都是浏览器自动完成,不须要用户参与。对于开发者来讲,CORS通讯与同源的AJAX通讯没有差异,代码彻底同样。浏览器一旦发现AJAX请求跨源,就会自动添加一些附加的头信息,有时还会多出一次附加的请求,但用户不会有感受。

所以,实现CORS通讯的关键是服务器。只要服务器实现了CORS接口,就能够跨源通讯。

CORS标准容许在下列场景中使用跨域http请求:

  • 由 XMLHttpRequest 或 Fetch 发起的跨域 HTTP 请求。
  • Web 字体 (CSS 中经过 @font-face 使用跨域字体资源), 所以,网站就能够发布 TrueType
  • 字体资源,并只容许已受权网站进行跨站调用。
  • WebGL 贴图
  • 使用 drawImage 将 Images/video 画面绘制到 canvas
  • 样式表(使用 CSSOM)
  • Scripts (未处理的异常)

浏览器将CORS请求分红两类:简单请求(simple request)和非简单请求(not-so-simple request)。

  • 使用下列方法之一: * GET * HEAD * POST
  • Fetch 规范定义了对 CORS安全的首部字段集合,不得人为设置该集合以外的其余首部字段。该集合为:
    • Accept
    • Accept-Language
    • Content-Language
    • Last-Event-ID
    • Content-Type (须要注意额外的限制)
  • Content-Type 的值仅限于下列三者之一:
    • text/plain
    • multipart/form-data
    • application/x-www-form-urlencoded 只要同时知足如下两大条件,就属于简单请求。

凡是不一样时知足上面两个条件,就属于非简单请求。 浏览器对这两种请求的处理,是不同的。

1. 简单请求

对于简单请求,浏览器直接发出CORS请求。具体来讲,就是在头信息之中,增长一个Origin字段。 下面是一个例子,浏览器发现此次跨源AJAX请求是简单请求,就自动在头信息之中,添加一个Origin字段。

GET /cors HTTP/1.1
Origin: http://api.bob.com
Host: api.alice.com
Accept-Language: en-US
Connection: keep-alive
User-Agent: Mozilla/5.0...
复制代码

上面的头信息中,Origin字段用来讲明,本次请求来自哪一个源(协议 + 域名 + 端口)。服务器根据这个值,决定是否赞成此次请求。

若是Origin指定的源,不在许可范围内,服务器会返回一个正常的HTTP回应。浏览器发现,这个回应的头信息没有包含Access-Control-Allow-Origin字段(详见下文),就知道出错了,从而抛出一个错误,被XMLHttpRequestonerror回调函数捕获。注意,这种错误没法经过状态码识别,由于HTTP回应的状态码有多是200。

若是Origin指定的域名在许可范围内,服务器返回的响应,会多出几个头信息字段。

Access-Control-Allow-Origin: http://api.bob.com //必填
Access-Control-Allow-Credentials: true //非必填
Access-Control-Expose-Headers: FooBar //非必填
Content-Type: text/html; charset=utf-8
复制代码

上面的头信息之中,有三个与CORS请求相关的字段,都以Access-Control-开头。

(1)Access-Control-Allow-Origin

该字段是必须的。它的值要么是请求时Origin字段的值,要么是一个*,表示接受任意域名的请求。

(2)Access-Control-Allow-Credentials

该字段可选。它的值是一个布尔值,表示是否容许发送Cookie。默认状况下,Cookie不包括在CORS请求之中。设为true,即表示服务器明确许可,Cookie能够包含在请求中,一块儿发给服务器。这个值也只能设为true,若是服务器不要浏览器发送Cookie,删除该字段便可。

(3)Access-Control-Expose-Headers

该字段可选。CORS请求时,XMLHttpRequest对象的getResponseHeader()方法只能拿到6个基本字段:Cache-ControlContent-LanguageContent-TypeExpiresLast-ModifiedPragma。若是想拿到其余字段,就必须在Access-Control-Expose-Headers里面指定。上面的例子指定,getResponseHeader('FooBar')能够返回FooBar字段的值。

3.2 withCredentials 属性 上面说到,CORS请求默认不发送CookieHTTP认证信息。若是要把Cookie发到服务器,一方面要服务器赞成,指定Access-Control-Allow-Credentials字段。

Access-Control-Allow-Credentials: true
复制代码

另外一方面,开发者必须在AJAX请求中打开withCredentials属性。

var xhr = new XMLHttpRequest();
xhr.withCredentials = true;
复制代码

不然,即便服务器赞成发送Cookie,浏览器也不会发送。或者,服务器要求设置Cookie,浏览器也不会处理。

可是,若是省略withCredentials设置,有的浏览器仍是会一块儿发送Cookie。这时,能够显式关闭withCredentials

xhr.withCredentials = false;
复制代码

须要注意的是,若是要发送Cookie,Access-Control-Allow-Origin就不能设为星号,必须指定明确的、与请求网页一致的域名。同时,Cookie依然遵循同源政策,只有用服务器域名设置的Cookie才会上传,其余域名的Cookie并不会上传,且(跨源)原网页代码中的document.cookie也没法读取服务器域名下的Cookie。

2. 非简单请求(预检请求)

2.1 预检请求

非简单请求是那种对服务器有特殊要求的请求,好比请求方法是PUTDELETE,或者Content-Type字段的类型是application/json

非简单请求的CORS请求,会在正式通讯以前,增长一次HTTP查询请求,称为"预检"请求(preflight)。

浏览器先询问服务器,当前网页所在的域名是否在服务器的许可名单之中,以及可使用哪些HTTP动词和头信息字段。只有获得确定答复,浏览器才会发出正式的XMLHttpRequest请求,不然就报错。

下面是一段浏览器的JavaScript脚本。

var url = 'http://api.alice.com/cors';
var xhr = new XMLHttpRequest();
xhr.open('PUT', url, true);
xhr.setRequestHeader('X-Custom-Header', 'value');
xhr.send();
复制代码

上面代码中,HTTP请求的方法是PUT,而且发送一个自定义头信息X-Custom-Header

浏览器发现,这是一个非简单请求,就自动发出一个"预检"请求,要求服务器确承认以这样请求。下面是这个"预检"请求的HTTP头信息。

OPTIONS /cors HTTP/1.1
Origin: http://api.bob.com
Access-Control-Request-Method: PUT
Access-Control-Request-Headers: X-Custom-Header
Host: api.alice.com
Accept-Language: en-US
Connection: keep-alive
User-Agent: Mozilla/5.0...
复制代码

"预检"请求用的请求方法是OPTIONS,表示这个请求是用来询问的。头信息里面,关键字段是Origin,表示请求来自哪一个源。

除了Origin字段,"预检"请求的头信息包括两个特殊字段。

(1)Access-Control-Request-Method

该字段是必须的,用来列出浏览器的CORS请求会用到哪些HTTP方法,上例是PUT

(2)Access-Control-Request-Headers

该字段是一个逗号分隔的字符串,指定浏览器CORS请求会额外发送的头信息字段,上例是X-Custom-Header

2.2 预检请求的回应

服务器收到"预检"请求之后,检查了OriginAccess-Control-Request-MethodAccess-Control-Request-Headers字段之后,确认容许跨源请求,就能够作出回应。

HTTP/1.1 200 OK
Date: Mon, 01 Dec 2008 01:15:39 GMT
Server: Apache/2.0.61 (Unix)
Access-Control-Allow-Origin: http://api.bob.com
Access-Control-Allow-Methods: GET, POST, PUT
Access-Control-Allow-Headers: X-Custom-Header
Content-Type: text/html; charset=utf-8
Content-Encoding: gzip
Content-Length: 0
Keep-Alive: timeout=2, max=100
Connection: Keep-Alive
Content-Type: text/plain
复制代码

上面的HTTP回应中,关键的是Access-Control-Allow-Origin字段,表示http://api.bob.com能够请求数据。该字段也能够设为星号,表示赞成任意跨源请求。

Access-Control-Allow-Origin: *
复制代码

若是浏览器否认了"预检"请求,会返回一个正常的HTTP回应,可是没有任何CORS相关的头信息字段。这时,浏览器就会认定,服务器不一样意预检请求,所以触发一个错误,被XMLHttpRequest对象的onerror回调函数捕获。控制台会打印出以下的报错信息。

XMLHttpRequest cannot load http://api.alice.com.
Origin http://api.bob.com is not allowed by Access-Control-Allow-Origin.
复制代码

服务器回应的其余CORS相关字段以下。

Access-Control-Allow-Methods: GET, POST, PUT
Access-Control-Allow-Headers: X-Custom-Header
Access-Control-Allow-Credentials: true
Access-Control-Max-Age: 1728000
复制代码

(1)Access-Control-Allow-Methods

该字段必需,它的值是逗号分隔的一个字符串,代表服务器支持的全部跨域请求的方法。注意,返回的是全部支持的方法,而不单是浏览器请求的那个方法。这是为了不屡次"预检"请求。

(2)Access-Control-Allow-Headers

若是浏览器请求包括Access-Control-Request-Headers字段,则Access-Control-Allow-Headers字段是必需的。它也是一个逗号分隔的字符串,代表服务器支持的全部头信息字段,不限于浏览器在"预检"中请求的字段。

(3)Access-Control-Allow-Credentials

该字段与简单请求时的含义相同。

(4)Access-Control-Max-Age

该字段可选,用来指定本次预检请求的有效期,单位为秒。上面结果中,有效期是20天(1728000秒),即容许缓存该条回应1728000秒(即20天),在此期间,不用发出另外一条预检请求。

2.3 浏览器的正常请求和回应 一旦服务器经过了"预检"请求,之后每次浏览器正常的CORS请求,就都跟简单请求同样,会有一个Origin头信息字段。服务器的回应,也都会有一个Access-Control-Allow-Origin头信息字段。

下面是"预检"请求以后,浏览器的正常CORS请求。

PUT /cors HTTP/1.1
Origin: http://api.bob.com
Host: api.alice.com
X-Custom-Header: value
Accept-Language: en-US
Connection: keep-alive
User-Agent: Mozilla/5.0...
复制代码

上面头信息的Origin字段是浏览器自动添加的。

下面是服务器正常的回应。

Access-Control-Allow-Origin: http://api.bob.com
Content-Type: text/html; charset=utf-8
复制代码

上面头信息中,Access-Control-Allow-Origin字段是每次回应都一定包含的。

HTTP 响应首部字段

1. Access-Control-Allow-Origin

响应首部中能够携带一个 Access-Control-Allow-Origin 字段,其语法以下:

Access-Control-Allow-Origin: | *

其中,origin 参数的值指定了容许访问该资源的外域URI。对于不须要携带身份凭证的请求,服务器能够指定该字段的值为通配符,表示容许来自全部域的请求。

例如,下面的字段值将容许来自 mozilla.com 的请求:

Access-Control-Allow-Origin: mozilla.com

若是服务端指定了具体的域名而非“*”,那么响应首部中的 Vary 字段的值必须包含 Origin。这将告诉客户端:服务器对不一样的源站返回不一样的内容。

JSONP只支持GET请求,CORS支持全部类型的HTTP请求。JSONP的优点在于支持老式浏览器,以及能够向不支持CORS的网站请求数据。

DOM级别跨域

document.domain

页面可能会因某些限制而改变他的源。脚本能够将document.domain的值设置为其当前域或其当前域的超级域。若是将其设置为其当前域的超级域,则较短的域将用于后续源检查。 假设http://store.company.com/dir/other.html文档中的一个脚本执行如下语句:

document.domain = "company.com";
复制代码

这条语句执行以后,页面将会成功地经过对http://company.com/dir/page.html的同源检测(假设http://company.com/dir/page.html将其document.domain设置为company.com,以代表它但愿容许这样作 - 更多有关信息,请参阅 document.domain)。然而,company.com不能设置document.domainothercompany.com,由于它不是 company.com 的超级域

canvas中getImageData,toDataURL跨域

经过添加cross-origin属性便可解决getImageDatatoDataURL跨域问题,具体参见canvas跨域

相关文章
相关标签/搜索