Mysql高可用故障转移(双主热备)
基础环境规划:
- | 主机名 | IP地址 | 备注 |
---|---|---|---|
主库 | oraclelinux-mysql-node1 | 192.168.192.171 | keepalive主节点 |
从库 | oraclelinux-mysql-node2 | 192.168.192.172 | keepalive备节点 |
- | - | 192.168.192.170 | 虚拟IP,会在keepalive启动后分配到上面2台机器的主节点上 |
数据库安装
wget https://downloads.mysql.com/archives/get/p/23/file/mysql-8.0.37-linux-glibc2.28-x86_64.tar.xz
tar -xvJf mysql-8.0.37-linux-glibc2.28-x86_64.tar.xz
mv mysql-8.0.37-linux-glibc2.28-x86_64 /usr/local/mysql
groupadd mysql
useradd -r -g mysql mysql
mkdir /usr/local/mysql/{binlog,logs,tmp,data} -p
cd /data
chgrp -R mysql .
cd /usr/local/mysql
chown -R mysql .
chgrp -R mysql .
修改环境变量
vim /etc/profile
最后一行加上
export PATH=$PATH:/usr/local/mysql/bin:/usr/local/mysql/lib
source /etc/profile
初始化mysql数据库
rm -rf /etc/my.cnf
vim /etc/my.cnf
主库my.cnf
[mysql]
port = 3306
socket = /usr/local/mysql/data/mysql.sock
default-character-set = utf8mb4
[mysqld]
server-id = 1 # 任意自然数n,若部署主从同步时需要修改该id避免重复
port = 3306
datadir = /data
basedir = /usr/local/mysql
socket = /usr/local/mysql/data/mysql.sock
tmpdir = /usr/local/mysql/tmp
# MySQL启动用户
user = mysql
# 建议禁用符号链接以防止各种安全风险
symbolic-links = 0
explicit_defaults_for_timestamp = true
# 服务器的默认排序规则。
collation_server=utf8mb4_general_ci
# 默认的身份验证插件。(mysql_native_password,sha256_password,caching_sha2_password)
default-authentication-plugin=mysql_native_password
#mysql_native_password=ON
max_allowed_packet = 16M
# 允许的最大同时客户端连接数。(1-100000)
max_connections = 1000
max_user_connections = 1000
# 在max_connect_errors之后,来自主机的连续连接请求在没有成功连接的情况下被中断,服务器将阻止该主机进行进一步的连接。
max_connect_errors = 1000000
transaction_isolation = READ-COMMITTED
thread-cache-size = 50
open-files-limit = 65535
# 表名不区分大小写(修改后需要重新初始化mysql)
lower_case_table_names=1
# 设置sql_mode(没有NO_AUTO_CREATE_USER了)
sql_mode=STRICT_TRANS_TABLES,NO_ZERO_IN_DATE,NO_ZERO_DATE,ERROR_FOR_DIVISION_BY_ZERO,NO_ENGINE_SUBSTITUTION
group_concat_max_len=102400
character-set-server = utf8mb4
wait_timeout=1800
tmp_table_size=1G
sort_buffer_size=1048576
innodb_buffer_pool_size = 22G
innodb_buffer_pool_instances = 8
innodb_buffer_pool_load_at_startup = 1
innodb_buffer_pool_dump_at_shutdown = 1
innodb_print_all_deadlocks = 1
innodb_log_file_size = 1G
innodb_log_buffer_size = 16M
innodb_flush_log_at_trx_commit = 1
innodb_log_files_in_group = 3
innodb_log_group_home_dir = /usr/local/mysql/logs
gtid-mode=on
log-slave-updates=true
enforce-gtid-consistency=true
slave-parallel-type=LOGICAL_CLOCK
slave-parallel-workers=4
master-info-repository=TABLE
relay-log-info-repository=TABLE
sync-master-info=1
binlog-checksum=CRC32
#rpl_semi_sync_master_enabled=1
#rpl_semi_sync_master_timeout=1000
#rpl_semi_sync_master_trace_level=32
#rpl_semi_sync_master_wait_no_slave=on
#auto_increment_increment=2 #步进值auto_imcrement。一般有n台主MySQL就填n
#auto_increment_offset=1 #起始值。一般填第n台主MySQL。此时为第一台主MySQL
#binlog-ignore=mysql #忽略mysql库
#binlog-ignore=information_schema #忽略information_schema库
#replicate-do-db=test #要同步的数据库,默认所有库
log-bin = /usr/local/mysql/binlog/mysql-bin.log
expire_logs_days = 7
binlog_cache_size = 2097152
max-binlog-size=1G
log_bin_trust_function_creators = 1
sync-binlog = 1
binlog_format=row
log_slave_updates=1
master_info_repository = TABLE
relay_log_info_repository = TABLE
relay_log_recovery = 1
relay_log = /usr/local/mysql/binlog/mysql-relay.log
log-error = /usr/local/mysql/logs/mysqld.log
log-queries-not-using-indexes = 0
slow-query-log = 1
slow-query-log-file = /usr/local/mysql/logs/mysql-slow.log
long_query_time = 3
[mysqld_safe]
pid-file = /usr/local/mysql/data/mysqld.pid
从库my.cnf
[mysql]
port = 3306
socket = /usr/local/mysql/data/mysql.sock
default-character-set = utf8mb4
[mysqld]
port = 3306
datadir = /data
basedir = /usr/local/mysql
socket = /usr/local/mysql/data/mysql.sock
tmpdir = /usr/local/mysql/tmp
user = mysql
# Disabling symbolic-links is recommended to prevent assorted security risks
symbolic-links = 0
explicit_defaults_for_timestamp = true
collation_server=utf8mb4_general_ci
default-authentication-plugin=mysql_native_password
max_allowed_packet = 16M
max_connections = 1000
max_user_connections = 1000
max_connect_errors = 1000000
transaction_isolation = READ-COMMITTED
thread-cache-size = 50
open-files-limit = 65535
lower_case_table_names=1
sql_mode=STRICT_TRANS_TABLES,NO_ZERO_IN_DATE,NO_ZERO_DATE,ERROR_FOR_DIVISION_BY_ZERO,NO_ENGINE_SUBSTITUTION
group_concat_max_len=102400
character-set-server = utf8mb4
wait_timeout=1800
tmp_table_size=1G
sort_buffer_size=1048576
innodb_buffer_pool_size = 22G
innodb_buffer_pool_instances = 8
innodb_buffer_pool_load_at_startup = 1
innodb_buffer_pool_dump_at_shutdown = 1
innodb_print_all_deadlocks = 1
innodb_log_file_size = 1G
innodb_log_buffer_size = 16M
innodb_flush_log_at_trx_commit = 1
innodb_log_files_in_group = 3
innodb_log_group_home_dir = /usr/local/mysql/logs/
gtid-mode=on
log-slave-updates=true
enforce-gtid-consistency=true
slave-parallel-type=LOGICAL_CLOCK
slave-parallel-workers=4
master-info-repository=TABLE
relay-log-info-repository=TABLE
sync-master-info=1
binlog-checksum=CRC32
#rpl_semi_sync_slave_enabled=1
#auto_increment_increment=2 #步进值auto_imcrement。一般有n台主MySQL就填n
#auto_increment_offset=2 #起始值。一般填第n台主MySQL。此时为第一台主MySQL
#binlog-ignore=mysql #忽略mysql库
#binlog-ignore=information_schema #忽略information_schema库
#replicate-do-db=test #要同步的数据库,默认所有库
log-bin = /usr/local/mysql/binlog/mysql-bin.log
expire_logs_days = 7
binlog_cache_size = 2097152
max-binlog-size=1G
log_bin_trust_function_creators = 1
sync-binlog = 1
binlog_format=row
server-id = 2
log_slave_updates=1
# 从节点
#read_only=on #只读
#super_read_only=on
master_info_repository = TABLE
relay_log_info_repository = TABLE
relay_log_recovery = 1
relay_log = /usr/local/mysql/binlog/mysql-relay.log
log-error = /usr/local/mysql/logs/mysqld.log
log-queries-not-using-indexes = 1
slow-query-log = 1
slow-query-log-file = /usr/local/mysql/logs/mysql-slow.log
long_query_time = 3
[mysqld_safe]
pid-file = /usr/local/mysql/data/mysqld.pid
初始化配置
chmod 644 /etc/my.cnf
chown -R mysql:mysql /etc/my.cnf
chmod 755 /data
chown -R mysql:mysql /data
# 如果要重新初始化则需要删掉datadir下的所有文件
rm -rf /data/*
rm -rf /usr/local/mysql/data/*
rm -rf /usr/local/mysql/logs/*
touch /usr/local/mysql/logs/mysqld.log
touch /usr/local/mysql/logs/mysqld_safe.log
chown -R mysql:mysql /usr/local/mysql/logs/
# 初始化配置
mysqld --defaults-file=/etc/my.cnf --initialize
# 映射mysql服务
ln -s /usr/local/mysql/bin/mysql /usr/bin
开启mysql服务
# mysqld_safe方式启动
/usr/local/mysql/bin/mysqld_safe --defaults-file=/etc/my.cnf --user=mysql > /usr/local/mysql/logs/mysqld_safe.log 2>&1 &
查看默认账号和密码:
cat /usr/local/mysql/logs/mysqld.log | grep "A temporary password"
通过本地localhost登录
mysql -hlocalhost -uroot -p
修改密码
-- 先改密码之后才能查询mysql.user
ALTER USER 'root'@'localhost' IDENTIFIED WITH mysql_native_password BY 'mysql@123456';
flush privileges;
select host , user , plugin ,authentication_string from mysql.user;
quit;
放行防火墙
#centos/oracle
firewall-cmd --add-port=3306/tcp --permanent --zone=public
#重启防火墙(修改配置后要重启防火墙)
firewall-cmd --reload
# ubuntu/debian
ufw allow 3306/tcp
ufw reload
设置自动启动
vim /etc/systemd/system/mysql.service
修改为我们自定义的配置
[Unit]
Description=MySQL Server
After=network.service
[Install]
WantedBy=multi-user.target
[Service]
User=mysql
Group=mysql
Type=forking
TimeoutSec=0
PermissionsStartOnly=true
ExecStart=/usr/local/mysql/bin/mysqld_safe --defaults-file=/etc/my.cnf --user=mysql --log-error=/usr/local/mysql/logs/mysqld_safe.log
LimitNOFILE=5000
Restart=on-failure
RestartSec=10
RestartPreventExitStatus=1
PrivateTmp=false
# 重载系统服务
systemctl daemon-reload
# 实现开机自启
systemctl enable mysql
关闭SELINUX:
vim /etc/selinux/config
#SELINUX=enforcing 修改为 SELINUX=disabled
重启服务器验证
reboot
#重新连接服务器,查看mysql 服务的状态:
systemctl status mysql
#查看端口
netstat -tnap
#查看mysql启动情况
ps aux|grep mysql
创建主从同步进程(需要修改对应IP地址)
通过本地localhost登录
mysql -hlocalhost -uroot -p
主库、从库操作:
CREATE USER 'rep_user'@'192.168.192.172' IDENTIFIED WITH mysql_native_password BY 'repluser@123456';
CREATE USER 'rep_user'@'192.168.192.171' IDENTIFIED WITH mysql_native_password BY 'repluser@123456';
GRANT REPLICATION SLAVE,replication client ON *.* TO 'rep_user'@'192.168.192.172' ;
GRANT REPLICATION SLAVE,replication client ON *.* TO 'rep_user'@'192.168.192.171' ;
install plugin rpl_semi_sync_master SONAME 'semisync_master.so';
set global rpl_semi_sync_master_enabled=1;
set global rpl_semi_sync_master_timeout=1000;
INSTALL PLUGIN rpl_semi_sync_slave SONAME 'semisync_slave.so';
SET GLOBAL rpl_semi_sync_slave_enabled=1;
查看生成的binlog日志,记录下来日志名字和起始位置
查询主库、从库
show master status\G
主库输出:
mysql> show master status\G
*************************** 1. row ***************************
File: mysql-bin.000003
Position: 1319
Binlog_Do_DB:
Binlog_Ignore_DB:
Executed_Gtid_Set: 6ac2190e-a271-11ef-8ec6-bc2411799364:1-6
1 row in set (0.00 sec)
从库输出:
mysql> show master status\G
*************************** 1. row ***************************
File: mysql-bin.000004
Position: 1319
Binlog_Do_DB:
Binlog_Ignore_DB:
Executed_Gtid_Set: 6c6540e1-a271-11ef-985e-bc241191a340:1-6
1 row in set (0.00 sec)
主库使用从库的日志起点
CHANGE MASTER TO MASTER_HOST='192.168.192.172', MASTER_USER='rep_user',MASTER_PASSWORD='repluser@123456',MASTER_LOG_FILE='mysql-bin.000004', MASTER_LOG_POS=1319;
从库使用主库的日志起点
CHANGE MASTER TO MASTER_HOST='192.168.192.171',MASTER_USER='rep_user',MASTER_PASSWORD='repluser@123456',MASTER_LOG_FILE='mysql-bin.000003',MASTER_LOG_POS=1319;
启动并查询同步状态
start slave; #启动
show slave status\G #查看
主库操作-开启远程访问
select host , user , plugin ,authentication_string from mysql.user;
update mysql.user set host = '%',plugin='mysql_native_password' where user='root';
flush privileges;
quit;
部署keepalived做高可用配置
Keepalived是Linux下一个轻量级的高可用解决方案。 官网地址:https://www.keepalived.org
在线安装
#centos/oracle
yum install keepalived -y
# ubuntu/debian
apt-get install keepalived -y
源码编译
1、官网下载源码包,解压
wget https://www.keepalived.org/software/keepalived-2.3.1.tar.gz
tar xvf keepalived-2.3.1.tar.gz
#centos/oracle
yum install -y gcc gcc-c++ wget popt-devel openssl openssl-devel
yum install -y libnl3 libnl3-devel
# ubuntu/debian
sudo apt-get install make -y
sudo apt-get install wget -y
sudo apt-get install gcc -y
sudo apt-get install g++ -y
sudo apt-get install libssl-dev -y
sudo apt-get install openssl -y
sudo apt-get install libpopt-dev -y
sudo apt-get install libnl-3-dev -y
sudo apt-get install libnl-genl-3-dev -y
2、编译安装,这里需要指定一下安装路径
cd keepalived-2.3.1/
./configure --prefix=/usr/local/keepalived
make && make install
配置文件位置:/etc/keepalived 启动位置:/usr/local/keepalived/sbin/keepalived keepalived 日志: tail -22f /var/log/messages | grep Keepalived
3、拷贝配置文件和启动脚本到相应的目录(debian/ubuntu系列忽略),并设置开机启动
centos/oracle
mkdir /etc/keepalived
#cp /usr/local/keepalived/etc/keepalived/keepalived.conf /etc/keepalived/keepalived.conf
#cp /usr/local/keepalived/etc/rc.d/init.d/keepalived /etc/init.d/
cp /usr/local/keepalived/etc/sysconfig/keepalived /etc/sysconfig/
cp /usr/local/keepalived/sbin/keepalived /usr/sbin/
chkconfig keepalived on
ubuntu/debian
cat <<EOF | sudo tee /etc/systemd/system/keepalived.service
[Unit]
Description=Keepalived
After=network.target
[Service]
ExecStart=/usr/local/keepalived/sbin/keepalived -D
ExecReload=/bin/kill -HUP $MAINPID
KillMode=process
Restart=on-failure
Type=forking
[Install]
WantedBy=multi-user.target
EOF
# 重载系统服务
systemctl daemon-reload
systemctl enable keepalived
MySQL状态监测脚本
mkdir /etc/keepalived
#check_mysql.sh:
cat>/etc/keepalived/check_mysql.sh<<EOF
#!/bin/bash
service_name="mysqld"
get_pid=`pidof $service_name`
if [ "${get_pid}" == "" ]; then
echo "mysql_status=1"
exit 0
else
systemctl stop keepalived.service
fi
EOF
chmod +x /etc/keepalived/check_mysql.sh
#shutdown.sh:
cat>/etc/keepalived/shutdown.sh<<EOF
systemctl stop keepalived.service
EOF
chmod +x /etc/keepalived/shutdown.sh