redis 三主六从高可用docker(不固定ip)

本文介绍如何在Docker Swarm环境下搭建一个三主六从的Redis集群,并解决了因IP变动导致集群异常的问题。通过详细步骤展示了集群的构建过程。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

redis集群(cluster)笔记

redis 三主三从高可用集群docker swarm

redis 三主六从高可用docker(不固定ip)

redis 三主六从高可用dockerswarm高级版(不固定ip)

此博客解决,redis加入集群后,是用于停掉后重启,将nodes.conf中的旧的Ip替换为新的IP,从而达到不会因为IP变化导致集群无法正常使用

跨主机安装rediscluster集群,本文采用swarm的方式,使用同一个网络,然后分别在对应的机器启动

1.环境准备

  • docker
  • docker-compose
  • swarm集群
  • 安装文件

1.1 swarm环境安装

主机IP
node1192.168.56.100
node2192.168.56.101
node3192.168.56.102

在三台分别执行 开放防火墙:

firewall-cmd --permanent --add-rich-rule="rule family="ipv4" source address="192.168.56.100" accept"
firewall-cmd --permanent --add-rich-rule="rule family="ipv4" source address="192.168.56.101" accept"
firewall-cmd --permanent --add-rich-rule="rule family="ipv4" source address="192.168.56.102" accept"
firewall-cmd --reload
firewall-cmd --list-all
systemctl restart docker

1.2 在3台主机上配置swarm

  • 在node1上执行:
docker swarm init --advertise-addr 192.168.56.100
docker swarm join-token manager

返回类似以下内容:

docker swarm join --token SWMTKN-1-614xi9dvksycykobgifxb4pgopc1wwgczwqct5wqkq8zao6tmx-0ds4jj3ozclrr2wukcaoakxso 192.168.56.100:2377
  • 在node2、node3上执行上面的返回结果:
docker swarm join --token SWMTKN-1-2c2xopn2rld8oltcof24sue370681ijhbo3bwcqarjlhq9lkea-2g53o5qn2anre4j9puv4hecrn 192.168.0.101:2377

1.3 创建swarm网络

在node1上执行以下命令:

docker network create -d overlay --attachable redis-net

1.4 查看node

docker node ls

在这里插入图片描述

2.安装文件准备

2.1 redisnode1文件

filePath="/home/redis/redisnode1"
mkdir -p $filePath
cd $filePath
cat > ${filePath}/docker-compose.yml << EOF
version: '3'

networks:
  redis-net:
    external: true 

services:
  redis1:
    image: redis:6.0-alpine
    hostname: redis1
    container_name: redis1
    volumes:
      - "\$PWD/redis1/data:/data"
      - "\$PWD/redis.conf:/etc/redis.conf"
      - "\$PWD/redisStart.sh:/redis/redisStart.sh"
    command: "sh /redis/redisStart.sh"
    restart: always
    privileged: true
    networks:
     - redis-net
  redis2:
    image: redis:6.0-alpine
    hostname: redis2
    container_name: redis2
    volumes:
      - "\$PWD/redis2/data:/data"
      - "\$PWD/redis.conf:/etc/redis.conf"
      - "\$PWD/redisStart.sh:/redis/redisStart.sh"
    command: "sh /redis/redisStart.sh"
    restart: always
    privileged: true
    networks:
     - redis-net
  redis3:
    image: redis:6.0-alpine
    hostname: redis3
    container_name: redis3
    volumes:
      - "\$PWD/redis3/data:/data"
      - "\$PWD/redis.conf:/etc/redis.conf"
      - "\$PWD/redisStart.sh:/redis/redisStart.sh"
    command: "sh /redis/redisStart.sh"
    restart: always
    privileged: true
    networks:
     - redis-net
EOF

cat > ${filePath}/redis.conf << EOF
port 6379
masterauth '123456' 
requirepass '123456'
cluster-enabled yes
cluster-config-file nodes.conf
cluster-node-timeout 5000
appendonly yes
EOF




cat > ${filePath}/redisStart.sh << EOF
#!/bin/sh

CURRENT_DIR=\$(
   cd "\$(dirname "\$0")"
   pwd
)
nowDate=\$(date "+%Y%m%d%H%M%S")
logFileName="redisStart_"\$nowDate.log
touch \$CURRENT_DIR/\${logFileName}
# 日志函数
log(){
	echo "["\$(date "+%Y-%m-%d %H:%M:%S")"]"$1 | tee -a \$CURRENT_DIR/\$logFileName
}

# 通过ifconfig命令获取IP地址
IP=\$(ifconfig eth0 | grep 'inet addr:' | cut -d: -f2 | awk '{print \$1}')
log "Container IP Address is: \${IP}"
CLUSTER_CONFIG="/data/nodes.conf"

if [ -f \${CLUSTER_CONFIG} ]; then
    if [ -z "\${IP}" ]; then
	log "Unable to determine IP address!"
	exit 1
    fi     
	log "Updating my IP to \${IP} in \${CLUSTER_CONFIG}"
	sed -i -e "/myself/s/[0-9]\{1,3\}\.[0-9]\{1,3\}\.[0-9]\{1,3\}\.[0-9]\{1,3\}/\${IP}/g" \${CLUSTER_CONFIG}
fi

redis-server /etc/redis.conf 

EOF


cat > ${filePath}/redisSalveTem.sh << EOF
#!/bin/bash

# redis1 容器内执行集群创建

# redis1  redis4  redis7
# redis2  redis5  redis8
# redis3  redis6  redis9
# 1 5 9
# 4 2 8
# 7 3 6
# 进入 redis1 容器
docker exec -it redis1 /bin/sh
# 指定主从节点交叉分布
# redis1 容器内执行集群创建
# 创建主节点
redis-cli --cluster create REDIS1:6379 REDIS4:6379 REDIS7:6379 -a '123456'

# 确认
yes

# 指定主从节点交叉分布

redis-cli --cluster add-node REDIS5:6379 REDIS1:6379 --cluster-slave -a '123456'
redis-cli --cluster add-node REDIS9:6379 REDIS1:6379 --cluster-slave -a '123456'
redis-cli --cluster add-node REDIS2:6379 REDIS4:6379 --cluster-slave -a '123456'
redis-cli --cluster add-node REDIS8:6379 REDIS4:6379 --cluster-slave -a '123456'
redis-cli --cluster add-node REDIS3:6379 REDIS7:6379 --cluster-slave -a '123456'
redis-cli --cluster add-node REDIS6:6379 REDIS7:6379 --cluster-slave -a '123456'

# 检查集群状态"
redis-cli -c -a '123456'
# 查看集群节点 9个
CLUSTER NODES

# 退出 redis
exit

# 退出容器
exit

EOF

cat > ${filePath}/redisSetIp.sh << EOF
#!/bin/bash
\cp redisSalveTem.sh redisSalve.sh

function getIp(){
    redis1=\$(docker exec -it redis1 sh -c "ping \$1 -c 1 | sed '1{s/[^(]*(//;s/).*//;q}'")
    echo "\${redis1}"
}



echo "redis1=\$(getIp redis1)"
echo "redis2=\$(getIp redis2)"
echo "redis3=\$(getIp redis3)"
echo "redis4=\$(getIp redis4)"
echo "redis5=\$(getIp redis5)"
echo "redis6=\$(getIp redis6)"
echo "redis7=\$(getIp redis7)"
echo "redis8=\$(getIp redis8)"
echo "redis9=\$(getIp redis9)"


echo 'sed -i "s#REDIS1#\$redis1#g" redisSalve.sh'
echo 'sed -i "s#REDIS2#\$redis2#g" redisSalve.sh'
echo 'sed -i "s#REDIS3#\$redis3#g" redisSalve.sh'
echo 'sed -i "s#REDIS4#\$redis4#g" redisSalve.sh'
echo 'sed -i "s#REDIS5#\$redis5#g" redisSalve.sh'
echo 'sed -i "s#REDIS6#\$redis6#g" redisSalve.sh'
echo 'sed -i "s#REDIS7#\$redis7#g" redisSalve.sh'
echo 'sed -i "s#REDIS8#\$redis8#g" redisSalve.sh'
echo 'sed -i "s#REDIS9#\$redis9#g" redisSalve.sh'
echo 'cat redisSalve.sh'
EOF

2.2 redisnode2文件

filePath="/home/redis/redisnode2"
mkdir -p $filePath
cat > ${filePath}/docker-compose.yml << EOF
version: '3'

networks:
  redis-net:
    external: true 

services:
  redis4:
    image: redis:6.0-alpine
    hostname: redis4
    container_name: redis4
    volumes:
      - "\$PWD/redis4/data:/data"
      - "\$PWD/redis.conf:/etc/redis.conf"
      - "\$PWD/redisStart.sh:/redis/redisStart.sh"
    command: "sh /redis/redisStart.sh"
    restart: always
    privileged: true
    networks:
     - redis-net
  redis5:
    image: redis:6.0-alpine
    hostname: redis5
    container_name: redis5
    volumes:
      - "\$PWD/redis5/data:/data"
      - "\$PWD/redis.conf:/etc/redis.conf"
      - "\$PWD/redisStart.sh:/redis/redisStart.sh"
    command: "sh /redis/redisStart.sh"
    restart: always
    privileged: true
    networks:
     - redis-net
  redis6:
    image: redis:6.0-alpine
    hostname: redis6
    container_name: redis6
    volumes:
      - "\$PWD/redis6/data:/data"
      - "\$PWD/redis.conf:/etc/redis.conf"
      - "\$PWD/redisStart.sh:/redis/redisStart.sh"
    command: "sh /redis/redisStart.sh"
    restart: always
    privileged: true
    networks:
     - redis-net
EOF

cat > ${filePath}/redis.conf << EOF
port 6379
masterauth '123456' 
requirepass '123456'
cluster-enabled yes
cluster-config-file nodes.conf
cluster-node-timeout 5000
appendonly yes
EOF





cat > ${filePath}/redisStart.sh << EOF
#!/bin/sh

CURRENT_DIR=\$(
   cd "\$(dirname "\$0")"
   pwd
)
nowDate=\$(date "+%Y%m%d%H%M%S")
logFileName="redisStart_"\$nowDate.log
touch \$CURRENT_DIR/\${logFileName}
# 日志函数
log(){
	echo "["\$(date "+%Y-%m-%d %H:%M:%S")"]"$1 | tee -a \$CURRENT_DIR/\$logFileName
}

# 通过ifconfig命令获取IP地址
IP=\$(ifconfig eth0 | grep 'inet addr:' | cut -d: -f2 | awk '{print \$1}')
log "Container IP Address is: \${IP}"
CLUSTER_CONFIG="/data/nodes.conf"

if [ -f \${CLUSTER_CONFIG} ]; then
    if [ -z "\${IP}" ]; then
	log "Unable to determine IP address!"
	exit 1
    fi     
	log "Updating my IP to \${IP} in \${CLUSTER_CONFIG}"
	sed -i -e "/myself/s/[0-9]\{1,3\}\.[0-9]\{1,3\}\.[0-9]\{1,3\}\.[0-9]\{1,3\}/\${IP}/g" \${CLUSTER_CONFIG}
fi

redis-server /etc/redis.conf 

EOF

2.3 redisnode3文件

filePath="/home/redis/redisnode3"
mkdir -p $filePath
cat > ${filePath}/docker-compose.yml << EOF
version: '3'

networks:
  redis-net:
    external: true 

services:
  redis7:
    image: redis:6.0-alpine
    hostname: redis7
    container_name: redis7
    volumes:
      - "\$PWD/redis7/data:/data"
      - "\$PWD/redis.conf:/etc/redis.conf"
      - "\$PWD/redisStart.sh:/redis/redisStart.sh"
    command: "sh /redis/redisStart.sh"
    restart: always
    privileged: true
    networks:
     - redis-net
  redis8:
    image: redis:6.0-alpine
    hostname: redis8
    container_name: redis8
    volumes:
      - "\$PWD/redis8/data:/data"
      - "\$PWD/redis.conf:/etc/redis.conf"
      - "\$PWD/redisStart.sh:/redis/redisStart.sh"
    command: "sh /redis/redisStart.sh"
    restart: always
    privileged: true
    networks:
     - redis-net
  redis9:
    image: redis:6.0-alpine
    hostname: redis9
    container_name: redis9
    volumes:
      - "\$PWD/redis9/data:/data"
      - "\$PWD/redis.conf:/etc/redis.conf"
      - "\$PWD/redisStart.sh:/redis/redisStart.sh"
    command: "sh /redis/redisStart.sh"
    restart: always
    privileged: true
    networks:
     - redis-net
EOF

cat > ${filePath}/redis.conf << EOF
port 6379
masterauth '123456' 
requirepass '123456'
cluster-enabled yes
cluster-config-file nodes.conf
cluster-node-timeout 5000
appendonly yes
EOF


cat > ${filePath}/redisStart.sh << EOF
#!/bin/sh

CURRENT_DIR=\$(
   cd "\$(dirname "\$0")"
   pwd
)
nowDate=\$(date "+%Y%m%d%H%M%S")
logFileName="redisStart_"\$nowDate.log
touch \$CURRENT_DIR/\${logFileName}
# 日志函数
log(){
	echo "["\$(date "+%Y-%m-%d %H:%M:%S")"]"$1 | tee -a \$CURRENT_DIR/\$logFileName
}

# 通过ifconfig命令获取IP地址
IP=\$(ifconfig eth0 | grep 'inet addr:' | cut -d: -f2 | awk '{print \$1}')
log "Container IP Address is: \${IP}"
CLUSTER_CONFIG="/data/nodes.conf"

if [ -f \${CLUSTER_CONFIG} ]; then
    if [ -z "\${IP}" ]; then
	log "Unable to determine IP address!"
	exit 1
    fi     
	log "Updating my IP to \${IP} in \${CLUSTER_CONFIG}"
	sed -i -e "/myself/s/[0-9]\{1,3\}\.[0-9]\{1,3\}\.[0-9]\{1,3\}\.[0-9]\{1,3\}/\${IP}/g" \${CLUSTER_CONFIG}
fi

redis-server /etc/redis.conf 

EOF

3. 启动 redisnode1

进入 node1服务器

cd /home/redis/redisnode1

docker-compose up -d 

4. 启动 redisnode2

进入 node2服务器

cd /home/redis/redisnode2

docker-compose up -d 

5. 启动 redisnode3

进入 node3服务器

cd /home/redis/redisnode3

docker-compose up -d 

6. 创建集群

进入 node1服务器

bash redisSetIp.sh

在这里插入图片描述

复制上面打印的命令进行执行生成加入集群的语句
在这里插入图片描述

按照步骤一步一步的加入

# redis1 容器内执行集群创建

# redis1  redis4  redis7
# redis2  redis5  redis8
# redis3  redis6  redis9
# 1 5 9
# 4 2 8
# 7 3 6
# 进入 redis1 容器
docker exec -it redis1 /bin/sh
# 指定主从节点交叉分布
# redis1 容器内执行集群创建
# 创建主节点
redis-cli --cluster create 10.0.1.149:6379 10.0.1.157:6379 10.0.1.29:6379 -a '123456'

# 确认
yes

# 指定主从节点交叉分布

redis-cli --cluster add-node 10.0.1.156:6379 10.0.1.149:6379 --cluster-slave -a '123456'
redis-cli --cluster add-node 10.0.1.176:6379 10.0.1.149:6379 --cluster-slave -a '123456'
redis-cli --cluster add-node 10.0.1.150:6379 10.0.1.157:6379 --cluster-slave -a '123456'
redis-cli --cluster add-node 10.0.1.175:6379 10.0.1.157:6379 --cluster-slave -a '123456'
redis-cli --cluster add-node 10.0.1.146:6379 10.0.1.29:6379 --cluster-slave -a '123456'
redis-cli --cluster add-node 10.0.1.158:6379 10.0.1.29:6379 --cluster-slave -a '123456'

# 检查集群状态"
redis-cli -c -a '123456'
# 查看集群节点 9个
CLUSTER NODES

# 退出 redis
exit

# 退出容器

在这里插入图片描述

7.测试用例

7.1 用例1

设置一个值
将对应值的节点关闭
查看集群,查看设置的参数
启动关闭的节点,查看集群是否加入

在这里插入图片描述
在这里插入图片描述在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

### 配置和实现 Redis 从集群的备自动切换 #### 备切换机概述 Redis Cluster 是一种分布式解决方案,在面对单点故障时能够提供高可用性和数据冗余。当一个节点发生故障时,其对应的从节点会通过选举过程晋升为节点,从而保证系统的持续运行和服务中断[^2]。 #### 安装与基本设置 为了构建一个稳定的从架构,首先需要完成基础环境准备: 1. 下载并安装最新版本的 Redis 软件包。 2. 创建个实例分别作为服务器及其相应的备份副本。 #### 开启持久化功能 编辑每个节点的 `redis.conf` 文件以启用 AOF 或 RDB 持久化选项,这有助于防止因意外断电等原因造成的数据丢失问题。 ```bash appendonly yes ``` #### 设置密码保护(可选) 如果希望增加安全性,则可以在所有配置文件中加入如下指令来设定访问控列表 (ACL),并通过命令行参数传递给守护进程: ```bash requirepass your_password_here masterauth same_as_above ``` #### 启动多个 Redis 实例 对于每一个独立的服务端口都需要单独创建一份配置文档,并调整监听地址以及端口号等相关属性;之后利用 Docker 命令或者其他方式启动这些服务实例[^4]: ```dockerfile docker run -p external_port:internal_redis_port \ --name unique_container_name \ -v host_config_path:/etc/redis/redis.conf \ -v host_data_directory:/data \ -d redis_image_tag \ redis-server /etc/redis/redis.conf ``` #### 自定义哨兵监控程序 编写一组 Sentinel 进程用于监测整个网络内的健康状况,一旦检测到某个 master 出现异常即刻触发 failover 流程将 slave 提升为新的 leader[^3]. ```ini sentinel monitor mycluster ip_address_of_master port quorum_value sentinel down-after-milliseconds mycluster timeout_duration sentinel parallel-syncs mycluster max_sync_count sentinel failover-timeout mycluster time_limit_for_failover_operations ``` #### 确认集群状态正常工作 最后一步是验证各个组件之间的通信是否顺畅无阻塞现象存在,可以通过 CLI 工具发送 INFO 请求查看当前拓扑结构图样。 ---
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

liuhm~

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值