githubhtml
近日发现一个问题:应用程序在返回Http Redirect的时候丢失了原先访问的端口。好比,咱们这样访问http://IP-A:Port-A/app/delete
,这个url会响应302,可是它返回的Response header Location
里丢失了端口,正确的结果应该是这样:http://IP-A:Port-A/app/index
,但返回的倒是:http://IP-A/app/index
,把端口丢失了。java
咱们的部署状况是这样的:nginx
服务器信息:git
Server Name | NAT Server | K8S Node | Nginx Ingress Svc | Nginx Ingress Pod | App Svc | App Pod |
---|---|---|---|---|---|---|
IP | IP-A | IP-B | IP-C(Cluster IP/VIP) | IP-D(Cluster IP) | IP-E(Cluster IP/VIP) | IP-F(ClusterIP) |
Port | Port-A | Port-B(Nginx Ingress Svc's NodePort) | Port-C | 80(Container Port) | Port-E | Port-F |
其实以上也不全是服务器,其中有两个K8S Service不是服务器,它们是VIP,关于这个请看K8S - Using Source IP一文,当访问http://IP-A:Port-A/app/delete
的时候,这个请求从左到右贯穿了这些服务器。github
顺便一提上面的NAT Server是一台普通的服务器,咱们用它作了PAT使咱们的Nginx Ingress可以被外网访问到。apache
咱们使用以前提到过的Echo Server来观察透过Ingress访问Echo Server时传递给Echo Server的Request header:http://IP-A:Port-A/echo-server
,获得了这些有趣的Request header:segmentfault
host=IP-A:Port-A x-original-uri=/echo-server x-forwarded-for=IP-B x-forwarded-host=IP-A:Port-A x-forwarded-port=80 x-forwarded-proto=http
而后直接访问Echo Server Svc,发现是没有上面提到的x-*
Request header的。因而怀疑问题出在这几个header上。api
来说一下这些头各自表明什么意思。浏览器
注意,前三个是事实标准,MDN有收录,x-forwarded-port
和x-original-uri
彷佛是私有扩展。tomcat
找一个趁手的Http Request工具(我用的是Postman),记得把Follow redirect关掉,而后模拟Nginx请求的方式(就是把上面提到的x-*
header带上/去掉/修改值)直接请求App Svc。
结果发现x-forwarded-port
是Response header Location
的关键,即若是x-forwarded-port=Port-A
的话,Location
就会带上正确的端口。
能够推测,App利用了host
和x-forwarded-*
这些header来构造redirect url。
在Java Servlet API中,在描述HttpServletResponse#sendRedirect的时候提到,其返回的URL必须是Absolute URL。
Tomcat的org.apache.catalina.connector.Response
的toAbsolute方法负责构造Absolute URL。
那么它又是如何知道选用什么Port的呢?这个和RemoteIPValve有关,有兴趣的话你能够查阅相关文档。
上面只是讲了Tomcat是如何构造redirect url的,但这个方法不是标准的,不一样的容器有各自的实现,毕竟Java Servlet API也没有规定如何构造Absolute URL。
我以前也写过一篇相关话题的文章《反向代理使用https协议,后台tomcat使用http,redirect时使用错误协议的解决办法》,你能够看一看。
那么问题来了,我明明访问的是IP-A:Port-A
,为什么Nginx取到的值是80?
这是由于在整个请求链路的前段:NAT Server > K8S Node > Nginx Ingress Svc 都是在第4层工做的,能够认为它们干的事情都是NAT,Nginx Ingress Pod是不知道这些服务器/网络节点的端口,所以它只能把本身的端口80(容器内Port)给x-forwarded-port。
关于这个逻辑你能够查看Nginx Ingress的配置文件就可以知道了:
kubectl -n kube-system exec -it <nginx-ingress-controller-pod-name> -- cat /etc/nginx/nginx.conf
查看Nginx Ingress配置文件发现若是最初请求的时候带上x-forwarded-port
的话,就可以改变它传递到后面的值,可是这有两个问题:
因此这个方法很差。
虽然能够经过修改tomcat的代码,让它从x-forward-host/host
header来取port,可是这个不现实。
这个方法比较靠谱,只要将NAT Server的端口改为80就没有问题了。
事实上,若是你直接访问K8S Node的话(NodePort方式),也是要将NodePort设置为80,记得前面说的吗?Nginx Ingress没法知道上层NAT的端口。
总而言之,就是你最初请求的URL不能是80以外的端口,必须是http://some-ip/app
才能够。
使用Nginx Ingress提供的Proxy redirect annotations,将Location
的值作文本替换。