Load Balance HA MySQL with HAProxy.
HAProxy or High Availability Proxy is a high performance open-source load balancer created to work (load balance) over TCP/HTTP. This is a Software that has several use cases you can chose, but it’s great for a MySQL load balancer. The company who makes HAProxy also develops many more products paid and free. Now for this lab we will use the community version as load balancer software on our servers. The key theme for this post will be HAProxy Docker to Load Balance MySQL. In other words a lab of HAProxy, Docker and MySQL.
In general Load Balancers power the uptime of organizations with large infrastructures and enormous traffic demands by giving them the flexibility and confidence to deliver websites and applications with high availability, performance and security at any scale in any environment. Many companies use load balancers in their product delivery process. You can check this article to configure a Master-Slave on MySQL.
data:image/s3,"s3://crabby-images/65653/656530e103c9763ef8780653608b02bb0a440ddf" alt="haproxy load balance mysql docker"
In this post, we will create a master replication master on MySQL and make MySQL load balancer replication with HAProxy. Load balance is useful if we want to create a reliable database with multiple MySQL services on multiple instances or servers. For this purpose, I have used docker to create a load balance with HAProxy and MySQL. However, you can also deploy it on an AWS instance or server without running docker.
Okay, let’s get started with creating a Master-Master replication with load balance on HAProxy using MySQL.
Note: I am using Ubuntu Server 18.04 LTS when creating this article.
How To Load Balance with MySQL: Editing The Docker-Compose yml File
If we create MySQL replication with HAProxy load balancer to load Balance using docker containers. The easiest way or you could even say best way for a lab is to make it is to use Docker Compose. This way you configure docker to create multiple containers using a dot yml (yaml) configuration file.
Docker compose helps us in managing containers in docker. You can edit the configuration in docker compose file. Let’s create docker compose files for MySQL and HAProxy containers by following the steps below.
version: '3.9'
services:
mysql-master1:
image: mysql:8.0
container_name: mysql-master1
restart: unless-stopped
env_file: master1/.env
command: --default-authentication-plugin=mysql_native_password
cap_add:
- all
volumes:
- ./master1/my.cnf:/etc/mysql/my.cnf
- ./master1/data:/var/lib/mysql
environment:
- TZ:${TZ}
- MYSQL_USER:${MYSQL_USER}
- MYSQL_PASSWORD:${MYSQL_PASSWORD}
- MYSQL_ROOT_PASSWORD:${MYSQL_PASSWORD}
networks:
default:
aliases:
- mysql
mysql-master2:
image: mysql:8.0
container_name: mysql-master2
restart: unless-stopped
command: --default-authentication-plugin=mysql_native_password
env_file: master2/.env
cap_add:
- all
volumes:
- ./master2/my.cnf:/etc/mysql/my.cnf
- ./master2/data:/var/lib/mysql
environment:
- TZ:${TZ}
- MYSQL_USER:${MYSQL_USER}
- MYSQL_PASSWORD:${MYSQL_PASSWORD}
- MYSQL_ROOT_PASSWORD:${MYSQL_PASSWORD}
networks:
default:
aliases:
- mysql
haproxy:
image: haproxytech/haproxy-ubuntu:latest
container_name: haproxy
restart: unless-stopped
ports:
- "2999:2999"
cap_add:
- all
volumes:
- ./haproxy.cfg:/etc/haproxy/haproxy.cfg
networks:
default:
aliases:
- mysql
Creating MySQL Master 1 Config for Load Balance with MySQL
I have used the latest version of MySQL to create a Master-Master MySQL-Replication on server 1. To make it easier to install MySQL, I used docker and created a persistent volume to allow the data in the container to be stored on the host server.
Additionally, create a persistent volume for my.cnf to connect to the container. Create a master1 folder to store data on MySQL replication on Master 1.
ubuntu@haproxy: mkdir /data/haproxy-mysql/master1
data:image/s3,"s3://crabby-images/93800/9380093b1ce53b0e66ac9331c1cceffa7d46a5d7" alt="create directory master1 load balance mysql"
Create a data folder in the Master 1 directory; this data folder is useful for storing persistent volumes from the MySQL master 1 container to the host server.
ubuntu@haproxy: mkdir /data/haproxy-mysql/master1/data
data:image/s3,"s3://crabby-images/793a4/793a40a86288aec82fafa1879dea535caa72c499" alt="create data directory in master1 load balance mysql"
Moreover, we need a custom configuration file in my.cnf on the host server that will create a persistent volume on the container.
ubuntu@haproxy: touch /data/haproxy-mysql/master1/my.cnf
data:image/s3,"s3://crabby-images/9ee60/9ee6042b62eea93d8d57aa210a9eba584301a971" alt="Create my.cnf file master1 load balance mysql"
After that, open the my.cnf file and add configuration to my.cnf file to do replication on MySQL Master 1.
# Copyright (c) 2017, Oracle and/or its affiliates. All rights reserved.
#
# This program is free software; you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation; version 2 of the License.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program; if not, write to the Free Software
# Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
#
# The MySQL Server configuration file.
#
# For explanations see
# http://dev.mysql.com/doc/mysql/en/server-system-variables.html
[mysqld]
pid-file = /var/run/mysqld/mysqld.pid
socket = /var/run/mysqld/mysqld.sock
datadir = /var/lib/mysql
secure-file-priv= NULL
server-id=1
binlog_format=ROW
log-bin
skip-name-resolve
# Custom config should go here
!includedir /etc/mysql/conf.d/
Then, create an environment variable (env) file to store the environment username and password in MySQL. As such, this lasts for the duration of terminal.
ubuntu@haproxy:~$ touch /data/haproxy-mysql/master2/.env
data:image/s3,"s3://crabby-images/5c77f/5c77f0cc7aaefac798c5338b2716cc43eb51dfa6" alt="env file master 1 load balance mysql"
Moreover, fill the env file on Master 1 like this:
### WORKSPACE #############################################
TZ=UTC
#MYSQL_DATABASE=master
MYSQL_USER=master
MYSQL_PASSWORD=Master123@
MYSQL_PORT=3306
MYSQL_ROOT_PASSWORD=Mastermaster123
The following is a tree view in the Master 1 directory for MySQL master1 replication.
data:image/s3,"s3://crabby-images/7b62e/7b62ef638385af2e5288dafadd76c0351ed50454" alt="tree view on master 1 mysql load balance mysql"
Create Master 2 MySQL Config File
Since we have made the configuration on mysql master 1, let’s make the configuration on MySQl Master 2 now.
To do this, create a Master 1 folder with this command:
ubuntu@haproxy:~$ mkdir /data/haproxy-mysql/master2
data:image/s3,"s3://crabby-images/2fa7a/2fa7a58b56e9544a543cd6be612f3f405d0d6916" alt="Create master2 folder mysql load balance mysql"
In addition, create a data folder in the Master 2 directory to create a persistent volume on mysql master2.
ubuntu@haproxy:~$ mkdir /data/haproxy-mysql/master2/data
data:image/s3,"s3://crabby-images/d4630/d4630a2f25e0e3e1fa2289a7a793d7fa02674172" alt="create data folder on master2 load balance mysql"
Therefore, don’t forget to create my.cnf file for MySQL Master 2 replication configuration.
ubuntu@haproxy:~$ touch /data/haproxy-mysql/master2/my.cnf
data:image/s3,"s3://crabby-images/7d566/7d56684e431b829214d8d03b66d3be0cfa579c2a" alt="create my.cnf file on master2 load balance mysql"
Add the contents of the my.cnf configuration as below.
# Copyright (c) 2017, Oracle and/or its affiliates. All rights reserved.
#
# This program is free software; you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation; version 2 of the License.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program; if not, write to the Free Software
# Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
#
# The MySQL Server configuration file.
#
# For explanations see
# http://dev.mysql.com/doc/mysql/en/server-system-variables.html
[mysqld]
pid-file = /var/run/mysqld/mysqld.pid
socket = /var/run/mysqld/mysqld.sock
datadir = /var/lib/mysql
secure-file-priv= NULL
server-id=2
binlog_format=ROW
log-bin
skip-name-resolve
# Custom config should go here
!includedir /etc/mysql/conf.d/
In addition, create env file for MySQL Master 2.
ubuntu@haproxy:~$ touch /data/haproxy-mysql/master2/.env
data:image/s3,"s3://crabby-images/fb8c7/fb8c78ef378fa6d23c4bf940fbe78ba2701912ba" alt="env file on master2 load balance mysql"
Let’s fill in the environment username and password in the env file master2.
### WORKSPACE #############################################
TZ=UTC
#MYSQL_DATABASE=master
MYSQL_USER=master
MYSQL_PASSWORD=Master123@
MYSQL_PORT=3306
MYSQL_ROOT_PASSWORD=Mastermaster123
Congratulations! You have created the replication configuration on MySQL Master 2. You can verify by using the tree:
data:image/s3,"s3://crabby-images/10a4a/10a4adedcc26ca2778a94100ab377fbe7a2a02e1" alt="tree directory mysql load balance mysql"
Make Build Up Container for Load Balance on MySQL
Before you make docker-compose, change the permissions of the Master 1 and Master 2 folders according to the user on your server first with chown.
ubuntu@haproxy:/data/haproxy-mysql$ sudo chown -R $(whoami): master*
data:image/s3,"s3://crabby-images/999a1/999a17ff56849ae28ebbd951b62cff01abf9df7e" alt="change permision user mysql load balance"
Not only that, but also change the data access rights so that it can be read by the container with this command (like the problems in Stack Overflow).
ubuntu@haproxy:/data/haproxy-mysql$ chmod -R a+rwx master1/data/
ubuntu@haproxy:/data/haproxy-mysql$ chmod -R a+rwx master2/data/
data:image/s3,"s3://crabby-images/628be/628bebe8e20b20aca23b494bfe37ea486102795b" alt="chmod data mysql load balance mysql"
We have made all the configurations of docker compose and the mysql folder structure, let’s build using docker compose. We go to the haproxy-mysql directory then run the command “docker-compose up -d”.
ubuntu@haproxy:/data/haproxy-mysql$ docker-compose up -d
data:image/s3,"s3://crabby-images/714b6/714b63ca528f2d13d340033b3b603eab8e63fa82" alt="docker compose mysql load balance mysql"
Let’s see the service container can run properly with the command “docker-compose ps”.
ubuntu@haproxy:/data/haproxy-mysql$ docker-compose ps
data:image/s3,"s3://crabby-images/c0473/c0473b516957d1f0230b2cf76db8d862191cb534" alt="docker compose ps"
The container is running fine, then we need to make a replication configuration on mysql. We will make a master master replication first before heading to loadbalance with haproxy.
Create Replications on MySQL
This is where you will enter the configuration stage in MySQL. Enter the MySQL Master 1 container to create the haproxy_user user. Then, enter the MySQL container with the root user and password that you added to the master1/.env file.
ubuntu@haproxy:/data/haproxy-mysql$ docker-compose exec mysql-master1 bash
data:image/s3,"s3://crabby-images/81aaa/81aaa630e35320b2f1b91c04b5b3fdd5fab63fc7" alt="docker compose exec mysql 1"
In this step you will create the user haproxy_user on mysql master1 by logging in using the root user firstL
mysql -u root -p
data:image/s3,"s3://crabby-images/5a71e/5a71ef933d05602e5fb174a164b008a9bbbca507" alt="mysql 1 root login load balance mysql"
Hence, create a new user “replication” on Master 1 MySQL
CREATE USER 'replication'@'%' IDENTIFIED WITH mysql_native_password BY 'Slaverepl123';
data:image/s3,"s3://crabby-images/ac961/ac961d9908c4675e08f209e9ac7724d3769a748b" alt="create user mysql load balance mysql"
Let’s allow user replication so that you can use it as replication in MySQL.
mysql> GRANT REPLICATION SLAVE ON *.* to 'replication'@'%';
data:image/s3,"s3://crabby-images/26d82/26d826c4fe7a92338bbc27bfb2cdc3b0089cc713" alt="grant user replication"
After creating slave on MySQL master1, check master slave status on MySQL Master 1.
mysql> SHOW MASTER STATUS;
data:image/s3,"s3://crabby-images/52b0c/52b0ca10b6b953eea9958a432763bf75e97e00d8" alt="Show master status mysql"
Then, go to MySQL Master 2 to create user replications and allow master master replications. Similarly, enter into the mysql-master2 container now.
ubuntu@haproxy:/data/haproxy-mysql$ docker-compose exec mysql-master2 bash
Subsequently, login to the root user on MySQL Master 2.
root@a46ab42e4e7c:/# mysql -u root -p
data:image/s3,"s3://crabby-images/85ce1/85ce1d0f6b10773e5a399c16a443210eeeedf238" alt="login into mysql master2"
Create user replications on master2 mysql.
mysql> CREATE USER 'replication'@'%' IDENTIFIED WITH mysql_native_password BY 'Slaverepl123';
data:image/s3,"s3://crabby-images/e8f7f/e8f7fdb31b723b5f081f8b0280e2c8fa647dca44" alt="create user replication"
After that, grant replication slave on Master2 MySQL with user MySQL.
mysql> GRANT REPLICATION SLAVE ON *.* to 'replication'@'%';
data:image/s3,"s3://crabby-images/9bb16/9bb169f20baaa0e1da86dde3691663374e538d05" alt="grant all replication"
Congratulations! You have created user replications and granted replication slaves on MySQL Master 2. To confirm, let’s see the status of the master.
mysql> SHOW MASTER STATUS;
data:image/s3,"s3://crabby-images/21bcc/21bccb97adc00cc11105ca66fc8cd0f2ed5c169d" alt="show master status mysql2"
Now, stop the slave on MySQL Master 1.
mysql> STOP SLAVE;
data:image/s3,"s3://crabby-images/9f2aa/9f2aa95667100b97074b2b185a5c4772f641b0b1" alt="stop slave on mysql1"
Accessing The Slave Logs on Master 1
You can access the initial bin log and slave position on MySQ with master slave log on MySQL Master 1.
mysql> CHANGE MASTER TO MASTER_HOST = 'mysql-master1', MASTER_USER = 'replication', MASTER_PASSWORD = 'Slaverepl123', MASTER_LOG_FILE = '6e360f05f522-bin.000001', MASTER_LOG_POS = 673;
data:image/s3,"s3://crabby-images/3e755/3e755c55e83e9cc868d3cb78ee38ff3010481bd7" alt="change master mysql"
Then, start the slave on MySQL Master 1.
mysql> START SLAVE;
After doing that, check the status of the slave on MySQL Master 1.
mysql> SHOW SLAVE STATUS \G;
data:image/s3,"s3://crabby-images/b0b54/b0b54209bc8233d775193089743014c1b6c5c182" alt="mysql2 success replication"
Accessing The Slave Logs on Master 2
You can access the initial bin log and slave positions on MySQL 1 with master slave log on MySQL Master 2.
mysql> CHANGE MASTER TO MASTER_HOST = 'mysql-master2', MASTER_USER = 'replication', MASTER_PASSWORD = 'Slaverepl123', MASTER_LOG_FILE = 'f9ce66dff73f-bin.000001', MASTER_LOG_POS = 673;
data:image/s3,"s3://crabby-images/d250b/d250b9e90089310deb41c01583ecc74f43d647d2" alt="change master mysql"
In addition, start slave on MySQL Master 1.
mysql> START SLAVE;
You can examine the slave status on MySQL Master 1 now.
mysql> SHOW SLAVE STATUS \G;
data:image/s3,"s3://crabby-images/0328f/0328fd12a4da6d22f00cbd42279ba0e9ae4c72c7" alt="Mysql replication success load balance mysql"
Likewise, now you can stop the slave on MySQL Master 2.
data:image/s3,"s3://crabby-images/eca66/eca660f2a167c554781a419fffd4a9ac545f28f7" alt="stop slave on mysql2 load balance mysql"
Creating The HAProxy User for HAProxy docker container to use.
With this step you will get to create the user haproxy_user on Master 1.
mysql> CREATE USER 'haproxy_user'@'%';
data:image/s3,"s3://crabby-images/77983/7798340da6d2df02c3817d00d6904579d167806f" alt="create haproxy user load balance mysql"
Likewise, create a user haproxy_root– where only haproxy instances will be able to use the haproxy_root user.
mysql> CREATE USER 'haproxy_root'@'%' IDENTIFIED WITH mysql_native_password BY 'Haproxymysql123';
data:image/s3,"s3://crabby-images/2559a/2559a4d51e065e1b2e7b226c6fe1422ea05e1081" alt="create user haproxy_root load balance mysql"
To confirm this step, write a query to fetch the users list in MySQL Master 1 which will be synchronized to MySQL Master 2.
mysql> SELECT DISTINCT User FROM mysql.user;
Here’s how the output looks on MySQL Master 1.
data:image/s3,"s3://crabby-images/f494a/f494a0542071e71f6055350170844a631d1cd352" alt="show user mysql1 load balance mysql"
Similarly, this is the output on MySQL Master 2.
data:image/s3,"s3://crabby-images/cf06a/cf06a3d0322a53916fd61c76c24e73f6adda4b12" alt="show user mysql2 load balance mysql"
HAProxy MySQL; Configure The HAProxy to Load Balance
How cool is that? You have successfully created a MySQL replication with Master-Master for your haproxy mysql setup.
Therefore, the logical next step is to configure haproxy to make MySQL Loadbalance with TCO.
To do this, you will have to enter the haproxy container.
ubuntu@haproxy:/data/haproxy-mysql$ docker-compose exec haproxy bash
data:image/s3,"s3://crabby-images/04c0c/04c0cc4069b93699957513859df1be35da413fcd" alt="docker exec haproxy load balance mysql"
Great! With that done, let’s start configuring loadbalance on haproxy.
Firstly, open haproxy configuration path at /etc/haproxy/haproxy.cfg.
root@016fbb5028ee:/# vi /etc/haproxy/haproxy.cfg
data:image/s3,"s3://crabby-images/e66e1/e66e12a8c07909af33033b8070360b7466282439" alt="config haproxy file load balance mysql"
Then, add the configuration to the path /etc/haproxy/haproxy.cfg shown below. This will be used by the HAProxy docker container.
#---------------------------------------------------------------------
# Example configuration for a possible web application. See the
# full configuration options online.
#
# https://www.haproxy.org/download/1.8/doc/configuration.txt
#
#---------------------------------------------------------------------
#---------------------------------------------------------------------
# Global settings
#---------------------------------------------------------------------
global
# to have these messages end up in /var/log/haproxy.log you will
# need to:
#
# 1) configure syslog to accept network log events. This is done
# by adding the '-r' option to the SYSLOGD_OPTIONS in
# /etc/sysconfig/syslog
#
# 2) configure local2 events to go to the /var/log/haproxy.log
# file. A line like the following can be added to
# /etc/sysconfig/syslog
#
# local2.* /var/log/haproxy.log
#
log 127.0.0.1 local2
chroot /var/lib/haproxy
pidfile /var/run/haproxy.pid
maxconn 4000
user haproxy
group haproxy
daemon
# turn on stats unix socket
stats socket /var/lib/haproxy/stats
# utilize system-wide crypto-policies
ssl-default-bind-ciphers PROFILE=SYSTEM
ssl-default-server-ciphers PROFILE=SYSTEM
#---------------------------------------------------------------------
# common defaults that all the 'listen' and 'backend' sections will
# use if not designated in their block
#---------------------------------------------------------------------
defaults
#mode http
log global
option tcplog
retries 2
timeout connect 4s
timeout client 30m #1440m
timeout server 30m #1440m
timeout check 10s
#maxconn 3000
listen stats
mode http
bind *:2999
stats enable
stats uri /stats
stats realm Strictly\ Private
stats auth devops:Haproxymysql2021@
listen mysql-cluster
mode tcp
option tcpka
bind *:3306
option mysql-check user haproxy_user
balance roundrobin
server master1 mysql-master1:3306 check
server master2 mysql-master2:3306 check
Here, it is crucial to validate the haproxy configuration with this command below:
root@016fbb5028ee:/# haproxy -c -V -f /etc/haproxy/haproxy.cfg
data:image/s3,"s3://crabby-images/310ea/310eae97574fefee9b0efface00586278dac2ee9" alt="checking configure haproxy load balance mysql"
After confirming that the haproxy configuration is valid, restart the haproxy container so that the TCP Loadbalance on port 3306 can run on haproxy.
ubuntu@haproxy:/data/haproxy-mysql$ docker-compose restart haproxy
data:image/s3,"s3://crabby-images/ade71/ade71efb9c58142ae1845f538588ecbc2700a22e" alt="restart haproxy container load balance mysql"
After restarting, check if port 3306 is running on the haproxy container.
root@016fbb5028ee:/# netstat -ntlp
data:image/s3,"s3://crabby-images/0d5ae/0d5ae8605de8af9a1c97c816cbf54918a5b4006d" alt="checking port with netstat load balance mysql"
In the same way, try to test with the MySQL client– to test it we need to install MySQL client on haproxy.
root@016fbb5028ee:/# apt update && apt install mysql-client -y
data:image/s3,"s3://crabby-images/e0083/e0083ac1a61746d174a102ca0f77424176f3e6ca" alt="Install mysql-client haproxy load balance mysql"
In the end, check for the server_id using this command:
root@8e949cb0bda6:/# mysql -u root -p -h 127.0.0.1 -e "show variables like 'server_id'";
data:image/s3,"s3://crabby-images/bc4f9/bc4f9cd5ed89f78e9789ff7edb2eefcfcf8c73d6" alt="show varables server_id mysql load balance"
Note: You can also check HAProxy docker container stats on the browser using Server IP.
data:image/s3,"s3://crabby-images/d3384/d3384fad5808ac4ff9fdf7eb47e794b9ccb021c1" alt="haproxy stats dashboard loadbalance"
Conclusion
In conclusion, it is safe to say HAProxy is quite useful for HTTP and TCP Loadbalancers. This way we can use it for TCP loadbalance on MySQL with Master-Master replication on MySQL. Besides being useful for simplifying the development process, you can also make MySQL work powerfully. I hope you liked this post and were able to follow along.
Finally feel free to check out the HAProxy Load balance using MySQL code on github.
Edited by: Syed Umar Bukhari