跨域
2020-04-16 11:08:19 1 举报
AI智能生成
跨域解决办法梳理
作者其他创作
大纲/内容
解决办法
CORS 机制
server.js
const express = require("express");<br>const app = express();<br>const port = 3003;<br><br>app.get("/", (req, res) => {<br> //(1)跨域资源共享(CORS) 机制<br><b><font color="#c41230"> res.append("Access-Control-Allow-Origin", "*");</font></b><br> res.send("Hello World!");<br>});<br><br>app.listen(port, () => console.log(`Example app listening on port ${port}!`));<br>
附带身份凭证的请求
Fetch 与 CORS 的一个有趣的特性是,可以基于 HTTP cookies 和 HTTP 认证信息发送身份凭证。<br>一般而言,对于跨域 XMLHttpRequest 或 Fetch 请求,浏览器不会发送身份凭证信息。<br>如果要发送凭证信息,需要 <b>设置 XMLHttpRequest 对象 </b>或者 Request 构造器的 <b>某个特殊标志位</b>。<br>
var<b> invocation = new XMLHttpRequest();</b><br>var url = 'http://bar.other/resources/credentialed-content/';<br> <br>function callOtherDomain(){<br> if(invocation) {<br> invocation.open('GET', url, true);<br><b><font color="#0076b3"> invocation.withCredentials</font> = true;</b><br> invocation.onreadystatechange = handler;<br> invocation.send(); <br> }<br>}<br><br>如果 <b>服务器端的响应 </b>中未携带 <b><font color="#0076b3">Access-Control-Allow-Credentials: true</font></b> ,<br>浏览器会拒绝接受响应。<br>
对于附带身份凭证的请求,服务器<b>不得设置 Access-Control-Allow-Origin 的值为“*”</b>。<br><br>这是因为请求的首部中携带了 Cookie 信息,如果 Access-Control-Allow-Origin 的值为“*”,请求将会失败。<br>而将 Access-Control-Allow-Origin 的值设置为 http://foo.example,则请求将成功执行。<br><br>另外,响应首部中也携带了 Set-Cookie 字段,尝试对 Cookie 进行修改。如果操作失败,将会抛出异常。<br>
后台代理
Nginx代理跨域
Nginx配置解决iconfont跨域
浏览器跨域访问js、css、img等常规静态资源被同源策略许可,但iconfont字体文件(eot|otf|ttf|woff|svg)例外,<br>此时可在nginx的静态资源服务器中加入以下配置。<br><br>location / {<br> add_header Access-Control-Allow-Origin *;<br>}<br>
Nginx反向代理接口跨域
原理
<font color="#c41230"> 同源策略是浏览器的安全策略,不是HTTP协议的一部分。<br><br>服务器端调用HTTP接口只是使用HTTP协议,不会执行JS脚本,不需要同源策略,也就不存在跨越问题。</font><br>
思路
通过nginx配置一个代理服务器(域名与domain1相同,端口不同)做跳板机,反向代理访问domain2接口,<br><br><b>并且可以顺便修改cookie中domain信息,方便当前域cookie写入,实现跨域登录。</b><br>
配置
Nginx
#proxy服务器<br>server {<br> listen 81;<br> server_name www.domain1.com;<br><br><font color="#c41230"> location / {<br><b> proxy_pass http://www.domain2.com:8080; #反向代理<br> proxy_cookie_domain www.domain2.com www.domain1.com; #修改cookie里域名</b><br> index index.html index.htm;<br><br> # 当用webpack-dev-server等中间件代理接口访问nignx时,此时无浏览器参与,故没有同源限制,下面的跨域配置可不启用<br><b> add_header Access-Control-Allow-Origin http://www.domain1.com; #当前端只跨域不带cookie时,可为*<br> add_header Access-Control-Allow-Credentials true;</b><br> }</font><br>}
前端
var xhr = new XMLHttpRequest();<br><br>// 前端开关:浏览器是否读写cookie<br><b><font color="#c41230">xhr.withCredentials = true;</font></b><br><br>// 访问nginx中的代理服务器<br>xhr.open('get', 'http://www.domain1.com:81/?user=admin', true);<br>xhr.send();<br>
Nodejs后台
var http = require('http');<br>var server = http.createServer();<br>var qs = require('querystring');<br><br>server.on('request', function(req, res) {<br> var params = qs.parse(req.url.substring(2));<br><br> // 向前台写cookie<br> res.writeHead(200, {<br> 'Set-Cookie': 'l=a123456;Path=/;Domain=www.domain2.com;HttpOnly' // HttpOnly:脚本无法读取<br> });<br><br> res.write(JSON.stringify(params));<br> res.end();<br>});<br><br>server.listen('8080');<br>console.log('Server is running at port 8080...');<br>
Nodejs中间件代理跨域
原理
node中间件实现跨域代理,原理大致与nginx相同,都是通过启一个代理服务器,实现数据的转发,<br><br>也可以通过设置cookieDomainRewrite参数修改响应头中cookie中域名,实现当前域的cookie写入,方便接口登录认证。<br>
http-proxy-middleware<br><br>这里说是可以用于线上<br>以后遇到了再说吧,<br>自己不好尝试线上环境<br>
利用node + express + http-proxy-middleware搭建一个proxy服务器。
配置
前端
var xhr = new XMLHttpRequest();<br><br>// 前端开关:浏览器是否读写cookie<br>xhr.withCredentials = true;<br><br>// 访问http-proxy-middleware代理服务器<br>xhr.open('get', 'http://www.domain1.com:3000/login?user=admin', true);<br>xhr.send();<br>
中间件服务器
var express = require('express');<br><font color="#c41230">var proxy = require('http-proxy-middleware');</font><br>var app = express();<br><br><font color="#c41230">app.use('/', proxy</font>({<br> // 代理跨域目标接口<br><font color="#c41230"><b> target: 'http://www.domain2.com:8080',</b><br><b> changeOrigin: true,</b></font><br><br> // 修改响应头信息,实现跨域并允许带cookie<br><font color="#c41230"> onProxyRes: function(proxyRes, req, res) {<br><b> res.header('Access-Control-Allow-Origin', 'http://www.domain1.com');<br> res.header('Access-Control-Allow-Credentials', 'true');</b><br> },</font><br><br> // 修改响应信息中的cookie域名<br><b><font color="#c41230"> cookieDomainRewrite: 'www.domain1.com' // 可以为false,表示不修改</font></b><br>}));<br><br>app.listen(3000);<br>console.log('Proxy server is listen at port 3000...');<br>
<font color="#c41230">开发环境下的跨域</font><br>
<font color="#c41230">利用node + webpack + webpack-dev-server代理接口跨域。</font><br><br>在开发环境下,由于vue渲染服务和接口代理服务都是webpack-dev-server同一个,<br>所以页面与代理接口之间不再跨域,无须设置headers跨域信息了。<br>
webpack.config.js部分配置:<br><br>module.exports = {<br> entry: {},<br> module: {},<br> ...<br> devServer: {<br> historyApiFallback: true,<br><b><font color="#c41230"> proxy: [{<br> context: '/login',<br> target: 'http://www.domain2.com:8080', // 代理跨域目标接口<br> changeOrigin: true,<br> secure: false, // 当代理某些https服务报错时用<br> cookieDomainRewrite: 'www.domain1.com' // 可以为false,表示不修改<br> }],</font></b><br> noInfo: true<br> }<br>}<br>
JSONP
只能够实现get请求
index.js
function callback(res) {<br> console.log("res:");<br> console.log(res);<br>}<br><br><b><font color="#c41230">let script = document.createElement("script");<br>script.src = `${url}?callback=callback`;<br>document.body.appendChild(script);</font></b><br>
server.js
const express = require("express");<br>const app = express();<br>const port = 3003;<br><b><font color="#c41230">const data = JSON.stringify({a:1,b:2});</font></b><br><br>app.get("/", (req, res) => {<br> // (2)JSONP<br><b> <font color="#c41230">let cb = req.query.callback;<br> cb && res.send(`${cb}(${data})`);</font></b><br>});<br><br>app.listen(port, () => console.log(`Example app listening on port ${port}!`));<br>
postMessage
语法
otherWindow.postMessage(message, targetOrigin, [transfer]);
otherWindow
<b>其他窗口的一个引用,比如iframe的contentWindow属性</b>、执行window.open返回的窗口对象、<br>或者是命名过或数值索引的window.frames。<br>
message
将要发送到其他 window的数据。它将会被结构化克隆算法序列化。<br><b>这意味着你可以不受什么限制的将数据对象安全的传送给目标窗口而无需自己序列化。</b><br>
targetOrigin
通过窗口的origin属性来指定哪些窗口能接收到消息事件,其值可以是字符串"*"(表示无限制)或者一个URI。<br><br><b>在发送消息的时候,如果目标窗口的协议、主机地址或端口这三者的任意一项不匹配targetOrigin提供的值,那么消息就不会被发送;<br>只有三者完全匹配,消息才会被发送。</b><br><br><b>这个机制用来控制消息可以发送到哪些窗口</b>;例如,当用postMessage传送密码时,这个参数就显得尤为重要,必须保证它的值与这条包含密码的信息的预期接受者的origin属性完全一致,来防止密码被恶意的第三方截获。<br><br><b>如果你明确的知道消息应该发送到哪个窗口,那么请始终提供一个有确切值的targetOrigin,而不是*</b>。不提供确切的目标将导致数据泄露到任何对数据感兴趣的恶意站点。<br>
示例
a.html(http://www.domain1.cn/a.html)
<b><iframe id="iframe" src="http://www.domain2.cn/b.html" style="display:none;"></iframe></b><br><script> <br> var iframe = document.getElementById('iframe');<br> iframe.onload = function() {<br> var data = {<br> name: 'aym'<br> };<br> // 向domain2传送跨域数据<br><b> iframe.contentWindow.postMessage</b>(JSON.stringify(data), 'http://www.domain2.cn');<br> };<br><br> // 接受domain2返回数据<br><b> window.addEventListener('message', function(e) {<br> alert('data from domain2 ---> ' + e.data);<br> }, false);</b><br></script><br><br>
HTMLIFrameElement.contentWindow
<font color="#c41230">contentWindow属性返回<iframe>元素的Window对象。<br><br>你可以使用这个Window对象来访问iframe的文档及其内部DOM。</font><br><br>contentWindow为只读,但是可以像操作全局Window对象一样操作其属性。<br>
b.html(http://www.domain2.cn/b.html)
<script><br> // 接收domain1的数据<br><b> window.addEventListener('message', function(e) {</b><br> alert('data from domain1 ---> ' + e.data);<br><br> var data = JSON.parse(e.data);<br> if (data) {<br> data.number = 16;<br><br> // 处理后再发回domain1<br><b> window.parent.postMessage</b>(JSON.stringify(data), 'http://www.domain1.cn');<br> }<br> }, false);<br></script><br><br>
window.parent
返回当前窗口的父窗口对象.<br><br>如果一个窗口没有父窗口,则它的 parent 属性为自身的引用.<br><br>如果当前窗口是一个 <iframe>, <object>, 或者 <frame>,则它的父窗口是嵌入它的那个窗口<br>
WebSocket协议跨域
<b>WebSocket protocol是HTML5一种新的协议。<font color="#c41230">它实现了浏览器与服务器全双工通信,同时允许跨域通讯</font></b>,<br>是server push技术的一种很好的实现。<br>原生WebSocket API使用起来不太方便,<font color="#c41230">我们使用Socket.io,它很好地封装了webSocket接口,<br>提供了更简单、灵活的接口,也对不支持webSocket的浏览器提供了向下兼容。</font><br>
配置
前端
<div>user input:<input type="text"></div><br><script src="https://cdn.bootcss.com/socket.io/2.2.0/socket.io.js"></script><br><script><br>var socket = io('http://www.domain2.com:8080');<br><br>// 连接成功处理<br>socket.on('connect', function() {<br> // 监听服务端消息<br> socket.on('message', function(msg) {<br> console.log('data from server: ---> ' + msg); <br> });<br><br> // 监听服务端关闭<br> socket.on('disconnect', function() { <br> console.log('Server socket has closed.'); <br> });<br>});<br><br>document.getElementsByTagName('input')[0].onblur = function() {<br> socket.send(this.value);<br>};<br></script><br>
Nodejs socket后台
var http = require('http');<br>var socket = require('socket.io');<br><br>// 启http服务<br>var server = http.createServer(function(req, res) {<br> res.writeHead(200, {<br> 'Content-type': 'text/html'<br> });<br> res.end();<br>});<br><br>server.listen('8080');<br>console.log('Server is running at port 8080...');<br><br>// 监听socket连接<br>socket.listen(server).on('connection', function(client) {<br> // 接收信息<br> client.on('message', function(msg) {<br> client.send('hello:' + msg);<br> console.log('data from client: ---> ' + msg);<br> });<br><br> // 断开处理<br> client.on('disconnect', function() {<br> console.log('Client socket has closed.'); <br> });<br>});<br>
document.domain + iframe
要求主域名相同
原理
<font color="#c41230">两个页面都通过js强制设置document.domain为基础主域,就实现了同域。</font>
配置
父窗口:(http://www.domain.com/a.html)
<iframe id="iframe" src="http://child.domain.com/b.html"></iframe><br><script><br> document.domain = 'domain.com';<br> var user = 'admin';<br></script><br>
子窗口:(http://child.domain.com/b.html)
<script><br><font color="#5c5c5c"> document.domain = 'domain.com';<br> // 获取父窗口中变量</font><br><b style="color: rgb(196, 18, 48);"> alert('get js data from parent ---> ' + window.parent.user);</b><br></script><br>
window.name + iframe
<font color="#c41230">window.name属性的独特之处:name值在不同的页面(甚至不同域名)加载后依旧存在,<br>并且可以支持非常长的 name 值(2MB)。<br></font><br><font color="#c41230">通过iframe的src属性由外域转向本地域,跨域数据即由iframe的window.name从外域传递到本地域。</font><br>这个就巧妙地绕过了浏览器的跨域访问限制,但同时它又是安全操作。<br>
配置
a.html:(http://www.domain1.com/a.html)
var proxy = function(url, callback) {<br> var state = 0;<br> var iframe = document.createElement('iframe');<br><br> // 加载跨域页面<br> iframe.src = url;<br><br><font color="#c41230"><b> // onload事件会触发2次,第1次加载跨域页,并留存数据于window.name</b></font><br> iframe.onload = function() {<br> if (state === 1) {<br><b> // 第2次onload(同域proxy页)成功后,读取同域window.name中数据<br> callback(iframe.contentWindow.name);<br> destoryFrame();</b><br><br> } else if (state === 0) {<br><b><font color="#c41230"> // 第1次onload(跨域页)成功后,切换到同域代理页面<br> iframe.contentWindow.location = 'http://www.domain1.com/proxy.html';</font><br> state = 1;</b><br> }<br> };<br><br> document.body.appendChild(iframe);<br><br> // 获取数据以后销毁这个iframe,释放内存;这也保证了安全(不被其他域frame js访问)<br> function destoryFrame() {<br> iframe.contentWindow.document.write('');<br> iframe.contentWindow.close();<br> document.body.removeChild(iframe);<br> }<br>};<br><br>// 请求跨域b页面数据<br>proxy('http://www.domain2.com/b.html', function(data){<br> alert(data);<br>});<br>
proxy.html:(http://www.domain1.com/proxy.html)
中间代理页,与a.html同域,内容为空即可。
b.html:(http://www.domain2.com/b.html)
<script><br> window.name = 'This is domain2 data!';<br></script><br>
location.hash + iframe
原理
a欲与b跨域相互通信,通过中间页c来实现。 三个页面,<br><font color="#c41230">不同域之间利用iframe的location.hash传值,相同域之间直接js访问来通信。</font><br>
实现
A域:a.html -> B域:b.html -> A域:c.html,<font color="#c41230">a与b不同域只能通过hash值单向通信,<br>b与c也不同域也只能单向通信,但c与a同域,所以c可通过parent.parent访问a页面所有对象。</font><br>
配置
a.html:(http://www.domain1.com/a.html)
<iframe id="iframe" src="http://www.domain2.com/b.html" style="display:none;"></iframe><br><script><br> var iframe = document.getElementById('iframe');<br><br><font color="#c41230"> // 向b.html传hash值<br> setTimeout(function() {<br> iframe.src = iframe.src + '#user=admin';<br> }, 1000);<br> <br> // 开放给同域c.html的回调方法<br> function onCallback(res) {<br> alert('data from c.html ---> ' + res);<br> }</font><br></script><br>
b.html:(http://www.domain2.com/b.html)
<iframe id="iframe" src="http://www.domain1.com/c.html" style="display:none;"></iframe><br><script><br> var iframe = document.getElementById('iframe');<br><br><font color="#c41230"> // 监听a.html传来的hash值,再传给c.html<br>// 我的理解,这里其实 b 页面可以做一些处理后再传递给 c<br>// b 页面不做处理的话,就没有必要绕这么一大圈了,那就是 a 页面自己传数据给自己了<br> window.onhashchange = function () {<br> iframe.src = iframe.src + location.hash;<br> };</font><br></script><br>
c.html:(http://www.domain1.com/c.html)
<script><br> // 监听b.html传来的hash值<br><font color="#c41230"> window.onhashchange = function () {<br> // 再通过操作同域a.html的js回调,将结果传回<br> window.parent.parent.onCallback('hello: ' + location.hash.replace('#user=', ''));<br> };</font><br></script><br>
参考链接
https://segmentfault.com/a/1190000011145364
0 条评论
下一页