Setup HTTP Load Balancer with HAProxy
When it comes to ensuring that we need our website needs to be online at all times, we need to start looking at High Availability.
Today we will be demonstrating a basic setup of Layer 4 (transport layer) load balancing making use of HAProxy Server with 2 backend nodes using the round robin algorithm, which essentially means that the first backend node will respond on the first request, and the second backend node on the second request.
Our Setup:
We will have 1 haproxy server that will be listening on port 80 and also Python's SimpleHTTPServer running on port 8000, this will act as a lightweight web server to serve a maintenance page when both of our nodes are down.
Then for our back-end nodes, both servers will be listening on port 8080 and only allowing our haproxy to communicate with port 8080. Website configuration and content has been synchronized with each other so that we have the exact same content on both servers.
Resources:
haproxy: 192.168.1.2 (HAProxy)
node-web1: 192.168.10.4 (LAMP)
node-web2: 192.168.11.4 (LAMP)
Dependencies:
$ yum install haproxy screen python openssl openssl-devel -y
HAProxy Configuration:
$ vi /etc/haproxy/haproxy.cfg
global
log 127.0.0.1 local2
chroot /var/lib/haproxy
pidfile /var/run/haproxy.pid
maxconn 4000
user haproxy
group haproxy
daemon
stats socket /var/lib/haproxy/stats mode 600 level admin
defaults
log global
option httplog
option dontlognull
option http-server-close
option forwardfor
option redispatch
retries 3
timeout http-request 20s
timeout queue 86400
timeout connect 86400
timeout client 86400
timeout server 86400
timeout http-keep-alive 30s
timeout check 20s
maxconn 50000
listen stats 192.168.1.2:1936
mode http
log global
maxconn 10
timeout queue 100s
stats enable
stats hide-version
stats refresh 30s
stats show-node
stats auth admin:sekret
stats uri /haproxy?stats
frontend http-in
bind *:80
mode http
acl web hdr_end(host) -i domain.com
acl www_web hdr_end(host) -i www.domain.com
default_backend web-backend
backend web-backend
mode http
balance roundrobin
option httpchk GET /status/healthcheck.html
option httpclose
option forwardfor
server node-web1 192.168.10.4:8080 check
server node-web2 192.168.11.4:8080 check
server maintenance 127.0.0.1:8000 backup
Rsyslog Configuration:
$ vi /etc/rsyslog.conf
Uncomment the following:
$ModLoad imudp
$UDPServerRun 514
Backend Node Pages:
Set the following on your index.html
for your backend nodes:
node-web1
<html>
Response From: node-web1 [192.168.10.4]
</html>
node-web2
<html>
Response From: node-web2 [192.168.11.4]
</html>
Maintenance Page:
Our maintenance page on Python Simple HTTP Server
$ mkdir -p /opt/python/simplehttpserver
$ cd /opt/python/simplehttpserver
$ vi index.html
<!doctype html>
<title>Site Maintenance</title>
<style>
body { text-align: center; padding: 150px; }
h1 { font-size: 50px; }
body { font: 20px Helvetica, sans-serif; color: #333; }
article { display: block; text-align: left; width: 650px; margin: 0 auto; }
a { color: #dc8100; text-decoration: none; }
a:hover { color: #333; text-decoration: none; }
</style>
<article>
<h1>We’ll be back soon!</h1>
<div>
<p>Sorry for the inconvenience but we’re performing some maintenance at the moment. If you need to reach out to us, you can always <a href="mailto:support@bekkersolutions.com">contact us</a>, otherwise we’ll be back online, shortly!</p>
<p>— Your Orginization </p>
</div>
</article>
Start the service in a screen session:
$ screen -S MaintenancePage -m -d sh -c "python –m SimpleHTTPServer"
Firewall Configuration:
For each node allow the HAProxy server address to communicate with port 8080/TCP:
$ iptables -A INPUT -p tcp -s [haproxy-source-address] --dport 8080 -j ACCEPT
Enable Services:
$ service rsyslog restart
$ service haproxy restart
$ chkconfig rsyslog on
$ chkconfig haproxy on
Testing:
$ curl http://192.168.1.2
Response From: node-web1 [192.168.10.4]
$ curl http://192.168.1.2
Response From: node-web2 [192.168.11.4]