Vue2.0设置反向代理解决跨域问题

Vue2.0设置反向代理解决跨域问题

1、了解cookie

Cookie的作用:

Cookie是用于维持服务端会话状态的,通常由服务端写入,在后续请求中,供服务端读取。

 

HTTP请求,Cookie的使用过程

1、server通过HTTP Response中的"Set-Cookie: header"把cookie发送给client

2、client把cookie通过HTTP Request 中的“Cookie: header”发送给server

3、每次HTTP请求,Cookie都会被发送。

http请求发送cookies的条件:

1、本地已经缓存有cookies

2、根据请求的URL来匹配cookies的domain、path属性,如果都符合才会发送。

举个例子:访问www.baidu.com时,就不发送www.qq.com的cookies.

Cookies查看与存放路径

cookies查看:

大家可以使用抓包工具,这种工具就不多说,网上搜索出很多。使用这些工具中可以清晰的看到Http Request 中的Cookie, 和Http Response中的cookie

 

2、跨域资源共享CORS

CORS是一个W3C标准,全称是"跨域资源共享"Cross-origin resource sharing)。

它允许浏览器向跨源服务器,发出XMLHttpRequest请求,从而克服了AJAX只能同源使用的限制。

本文详细介绍CORS的内部机制。

一、简介

CORS需要浏览器和服务器同时支持。目前,所有浏览器都支持该功能,IE浏览器不能低于IE10

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

因此,实现CORS通信的关键是服务器。只要服务器实现了CORS接口,就可以跨源通信。

二、两种请求

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

只要同时满足以下两大条件,就属于简单请求。

1) 请求方法是以下三种方法之一:

HEAD

GET

POST

2HTTP的头信息不超出以下几种字段:

Accept

Accept-Language

Content-Language

Last-Event-ID

Content-Type:只限于三个值application/x-www-form-urlencodedmultipart/form-datatext/plain

凡是不同时满足上面两个条件,就属于非简单请求。

浏览器对这两种请求的处理,是不一样的。

三、简单请求

3.1 基本流程

对于简单请求,浏览器直接发出CORS请求。具体来说,就是在头信息之中,增加一个Origin字段。

下面是一个例子,浏览器发现这次跨源AJAX请求是简单请求,就自动在头信息之中,添加一个Origin字段。

 

GET /cors HTTP/1.1Origin: http://api.bob.comHost: api.alice.comAccept-Language: en-USConnection: keep-aliveUser-Agent: Mozilla/5.0...

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

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

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

Access-Control-Allow-Origin: http://api.bob.comAccess-Control-Allow-Credentials: trueAccess-Control-Expose-Headers: FooBarContent-Type: text/html; charset=utf-8

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

1Access-Control-Allow-Origin

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

2Access-Control-Allow-Credentials

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

3Access-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;

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

 

3、Vue设置反向代理;

Config/index.js ----dev:{

...

proxyTable: {//设置反向代理,实现跨域.

        '/api': {

            target: 'http://192.168.1.110:8080', //设置调用接口域名和端口号别忘了加http

            changeOrigin: true,

            pathRewrite: {// 如果接口本身没有/api需要通过pathRewrite来重写了地址

                '^/api': '/' //这里理解成用‘/api’代替target里面的地址,组件中我们调接口时直接用/api代替

                    // 比如我要调用'http://0.0:300/user/add',直接写‘/api/user/add’即可 代理后地址栏显示/

            },

            //解决跨域引起的session问题,这段代码改变cookie 的作用于为 path="/"

            onProxyRes(proxyRes, req, res) {

                var oldCookie = proxyRes.headers['set-cookie']

                if (oldCookie == null || oldCookie.length == 0) {

                    delete proxyRes.headers['set-cookie']

                    return

                }

                var oldCookieItems = oldCookie[0].split(';')

                var newCookie = ''

                for(var i=0; i < oldCookieItems.length; ++i){

                    if(newCookie.length >0)

                        newCookie += ';'

                    if(oldCookieItems[i].indexOf('Path=') >= 0)

                        newCookie += 'Path=/'

                    else

                        newCookie += oldCookieItems[i]

                }

                proxyRes.headers['set-cookie'] = [newCookie]

            }

        }

    },

...

}

axios.defaults.withCredentials = true;

4、代理转发cookie导致session重置或session失效

 

环境:

    

 

    2个业务接口需要转发到82的服务器上:

    ../user/getCode.do

    ../user/doLogin.do

现象:

    

    使用上述的2个接口实现用户登录功能,首先显示登录页面,调用../user/getCode.do获取验证码,然后用户界面输入用户名、密码和验证码,点击登录之后调用../user/doLogin.do接口实现登录。结果../user/doLogin.do后端总是找不到验证码,因为浏览器没有把cookie中的JSESSIONID传递过来。无法登录的话,后续的所有其他接口都无法调用了。

 

分析:

    通过chromenetwork,可以看到如下信息:

 

    那么现在的问题就是为什么在点击立即登录之后,浏览器为什么没有在请求中带上上次返回的cookie信息呢?

    仔细看response header中的set-cookie头可以发现,其中有一个Path=/webserver/,说明这个cookie是有适用范围的。只能在/webserver/路径下使用

    

    而我们代码中使用的路径是这么写的:     ../user/getCode.do,没有包含webserver信息,这样浏览器自然就认为这个请求和上次返回的那个cookie不匹配。

 

解决方法:

    解决方法有2中:

    一种是在请求的时候带上/webserver/前缀,比如把上文的 ../user/getCode.do改为  ../webserver/user/getCode.do

    第二种是修改代理的实现,把82服务器上返回的http头上的set-cookie内的Path改为  Path=/,代码如下。

    vuewebpack脚手架中的build/dev-server.js内的  Object.keys(proxyTable).forEach(function (context)  的实现改为如下形式

 

Object.keys(proxyTable).forEach(function (context) {

  var options = proxyTable[context]

  if (typeof options === 'string') {

    options = {

      target: options,

      onProxyRes(proxyRes, req, res) {

        //set-cookie:JSESSIONID=6F766ED2EEEBEAA9245F7F908A848857; Path=/webserver/; HttpOnly

        

        var oldCookie = proxyRes.headers['set-cookie']

        if(oldCookie== null || oldCookie.length==0){

          delete proxyRes.headers['set-cookie']

          return

        }

        console.log(oldCookie)

        var oldCookieItems = oldCookie[0].split(';')

        var newCookie = ''

        for(var i=0; i < oldCookieItems.length; ++i){

          if(newCookie.length >0)

            newCookie += ';'

          if(oldCookieItems[i].indexOf('Path=') >= 0)

            newCookie += 'Path=/'

          else

            newCookie += oldCookieItems[i]

        }

        proxyRes.headers['set-cookie'] = [newCookie]

 

        //proxyRes.headers['x-addedygc'] = 'foobar';     // add new header to response

        //delete proxyRes.headers['connection'];       // remove header from response

      }

    }

  }

  app.use(proxyMiddleware(context, options))

})

 

    

结论:

    经过试验,以上2种方法都适用,问题解决。

 

 本文参考了很多大牛写的文章,并加以总结。

http://www.ruanyifeng.com/blog/2016/04/cors.html 

http://www.cnblogs.com/strinkbug/p/6073806.html 

posted @ 2018-05-10 14:26  cjsuning  阅读(833)  评论(0编辑  收藏  举报