问题
系统使用了vip与haproxy实现高可用以及对nginx进行负载均衡,但是发现在上游的应用服务无法拿到客户端的请求ip地址,拿到的是主haproxy机器的ip,以下是nginx与haproxy的缩减配置:
location ~* ^/(xx|xx) {
proxy_pass http://upsteam;
proxy_set_header X-Real-IP $remote_addr;
proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
}
haproxy配置,已缩减
frontend nginx
mode tcp
bind 0.0.0.0:443
default_backend nginx_cluster
backend nginx_cluster
mode tcp
balance leastconn
server inst1 10.17x.xx.xx:4433
haproxy配置参考地址
https://www.haproxy.com/documentation/haproxy-configuration-tutorials/load-balancing/tcp/
haproxy版本:2.4.22
nginx版本:1.24.0
网络架构图
原因分析
因为haproxy为了提升性能使用了tcp模式转发,但是由于工作在第四层协议,不会解析到第七层的http包,所以无法直接将客户端ip传递给上游的Nginx。
解决方案
对haproxy与nginx采用proxy协议,haproxy通过send-proxy指令将客户端ip传递给上游nginx,nginx获取到客户端ip后将其写入调用上游应用的请求头中,配置如下
- haproxy
frontend nginx
mode tcp
bind 0.0.0.0:443
default_backend nginx_cluster
backend nginx_cluster
mode tcp
balance leastconn
server inst1 10.17x.xx.xx:4433 send-proxy
- nginx
server {
listen 4433 ssl default_server proxy_protocol;
server_name localhost;
ssl_certificate "/etc/nginx/cert/xx.pem";
ssl_certificate_key "/etc/nginx/cert/xx.key";
charset utf-8;
location ~* ^/(xx|xx) {
proxy_pass http://upsteam;
proxy_set_header X-Real-IP $proxy_protocol_addr;
proxy_set_header X-Forwarded-For $proxy_protocol_addr;
}
}
参考文档地址:
https://docs.nginx.com/nginx/admin-guide/load-balancer/using-proxy-protocol/#realip