网站目录和数据库自动备份方案

下面是排版整理后的文章。通过分段、格式化和添加适当的小标题,使得内容更加清晰易读:


简要方案概述

昨天用 GPT 写了一个方案,测试了一晚上,通过 GPT 分析了日志情况,暂未发现什么毛病。

方案简要:
网站目录使用 rsync​ 同步增量更新,数据库使用 mysqldump​ 进行增量备份。
目前这个方案只适合做备份,“主机宕机,备用机马上接管业务”这种情况还是不太适合(后续会有关于这个的想法,欢迎各位大佬指点)。

我的做法

准备两台设备,分别安装 rsync​ 工具:

apt install rsync  # Debian/Ubuntu 系统
或者
yum install rsync  # CentOS/RHEL 系统

设备A 为主用机,设备B 为备用机。

1. 设备A创建SSH公钥

命令:

ssh-keygen -t rsa -b 2048

默认按回车即可生成密钥。
1734678597712

然后通过设备A将公钥传到设备B上:

ssh-copy-id -i /root/.ssh/id_rsa.pub -p 设备A的SSH端口 root@设备B的IP

image

测试是否配置成功:

ssh -p 设备B的SSH端口 root@设备B的IP

image

如果成功,进入设备B的SSH时不需要输入密码,说明公钥上传成功。

2. 使用rsync同步目录

同步网站目录的命令如下:

rsync -avz --delete -e "ssh -p 设备B的SSH端口" /www/wwwroot/ root@设备B的IP:/www/wwwroot/backup/

可以尝试在设备A上执行测试命令:

rsync -avz --delete -e "ssh -p 22" /root/test/ root@设备B的IP:/root/test2/

执行后,若提示报错,通常是设备B没有安装 rsync​,只需在设备B上安装即可。

安装命令:

apt install rsync  # Debian/Ubuntu 系统
或者
yum install rsync  # CentOS/RHEL 系统

执行成功后的效果如下:

成功执行rsync

3. 设置定期自动执行

在宝塔面板中设置定时任务,定期执行 rsync​ 同步命令。
image

例如,命令:

rsync -avz --delete -e "ssh -p 22" /www/wwwroot/ root@127.0.0.2:/www/wwwroot/

该命令将设备A中 /www/wwwroot/​ 目录中的所有文件同步到设备B的 /www/wwwroot/​ 目录。

提醒:
设备B的 /www/wwwroot/​ 中不应包含与设备A不同的文件,避免同步时被删除。

数据库增量备份方案

数据库备份脚本:
创建一个新的脚本文件 mysqlback_shanghai.sh​,放在 /www​ 目录中。

给脚本执行权限:

chmod +x /www/mysqlback_shanghai.sh

脚本内容如下:

#!/bin/bash

# 源数据库连接信息
SOURCE_HOST="127.0.0.1"
SOURCE_USER="root"
SOURCE_PASSWORD="root"

# 目标数据库连接信息
TARGET_HOST="设备B的IP"
TARGET_USER="同root权限的mysql账号"
TARGET_PASSWORD="密码"

# 要排除的系统数据库列表
EXCLUDED_DATABASES="information_schema|performance_schema|mysql|sys"

# 1. 获取源数据库和目标数据库的数据库列表
get_databases() {
    echo "Fetching database lists from source and target servers..."
    # 获取源服务器的数据库列表并排除系统数据库
    SOURCE_DATABASES=$(mysql --host=$SOURCE_HOST --user=$SOURCE_USER --password=$SOURCE_PASSWORD -e "SHOW DATABASES;" | tail -n +2 | grep -Ev "$EXCLUDED_DATABASES")

    # 获取目标服务器的数据库列表并排除系统数据库
    TARGET_DATABASES=$(mysql --host=$TARGET_HOST --user=$TARGET_USER --password=$TARGET_PASSWORD -e "SHOW DATABASES;" | tail -n +2 | grep -Ev "$EXCLUDED_DATABASES")
}

# 2. 同步新数据库
sync_new_databases() {
    echo "Syncing new databases..."
    for DB in $SOURCE_DATABASES; do
        if ! echo "$TARGET_DATABASES" | grep -q "$DB"; then
            echo "New database detected: $DB. Synchronizing..."
            mysqldump --host=$SOURCE_HOST --user=$SOURCE_USER --password=$SOURCE_PASSWORD --single-transaction --databases $DB --skip-lock-tables | \
            mysql --host=$TARGET_HOST --user=$TARGET_USER --password=$TARGET_PASSWORD
            if [ $? -eq 0 ]; then
                echo "Database $DB synchronized successfully."
            else
                echo "Error occurred while synchronizing database $DB."
            fi
        fi
    done
}

# 3. 同步更新的数据库(如果数据发生变化)
sync_updated_databases() {
    echo "Syncing updated databases..."
    for DB in $SOURCE_DATABASES; do
        if echo "$TARGET_DATABASES" | grep -q "$DB"; then
            echo "Syncing updated database: $DB..."
            mysqldump --host=$SOURCE_HOST --user=$SOURCE_USER --password=$SOURCE_PASSWORD --single-transaction --databases $DB --skip-lock-tables | \
            mysql --host=$TARGET_HOST --user=$TARGET_USER --password=$TARGET_PASSWORD
            if [ $? -eq 0 ]; then
                echo "Database $DB synchronized successfully."
            else
                echo "Error occurred while synchronizing database $DB."
            fi
        fi
    done
}

# 4. 主程序
echo "Starting automatic database synchronization..."

# 获取数据库列表
get_databases

# 同步新创建的数据库
sync_new_databases

# 同步已经更新的数据库
sync_updated_databases

echo "Automatic database synchronization completed."

创建目标数据库用户

在设备B的宝塔面板中,创建一个与 root​ 同权的 MySQL 用户。
1734682902125_d
image
image
1734683759348_d

操作步骤:

  1. 在宝塔数据库管理中,新增一个用户,填写设备A的IP地址,并设置访问权限。
  2. 配置 MySQL 的 3306 端口,并允许设备A的IP访问。
    1734691627662_d

修改脚本中的目标数据库连接信息:

TARGET_HOST="设备B的IP"
TARGET_USER="同root权限的mysql账号"
TARGET_PASSWORD="密码"

执行备份脚本

在设备A上执行脚本:

bash /www/mysqlback_shanghai.sh

1734700588302_d

GPT日志分析结果:

从日志来看,数据库和用户权限的同步成功,所有相关的数据库已正确同步到目标服务器。

需要关注的地方:

  1. 密码暴露警告:
    虽然不会直接影响同步操作,但为了安全,建议使用 MySQL 配置文件(如 ~/.my.cnf​)或环境变量来隐藏密码,而不是直接在命令行中使用。
  2. 没有显示错误:
    日志中没有错误信息,意味着操作完成得很顺利。

后续建议:

  • 如果数据库同步过程中没有权限问题且数据已经正确同步,可以忽略这些警告。
  • 为了进一步提升安全性,建议通过配置文件管理密码,避免在命令行中暴露密码。

完成设置

1734691532134_d
image

在设备A的宝塔面板中,设置同步命令定时执行,确保备份任务自动化。
1734701238594_d


1734701525963_d
GPT排的版,应该会好看点

这个内容目前只能实现:网站目录和数据库增量同步,
我自己能想到后期要优化的话,希望能做到:直接和主服务器相同配置,连nginx中的网站配置都同步起来,宝塔板面上能显示数据库密码,服务器能同时向多台设备发起同步请求(感觉数据库很麻烦),然后emmm以后再说。
目前还是主打能用就用,后面有需求了再去折腾

阅读剩余
THE END