跨域资源共享- CORS
(AKA跨域AJAX请求)是大多数Web开发人员可能遇到的问题,根据Same-Origin-Policy,浏览器将客户端JavaScript限制在安全沙箱中,通常JS无法直接与远程服务器通信来自其他域。过去,开发人员创建了许多棘手的方法来实现跨域资源请求,最常用的方法是:
- 使用Flash / Silverlight或服务器端作为“代理”与远程通信。
- 带有填充的JSON(JSONP)。
- 将远程服务器嵌入到iframe中,并通过fragment或window.name进行通信,请参见此处。
这些棘手的方式或多或少都存在一些问题,例如,如果开发人员只是简单地“评估”,JSONP可能会导致安全漏洞;而上面的#3,虽然可行,但两个域之间应该建立严格的契约,既不灵活也不优雅恕我直言:)
W3C引入了跨域资源共享(CORS)作为标准解决方案,以提供安全,灵活和推荐的标准方式来解决此问题。
机制
从较高的层次上,我们可以简单地认为CORS是域A的客户端AJAX调用与域B上托管的页面之间的合同,典型的跨域请求/响应为:
DomainA AJAX请求标头
Host DomainB.com
User-Agent Mozilla/5.0 (Windows NT 6.1; WOW64; rv:2.0) Gecko/20100101 Firefox/4.0
Accept text/html,application/xhtml+xml,application/xml;q=0.9,*/*;q=0.8,application/json
Accept-Language en-us;
Accept-Encoding gzip, deflate
Keep-Alive 115
Origin http://DomainA.com
DomainB响应标头
Cache-Control private
Content-Type application/json; charset=utf-8
Access-Control-Allow-Origin DomainA.com
Content-Length 87
Proxy-Connection Keep-Alive
Connection Keep-Alive
我在上面标记的蓝色部分是核心事实,“ Origin”请求标头“指示跨域请求或预检请求的来源”,“ Access-Control-Allow-Origin”响应标头指示此页面允许来自远程请求DomainA(如果值为*,则表示允许来自任何域的远程请求)。
正如我上面提到的,W3建议浏览器在提交实际的跨域HTTP请求之前实现一个“ 预检请求 ”,简而言之,它是一个HTTP OPTIONS
请求:
OPTIONS DomainB.com/foo.aspx HTTP/1.1
如果foo.aspx支持OPTIONS HTTP动词,则它可能返回如下响应:
HTTP/1.1 200 OK
Date: Wed, 01 Mar 2011 15:38:19 GMT
Access-Control-Allow-Origin: http://DomainA.com
Access-Control-Allow-Methods: POST, GET, OPTIONS, HEAD
Access-Control-Allow-Headers: X-Requested-With
Access-Control-Max-Age: 1728000
Connection: Keep-Alive
Content-Type: application/json
仅当响应包含“ Access-Control-Allow-Origin”并且其值为“ *”或包含提交CORS请求的域时,通过满足此强制性条件,浏览器将提交实际的跨域请求,并缓存结果在“ 预检结果缓存 ”中。
我三年前写了关于CORS的博客:AJAX跨源HTTP请求
http://siteA/MyCode.js
。