nginx反向代理致使session失效的问题处理

一同事求援:后台系统的登陆成功了,但不能成功登进系统,仍然跳转到登陆页,但同一套代码另外一个环境却没有问题。html

背景

经了解,他对同一个项目使用tomcat部署了两个环境,一个在开发服务器上,一个在他本机,两个环境代码配置彻底相同。两边经过同一个nginx进行反向代理,nginx配置大体以下,nginx

location /health/ {
    proxy_pass  http://192.168.40.159:8081/health/;  #无问题的配置
 }

location /health-dev/ {
    proxy_pass  http://192.168.40.202:8080/health/;  #有问题的配置
}

一个反向代理到开发环境,一个反向代理到本机服务。浏览器

定位

既然代码配置彻底相同,那么问题很大可能就出如今nginx的反向代理上。tomcat

由于两边location路径不一样(即浏览器路径不一样),可是反向代理的服务端路径却相同,结合session的基本原理,以下图,服务器

httpsession

  1. 当浏览器第一次打开页面时,服务端会为此次会话建立一个session,并将session id经过response的header传递给浏览器,header通常为 Set-Cookie: JSESSIONID=xxxxx; Path=xxxx
  2. 浏览器接收到响应后,若是header Set-Cookie 中path的值与浏览器地址路径匹配,则将该header值存于浏览器的Cookie中
  3. 浏览器在下次请求服务器时,将Cookie中的JSESSIONID值经过request的header上报给服务端,header通常为 Cookie: JSESSIONID=xxxx;
  4. 服务端可经过该JSESSIONID来定位到对应的session

nginx反向代理按这种方式配置时cookie

location /health-dev/ {
    proxy_pass  http://192.168.40.202:8080/health/;
}

浏览器访问 http://www.domian.com/health-dev 时,服务端返回的 Set-Cookie 的 Path 值为 /health(由于中间有反向代理,服务端并不知道代理前的路径是啥,是按最终请求服务端的路径设置),如图session

nginx-proxy-cookie-1

由于浏览器访问地址的路径 /health-devSet-Cookie 的 Path /health 不匹配,因此浏览器并不会将其值存入Cookie中,如图dom

nginx-proxy-cookie-2

所以在下次请求服务器时,浏览器没法设置request Cookie header的 JSESSIONID 值,服务器没法定位到对应的session,所以会将其当作第一次请求,建立一个新的session,如此反复,所以就算你登陆认证经过了,但服务器返回的登陆凭证(JSESSIONID)浏览器不会保存,并在下次请求时携带,致使服务器认为你是一个新的请求,固然就会又跳到登陆页面了。代理

解决

nginx有一个命令 proxy_cookie_path(参考: proxy_cookie_path)可将服务器返回的 Set-Cookie 中的path进行修改,格式为 proxy_cookie_path 原路径 目标路径,咱们在配置中添加 proxy_cookie_path 以下。code

location /health-dev/ {
    proxy_pass  http://192.168.40.202:8080/health/;
    proxy_cookie_path  /health /health-dev;
}

重启nginx,问题解决。


[转载请注明出处]
做者:雨歌
欢迎关注做者公众号:半路雨歌,查看更多技术干货文章
qrcode

相关文章
相关标签/搜索