前端跨域大总结

跨域这两个字就像一块狗皮膏药一样黏在每个前端开辟者身上,不论你在事变上或许口试中无可防止会碰到这个题目。为了敷衍口试,我每次都随意背几个计划,也不晓得为何要如许干,横竖面完就能够够扔了,我想事变上也不会用到那末多杂乱无章的计划。到了真正事变,开辟环境有webpack-dev-server搞定,上线了效劳端的大佬们也会配好,配了什么我不论,横竖不会跨域就是了。日子也就这么混过去了,终究有一天,我以为不能再继承如许混下去了,我肯定要完全搞懂这个东西!因而就有了这篇文章。

要控制跨域,起首要晓得为何会有跨域这个题目涌现

确切,我们这类搬砖工人就是为了混口饭吃嘛,好好的调个接口告诉我跨域了,这类障碍我们轻松搬砖的事变真恶心!为何会跨域?是谁在搞事变?为了找到这个题目标始作俑者,请点击阅读器的同源战略。
这么官方的东西真难明,没紧要,最少你晓得了,因为阅读器的同源战略致使了跨域,就是阅读器在搞事变。
所以,阅读器为何要搞事变?就是不想给好日子我们过?关于如许的诘责,阅读器甩锅道:“同源战略限定了从同一个源加载的文档或剧本怎样与来自另一个源的资本举行交互。这是一个用于断绝潜伏歹意文件的主要平安机制。”
这么官方的话术真难明,没紧要,最少你晓得了,好像这是个平安机制。
所以,终究为何须要如许的平安机制?如许的平安机制处置惩罚了什么题目?别急,让我们继承研讨下去。

没有同源战略限定的两大风险场景

据我相识,阅读器是从两个方面去做这个同源战略的,一是针对接口的要求,二是针对Dom的查询。试想一下没有如许的限定上述两种行动有什么风险。

没有同源战略限定的接口要求

有一个小小的东西叫cookie人人应当晓得,平常用来处置惩罚登录等场景,目标是让效劳端晓得谁发出的此次要求。假如你要求了接口举行登录,效劳端考证经由过程后会在相应头到场Set-Cookie字段,然后下次再发要求的时刻,阅读器会自动将cookie附加在HTTP要求的头字段Cookie中,效劳端就能够晓得这个用户已登录过了。晓得这个以后,我们来看场景:
1.你预备去清空你的购物车,因而打开了买买买网站www.maimaimai.com,然后登录胜利,一看,购物车东西这么少,不可,还得买多点。
2.你在看有什么东西买的过程当中,你的好基友发给你一个链接www.nidongde.com,一脸yin笑地跟你说:“你懂的”,你坚决坚决打开了。
3.你饶有兴致地阅读着www.nidongde.com,谁知这个网站暗地里做了些不可形貌的事变!因为没有同源战略的限定,它向www.maimaimai.com提议了要求!智慧的你肯定想到上面的话“效劳端考证经由过程后会在相应头到场Set-Cookie字段,然后下次再发要求的时刻,阅读器会自动将cookie附加在HTTP要求的头字段Cookie中”,如许一来,这个非法网站就相当于登录了你的账号,能够随心所欲了!假如这不是一个买买买账号,而是你的银行账号,那……
这就是传说中的CSRF进击浅谈CSRF进击体式格局。
看了这波CSRF进击我在想,纵然有了同源战略限定,但cookie是明文的,还不是一样能拿下来。因而我看了一些cookie相干的文章聊一聊 cookie、Cookie/Session的机制与平安,晓得了效劳端能够设置httpOnly,使得前端没法操纵cookie,假如没有如许的设置,像XSS进击就能够够去猎取到cookieWeb平安测试之XSS;设置secure,则保证在https的加密通信中传输以防截获。

没有同源战略限定的Dom查询

1.有一天你刚睡醒,收到一封邮件,说是你的银行账号有风险,赶忙点进www.yinghang.com改暗码。你吓尿了,赶忙点进去,照样熟习的银行登录界面,你坚决输入你的账号暗码,登录进去看看钱有无少了。
2.睡眼模糊的你没看清晰,日常平凡接见的银行网站是www.yinhang.com,而如今接见的是www.yinghang.com,这个垂纶网站做了什么呢?

// HTML
<iframe name="yinhang" src="www.yinhang.com"></iframe>
// JS
// 因为没有同源战略的限定,垂纶网站能够直接拿到别的网站的Dom
const iframe = window.frames['yinhang']
const node = iframe.document.getElementById('你输入账号暗码的Input')
console.log(`拿到了这个${node},我还拿不到你方才输入的账号暗码吗`)

由此我们晓得,同源战略确切能躲避一些风险,不是说有了同源战略就平安,只是说同源战略是一种阅读器最基本的平安机制,毕竟能进步一点进击的本钱。实在没有刺不穿的盾,只是进击的本钱和进击胜利后取得的好处成不成正比。

跨域准确的打开体式格局

经由对同源战略的相识,我们应当要消弭对阅读器的误会,同源战略是阅读器做的一件功德,是用来防备来自邪门歪道的进击,但总不能为了不让暴徒进门而把悉数人都拒之门外吧。没错,我们这类正派人物只需打开体式格局准确,就应当能够跨域。
下面将一个个演示准确打开体式格局,但在此之前,有些预备事变要做。为了当地演示跨域,我们须要:
1.随意跑起一份前端代码(以下前端是随意跑起来的vue),地点是http://localhost:9099。
2.随意跑起一份后端代码(以下后端是随意跑起来的node koa2),地点是http://localhost:9971。

同源战略限定下接口要求的准确打开体式格局

1.JSONP

在HTML标签里,一些标签比方script、img如许的猎取资本的标签是没有跨域限定的,应用这一点,我们能够如许干:

后端写个小接口

// 处置惩罚胜利失败返回花样的东西
const {successBody} = require('../utli')
class CrossDomain {
  static async jsonp (ctx) {
    // 前端传过来的参数
    const query = ctx.request.query
    // 设置一个cookies
    ctx.cookies.set('tokenId', '1')
    // query.cb是前后端商定的要领名字,实在就是后端返回一个直接实行的要领给前端,因为前端是用script标签提议的要求,所以返回了这个要领后相当于立马实行,而且把要返回的数据放在要领的参数里。
    ctx.body = `${query.cb}(${JSON.stringify(successBody({msg: query.msg}, 'success'))})`
  }
}
module.exports = CrossDomain

简朴版前端

<!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>

简朴封装一下前端这个套路

/**
 * JSONP要求东西
 * @param url 要求的地点
 * @param data 要求的参数
 * @returns {Promise<any>}
 */
const request = ({url, data}) => {
  return new Promise((resolve, reject) => {
    // 处置惩罚传参成xx=yy&aa=bb的情势
    const handleData = (data) => {
      const keys = Object.keys(data)
      const keysLen = keys.length
      return keys.reduce((pre, cur, index) => {
        const value = data[cur]
        const flag = index !== keysLen - 1 ? '&' : ''
        return `${pre}${cur}=${value}${flag}`
      }, '')
    }
    // 动态建立script标签
    const script = document.createElement('script')
    // 接口返回的数据猎取
    window.jsonpCb = (res) => {
      document.body.removeChild(script)
      delete window.jsonpCb
      resolve(res)
    }
    script.src = `${url}?${handleData(data)}&cb=jsonpCb`
    document.body.appendChild(script)
  })
}
// 运用体式格局
request({
  url: 'http://localhost:9871/api/jsonp',
  data: {
    // 传参
    msg: 'helloJsonp'
  }
}).then(res => {
  console.log(res)
})

2.空iframe加form

仔细的朋侪能够发明,JSONP只能发GET要求,因为本质上script加载资本就是GET,那末假如要发POST要求怎么办呢?

后端写个小接口

// 处置惩罚胜利失败返回花样的东西
const {successBody} = require('../utli')
class CrossDomain {
  static async iframePost (ctx) {
    let postData = ctx.request.body
    console.log(postData)
    ctx.body = successBody({postData: postData}, 'success')
  }
}
module.exports = CrossDomain

前端

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
  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是一个W3C规范,全称是”跨域资本共享”(Cross-origin resource sharing)跨域资本共享 CORS 详解。看名字就晓得这是处置惩罚跨域题目标规范做法。CORS有两种要求,简朴要乞降非简朴要求。

这里援用上面链接阮一峰先生的文章申明一下简朴要乞降非简朴要求。
阅读器将CORS要求分红两类:简朴要求(simple request)和非简朴要求(not-so-simple request)。
只需同时满足以下两大前提,就属于简朴要求。
(1) 要求要领是以下三种要领之一:

HEAD
GET
POST

(2)HTTP的头信息不超越以下几种字段:

Accept
Accept-Language
Content-Language
Last-Event-ID
Content-Type:只限于三个值application/x-www-form-urlencoded、multipart/form-data、text/plain

1.简朴要求
后端

// 处置惩罚胜利失败返回花样的东西
const {successBody} = require('../utli')
class CrossDomain {
  static async cors (ctx) {
    const query = ctx.request.query
    // *时cookie不会在http要求中带上
    ctx.set('Access-Control-Allow-Origin', '*')
    ctx.cookies.set('tokenId', '2')
    ctx.body = successBody({msg: query.msg}, 'success')
  }
}
module.exports = CrossDomain

前端什么也不必干,就是一般发要求就能够够,假如须要带cookie的话,前后端都要设置一下,下面谁人非简朴要求例子会看到。

fetch(`http://localhost:9871/api/cors?msg=helloCors`).then(res => {
  console.log(res)
})

2.非简朴要求
非简朴要求会发出一次预检测要求,返回码是204,预检测经由过程才会真正发出要求,这才返回200。这里经由过程前端发要求的时刻增添一个分外的headers来触发非简朴要求。

后端

// 处置惩罚胜利失败返回花样的东西
const {successBody} = require('../utli')
class CrossDomain {
  static async cors (ctx) {
    const query = ctx.request.query
    // 假如须要http要求中带上cookie,须要前后端都设置credentials,且后端设置指定的origin
    ctx.set('Access-Control-Allow-Origin', 'http://localhost:9099')
    ctx.set('Access-Control-Allow-Credentials', true)
    // 非简朴要求的CORS要求,会在正式通信之前,增添一次HTTP查询要求,称为"预检"要求(preflight)
    // 这类情况下除了设置origin,还须要设置Access-Control-Request-Method以及Access-Control-Request-Headers
    ctx.set('Access-Control-Request-Method', 'PUT,POST,GET,DELETE,OPTIONS')
    ctx.set('Access-Control-Allow-Headers', 'Origin, X-Requested-With, Content-Type, Accept, t')
    ctx.cookies.set('tokenId', '2')

    ctx.body = successBody({msg: query.msg}, 'success')
  }
}
module.exports = CrossDomain

一个接口就要写这么多代码,假如想一切接口都统一处置惩罚,有什么更文雅的体式格局呢?见下面的koa2-cors。

const path = require('path')
const Koa = require('koa')
const koaStatic = require('koa-static')
const bodyParser = require('koa-bodyparser')
const router = require('./router')
const cors = require('koa2-cors')
const app = new Koa()
const port = 9871
app.use(bodyParser())
// 处置惩罚静态资本 这里是前端build好以后的目次
app.use(koaStatic(
  path.resolve(__dirname, '../dist')
))
// 处置惩罚cors
app.use(cors({
  origin: function (ctx) {
    return 'http://localhost:9099'
  },
  credentials: true,
  allowMethods: ['GET', 'POST', 'DELETE'],
  allowHeaders: ['t', 'Content-Type']
}))
// 路由
app.use(router.routes()).use(router.allowedMethods())
// 监听端口
app.listen(9871)
console.log(`[demo] start-quick is starting at port ${port}`)
前端

fetch(`http://localhost:9871/api/cors?msg=helloCors`, {
  // 须要带上cookie
  credentials: 'include',
  // 这里增加分外的headers来触发非简朴要求
  headers: {
    't': 'extra headers'
  }
}).then(res => {
  console.log(res)
})

4.代办

想一下,假如我们要求的时刻照样用前端的域名,然后有个东西帮我们把这个要求转发到真正的后端域名上,不就防止跨域了吗?这时刻,Nginx进场了。
Nginx设置

server{
    # 监听9099端口
    listen 9099;
    # 域名是localhost
    server_name localhost;
    #通常localhost:9099/api这个模样的,都转发到真正的效劳端地点http://localhost:9871 
    location ^~ /api {
        proxy_pass http://localhost:9871;
    }    
}

前端就不必干什么事变了,除了写接口,也没后端什么事变了

// 要求的时刻直接用回前端这边的域名http://localhost:9099,这就不会跨域,然后Nginx监听到通常localhost:9099/api这个模样的,都转发到真正的效劳端地点http://localhost:9871 
fetch('http://localhost:9099/api/iframePost', {
  method: 'POST',
  headers: {
    'Accept': 'application/json',
    'Content-Type': 'application/json'
  },
  body: JSON.stringify({
    msg: 'helloIframePost'
  })
})

Nginx转发的体式格局好像很轻易!但这类运用也是看场景的,假如后端接口是一个大众的API,比方一些大众效劳猎取天色什么的,前端挪用的时刻总不能让运维去设置一下Nginx,假如兼容性没题目(IE 10或许以上),CROS才是更通用的做法吧。

同源战略限定下Dom查询的准确打开体式格局

1.postMessage

window.postMessage() 是HTML5的一个接口,专注完成差别窗口差别页面的跨域通信。
为了演示轻易,我们将hosts改一下:127.0.0.1 crossDomain.com,如今接见域名crossDomain.com就即是接见127.0.0.1。

这里是http://localhost:9099/#/crossDomain,发音讯方

<template>
  <div>
    <button @click="postMessage">给http://crossDomain.com:9099发音讯</button>
    <iframe name="crossDomainIframe" src="http://crossdomain.com:9099"></iframe>
  </div>
</template>

<script>
export default {
  mounted () {
    window.addEventListener('message', (e) => {
      // 这里肯定要对泉源做校验
      if (e.origin === 'http://crossdomain.com:9099') {
        // 来自http://crossdomain.com:9099的效果复兴
        console.log(e.data)
      }
    })
  },
  methods: {
    // 向http://crossdomain.com:9099发音讯
    postMessage () {
      const iframe = window.frames['crossDomainIframe']
      iframe.postMessage('我是[http://localhost:9099], 贫苦你查一下你那里有无id为app的Dom', 'http://crossdomain.com:9099')
    }
  }
}
</script>

这里是http://crossdomain.com:9099,吸收音讯方

<template>
  <div>
    我是http://crossdomain.com:9099
  </div>
</template>

<script>
export default {
  mounted () {
    window.addEventListener('message', (e) => {
      // 这里肯定要对泉源做校验
      if (e.origin === 'http://localhost:9099') {
        // http://localhost:9099发来的信息
        console.log(e.data)
        // e.source能够是复书的对象,实在就是http://localhost:9099窗口对象(window)的援用
        // e.origin能够作为targetOrigin
        e.source.postMessage(`我是[http://crossdomain.com:9099],我晓得了兄弟,这就是你想晓得的效果:${document.getElementById('app') ? '有id为app的Dom' : '没有id为app的Dom'}`, e.origin);
      }
    })
  }
}
</script>

2.document.domain

这类体式格局只合适主域名雷同,但子域名差别的iframe跨域。
比方主域名是http://crossdomain.com:9099,子域名是http://child.crossdomain.com:9099,这类情况下给两个页面指定一下document.domain即document.domain = crossdomain.com就能够够接见各自的window对象了。

3.canvas操纵图片的跨域题目

这个应当是一个比较冷门的跨域题目,张大神已写过了我就不再班门弄斧了处置惩罚canvas图片getImageData,toDataURL跨域题目

末了
愿望看完这篇文章以后,再有人问跨域的题目,你能够嘴角轻轻上扬,嘲笑一声:“不要再问我跨域的题目了。”
拂袖而去。

    原文作者:虚竹
    原文地址: https://segmentfault.com/a/1190000017308784
    本文转自网络文章,转载此文章仅为分享知识,如有侵权,请联系博主进行删除。
点赞