HAProxy的调度算法
HAProxy调度算法分为静态调度算法和动态调度算法。
静态算法是按照事先定义好的调度规则轮询公平调度,不关心后端服务器的当前负载、连接数和响应速度等,并且无法实时修改权重,只能重启后生效。
静态调度算法有:static-rr,first
动态算法是根据后端服务器状态进行调度适当调整,比如优先调度至当前负载较低的服务器,并且权重可以在haproxy运行时调整,无需重启服务
动态调度算法有:roundrobin,leastconn,source,uri,url_param,hdr,rdp-cookie
定义方法:
balance <调度算法>
静态调度算法
static-rr
静态轮询,基于权重轮询调度,不支持HAProxy在运行时进行权重调整和后端服务器的慢启动,后端的主机数量没有限制
listen web_http
mode http
balance static-rr #设定静态轮询
bind 192.168.27.21:80
server web1 192.168.27.31:80 weight 2 check inter 3s fall 3 rise 5
server web2 192.168.27.32:80 weight 1 check inter 3s fall 3 rise 5
测试:
[root@localhost ~]# curl 192.168.27.21
server 1
[root@localhost ~]# curl 192.168.27.21
server 1
[root@localhost ~]# curl 192.168.27.21
server 2
first
HAProxy会根据服务器在列表中的位置,自上而下进行调度,只有当第一台服务器的连接数达到上限时,新的请求才会被分配到下一台服务器,此设置会忽略服务器的权重
示例:
listen web_http
mode http
balance first
bind 192.168.27.21:80
server web1 192.168.27.31:80 maxconn 10 check inter 3s fall 3 rise 5 #配置连接上线位为10,不设置权重
server web2 192.168.27.32:80 weight 2 check inter 3s fall 3 rise 5
测试:
[root@localhost ~]# curl 192.168.27.21
server 1
[root@localhost ~]# curl 192.168.27.21
server 1
[root@localhost ~]# curl 192.168.27.21
server 1
[root@localhost ~]# curl 192.168.27.21
server 1
[root@localhost ~]# curl 192.168.27.21
server 1
[root@localhost ~]# curl 192.168.27.21
server 1
所有的请求依旧会发往server1,server2的权重设置无效,只有当server1的连接数占满时才将请求发送给server2
动态调度算法
roundrobin
基于权重的轮询动态调度算法,支持权重的运行时调整,支持慢启动,每个后端backend中最多支持4095个server,此为默认调度算法。
示例:
1.修改配置文件
stats socket /var/lib/haproxy/haproxy.sock mode 600 level admin #将socket文件开启,在进行动态调节权重时,是通过socket进行的
listen web_http
mode http
balance roundrobin #调度算法设置为roundrobin
bind 192.168.27.21:80
server web1 192.168.27.31:80 weight 1 check inter 3s fall 3 rise 5
server web2 192.168.27.32:80 weight 1 check inter 3s fall 3 rise 5
2.创建socket文件的目录
[root@localhost ~]# mkdir /var/lib/haproxy
3.重启服务器,查看socket文件是否启用
[root@localhost ~]# ls /var/lib/haproxy/
haproxy.sock #已经存在
测试:
动态调节服务器状态时需要使用到socat命令,先安装此命令
[root@localhost ~]# yum install socat -y
socat的简单使用,可以通过echo+指令然后使用管道传送给socket文件
[root@localhost ~]# echo "help" | socat stdio /var/lib/haproxy/haproxy.sock
#此命令可以用来查看socat一些简单的使用帮助
1.获取服务器权重
[root@localhost ~]# echo "get weight web_http/web1" | socat stdio /var/lib/haproxy/haproxy.sock #获取权重时,指定后端服务器组和服务器名称
1 (initial 1) 权重为1
2.动态修改权重,然后再次查看
[root@localhost ~]# echo "set weight web_http/web1 2" | socat stdio /var/lib/haproxy/haproxy.sock #使用set来修改权重,
[root@localhost ~]# echo "get weight web_http/web1" | socat stdio /var/lib/haproxy/haproxy.sock
2 (initial 1)
测试访问
[root@localhost ~]# curl 192.168.27.21
server 1
[root@localhost ~]# curl 192.168.27.21
server 1
[root@localhost ~]# curl 192.168.27.21
server 2
#权重已经被调整,需要注意动态设置的方法临时有效,重启服务将失效
将服务器动态下线
[root@localhost ~]# echo "disable server web_http/web1" | socat stdio /var/lib/haproxy/haproxy.sock
测试访问
[root@localhost ~]# curl 192.168.27.21
server 2
[root@localhost ~]# curl 192.168.27.21
server 2
#server1被下线,访问只能被调度到server2
将服务器动态上线
[root@localhost ~]# echo "enable server web_http/web1" | socat stdio /var/lib/haproxy/haproxy.sock
测试访问
[root@localhost ~]# curl 192.168.27.21
server 1
[root@localhost ~]# curl 192.168.27.21
server 1
[root@localhost ~]# curl 192.168.27.21
server 2
#server1又能正常访问,并且之前设置的权重依旧有效
source
source调度算法是基于用户的源地址hash并将请求转发到后端服务器,默认为静态即取模方式(可以通过hash-type支持的选项进行更改),后续同一个地址发来的请求将都被转发为后端的同一个web服务器,比较适合用于session保持/缓存业务等场景
示例:
listen web_http
mode http
balance source
bind 192.168.27.21:80
server web1 192.168.27.31:80 check inter 3s fall 3 rise 5
server web2 192.168.27.32:80 check inter 3s fall 3 rise 5
测试
[root@localhost ~]# curl 192.168.27.21
server 1
[root@localhost ~]# curl 192.168.27.21
server 1
[root@localhost ~]# curl 192.168.27.21
server 1
#当第一次访问被调度到后天的server1后之后的所有请求都被调度到server1
调度算法为源地址哈希是是无法进行权重的修改的,默认为静态
[root@localhost ~]# echo "set weight web_http/web1 2" | socat stdio /var/lib/haproxy/haproxy.sock
Backend is using a static LB algorithm and only accepts weights '0%' and '100%'.
#无法进行对权重进行设置,只能开启或关闭
source 一致性hash算法
源地址哈希由于是静态的,如果后端的服务器发生改变,取模后的值也将发生改变,以至于整张表上的记录作废,从而重新进行计算。
一致性哈希是动态的,支持在线的权重调整,当服务器的总权重发生变化时,影响的只是一部分,不会引起大的变动
示例:
listen web_http
mode http
balance source
hash-type consistent #使用hash-type指定一致性哈希
bind 192.168.27.21:80
server web1 192.168.27.31:80 check inter 3s fall 3 rise 5
server web2 192.168.27.32:80 check inter 3s fall 3 rise 5
uri
uri哈希算法,可以通过用户访问的uri来进行调度,常用于缓存服务器。当用户第一访问某uri是,haproxy会从后台选出一台服务进行调度并记录,当再其他用户也访问此uri时,haproxy会固定的将其调度到后台上次所请求此uri的那台服务器上。
示例:
将配置文件改为基于uri哈希的算法
listen web_http
mode http
balance uri #设置为uri算法
bind 192.168.27.21:80
server web1 192.168.27.31:80 weight 1 check inter 3s fall 3 rise 5
server web2 192.168.27.32:80 weight 3 check inter 3s fall 3 rise 5
测试
分别使用不同的客户端对相同的资源进行访问
使用client1进行访问
[root@client1 ~]# curl 192.168.27.21/index.html
server 1
使用client2进行访问
[root@client2 ~]# curl 192.168.27.21/index.html
server 1
使用client3进行访问
[root@client3 ~]# curl 192.168.27.21/index.html
server 1
结论:不同用户访问同一个资源将会调度到后台的统一个服务器上
url_param
url_param的调度算法是对用户请求的url中{params}部分中的参数的值做哈希计算,并由服务器总权重相除后派发至选中的后端服务器;常用于追踪用户确保同一个用户的请求永远发向同一台后端服务器。
假设url为http://www.mylinuxops.com/index.php?name=wang
其中url_param为?之后的内容即name=wang,而其中name为参数,wang为值
示例:
listen web_http
mode http
balance url_param name #设定调度算法为 url_param,参数为name
bind 192.168.27.21:80
server web1 192.168.27.31:80 weight 2 check inter 3s fall 3 rise 5
server web2 192.168.27.32:80 weight 1 check inter 3s fall 3 rise 5
测试
当用户所带的参数的值相同时则请求后台相同的服务器
[root@localhost ~]# curl 192.168.27.21/index.html?name=123
server 2
[root@localhost ~]# curl 192.168.27.21/index.html?name=123
server 2
当用户使用未设定的参数时,将轮询
[root@localhost ~]# curl 192.168.27.21/index.html?aaa=123
server 1
[root@localhost ~]# curl 192.168.27.21/index.html?aaa=123
server 1
[root@localhost ~]# curl 192.168.27.21/index.html?aaa=123
server 2
hdr(NAME)
hdr调度算法是针对用户每个http请求头部中指定的信息做hash,然后由服务器的总权重相除取模后派发至后端的服务器,如果没有有效的值则将进行轮询调度。hdr可以使用的name有(cookie、User-agent、host)
示例:
listen web_http
mode http #此算法是针对http的所以模式需要使用http
balance hdr(user-agent) #使用hdr算法,以用户的客户端作调度
bind 192.168.27.21:80
server web1 192.168.27.31:80 weight 2 check inter 3s fall 3 rise 5
server web2 192.168.27.32:80 weight 1 check inter 3s fall 3 rise 5
示例:
使用curl访问时
[root@localhost ~]# curl 192.168.27.21
server 2
[root@localhost ~]# curl 192.168.27.21
server 2
[root@localhost ~]# curl 192.168.27.21
server 2
[root@localhost ~]# curl 192.168.27.21
server 2
使用curl模拟其他客户端访问
[root@localhost ~]# curl -A "12" 192.168.27.21
server 1
[root@localhost ~]# curl -A "12" 192.168.27.21
server 1
[root@localhost ~]# curl -A "12" 192.168.27.21
rdp-cookie
rdp-cookie调度算法是针对远程桌面的负载,使用cookie保持会话,用在远程为windows的情况下。
示例:
listen RDP
bind 192.168.27.21:3389
balance rdp-cookie #修改调度算法为rdp-cookie
mod tcp #由于后端为windows所以此处必须为tcp
server rdp1 192.168.27.40:3389 check inter 3s fall 3 rise 5
server rdp1 192.168.27.40:3389 check inter 3s fall 3 rise 5