乐趣区

JavaScript 关于’跨域’问题的延伸-CORS

同源策略
在 web 浏览器中,同源策略 限制了从同一个源加载的文档或脚本如何与来自另一个源的资源进行交互。这是一个用于隔离潜在恶意文件的重要安全机制。
如果两个页面的 1. 协议 2. 端口(如果有指定)3. 域名都相同,则两个页面具有相同的源。
举例说明 http://store.example.com/dir/page.html 同源检测的示例:

URL
结果
原因

http://store.example.com/index.html
成功

http://store.example.com/dir/another.html
成功

https://store.example.com/index.html
失败
不同协议 (https 和 http)

http://store.example.com:81/index.html
失败
不同端口 (81 和 80)

http://news.example.com/index.html
失败
不同域名 (news 和 store)

跨源场景:

跨源网络访问 同源策略控制了不同源之间的交互。问:如何允许跨源访问?答:使用 CORS 允许跨源访问。场景:由浏览器发起的跨域 HTTP 请求都用的这个,例如大部分的前端发出接口请求

跨源脚本 API 访问 Javascript 的 APIs 中,允许文档间直接相互引用。但是当两个文档的源不同时,一些引用方式将对 API 对象的访问添加限制 为了在不同源中文档进一步交流,可以使用 window.postMessage 场景:例如 <iframe> 嵌套的时候,父子页面的通信

跨源数据存储访问 存储在浏览器中的数据,如 localStorage 和 IndexedDB,以源进行分割。每个源都拥有自己单独的存储空间,一个源中的 Javascript 脚本不能对属于其它源的数据进行读写操作。
了解 CORS
CORS 是什么?
MDN 的网站给出了这样的解释:
CORS(Cross-Origin Resource Sharing,跨域资源共享)是一个系统,它由一系列传输的 HTTP 头组成,这些 HTTP 头决定浏览器是否阻止前端 JavaScript 代码获取跨域请求的响应。CORS 给了 web 服务器这样的权限,即服务器可以选择,允许跨域请求访问到它们的资源。
可以这样理解:CORS 赋予服务端(通常所说的后端)一个权力,让它可以自己控制哪一些浏览器的请求可以访问到它的资源,来解决跨域问题。
附:所有的 CORS 头

HTTP 头
功能

Access-Control-Allow-Origin
指示请求的资源能共享给哪些域。

Access-Control-Allow-Credentials
指示当请求的凭证标记为 true 时,是否响应该请求。

Access-Control-Allow-Headers
用在对预请求的响应中,指示实际的请求中可以使用哪些 HTTP 头。

Access-Control-Allow-Methods
指定对预请求的响应中,哪些 HTTP 方法允许访问请求的资源。

Access-Control-Expose-Headers
指示哪些 HTTP 头的名称能在响应中列出。

Access-Control-Max-Age
指示预请求的结果能被缓存多久。

Access-Control-Request-Headers
用于发起一个预请求,告知服务器正式请求会使用那些 HTTP 头。

Access-Control-Request-Method
用于发起一个预请求,告知服务器正式请求会使用哪一种 HTTP 请求方法。

Origin
指示获取资源的请求是从什么域发起的。

CORS 功能概述
TL;DR 规范要求,对那些可能对服务器数据产生副作用的 HTTP 请求方法(特别是 GET 以外的 HTTP 请求,或者搭配某些 MIME 类型的 POST 请求),浏览器必须首先使用 OPTIONS 方法发起一个预检请求(preflight request),从而获知服务端是否允许该跨域请求。服务器确认允许之后,才发起实际的 HTTP 请求。在预检请求的返回中,服务器端也可以通知客户端,是否需要携带身份凭证(包括 Cookies 和 HTTP 认证相关数据)。
一句话概述:非简单请求时,会先发送预检请求,允许后再发送实际请求
附:node-express 框架下,服务端的跨域设置
app.all(‘*’, function (req, res, next) {
res.header(‘Access-Control-Allow-Origin’, ‘*’);
res.header(‘Access-Control-Allow-Headers’, ‘Content-Type’);
res.header(‘Access-Control-Allow-Methods’, ‘*’);
next();
});

CORS 关于 Cookie
CORS 请求默认不发送 Cookie 和 HTTP 认证信息,如果想要知道用 cookie 就要注意 3 点
Client 端 new XMLHttpRequest() 中 withCredentials 设置为 trueSerive 端 HTTP 头 Access-Control-Allow-Credentials 设置为 trueSerive 端 HTTP 头 Access-Control-Allow-Origin 不能设为星号,必须指定明确的、与请求网页一致的域名
所以上面的例子要想发送 cookie
// service 端
`res.header(‘Access-Control-Allow-Credentials’, true);`
`res.header(‘Access-Control-Allow-Origin’, ‘ 具体的域名 ’);`
// client 端
Jquery `ajax()` `xhrFields: {withCredentials: true}`
Axios `axios.defaults.withCredentials = true`
CORS 的简单请求
上面讲了简单请求,但那些才是简单请求呢?答:若不会触发 CORS 的预检请求,称这样的请求为“简单请求”以下为简单请求:

HTTP Method 组成只能是以下几种

GET
POST
HEAD

HTTP Headers 组成

Accept
Accept-Language
Content-Language
Last-Event-ID

Content-Type 只包含以下类型 (form 表单请求)

application/x-www-form-urlencoded
multipart/form-data
text/plain

注:只有同时满足以上两个条件时, 才是简单请求, 否则为非简单请求
CORS 预检请求又是什么?
前面说了这么多预检请求,我们来讲一讲什么是预检请求:如果我们在 client 端发送请求时,例如:
// 原生
var invocation = new XMLHttpRequest();

invocation.setRequestHeader(‘X-EXAMPLE’, ‘xixihaha’);
invocation.setRequestHeader(‘Content-Type’, ‘application/xml’);

// axios
axios.defaults.headers[‘X-EXAMPLE’] = ‘xixihaha’;
axios.defaults.headers[‘Content-Type’] = ‘application/xml’;

POST 请求发送一个 XML 文档,该请求包含了一个自定义的请求首部字段(X-EXAMPLE: xixihaha)。另外,该请求的 Content-Type 为 application/xml。因此,该请求需要首先发起“预检请求”。
server 端 的 HTTP 头设置
Access-Control-Allow-Origin: ‘ 具体的域名 ’
Access-Control-Allow-Methods: POST, GET, OPTIONS // 可包含的参数
Access-Control-Allow-Headers: X-PINGOTHER, Content-Type // 允许的首部字段
Access-Control-Max-Age: 86400
非简单请求和简单请求无异,如果浏览器的预检请求被服务器接受,则发送实际请求,未被接受则拒绝请求。
其他

JSONP 动态创建 script 标签,然后利用 script 的 src 不受同源策略约束来跨域获取数据
function addScriptTag() {
var script = document.createElement(“script”);
script.src = “http://foo.example?callback=handleResponse”;
document.body.appendChild(script);
}

function handleResponse() {
console.log(‘ 跨域数据 ’);
};
以下跨域方案不做过多解释
上文提到的 postMessage()

nginx 转发,即架设服务器代理
window.name
document.domain
参考
阮一峰的网络日志 MDN web docs CORS

退出移动版