哨兵(sentinel)是Redis的高可用性(High Availability)的解决方案:

由一个或多个sentinel实例组成sentinel集群可以监视一个或多个主服务器和多个从服务器。

当主服务器进入下线状态时,sentinel可以将该主服务器下的某一从服务器升级为主服务器继续提供服务,从而保证redis的高可用性。

部署方案


搭建配置

在一台机器上采用伪分布式的方式部署。(生产环境应该是多台机器)

根据上面的部署方案搭建如下:

  • Redis-Master :127.0.0.1 6379

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    mkdir /opt/lagou/servers/redis-master

    cd /opt/lagou/software/redis-5.0.5/src

    make install PREFIX=/opt/lagou/servers/redis-master

    cp /opt/lagou/software/redis-5.0.5/redis.conf /opt/lagou/servers/redis-master/bin/

    cd /opt/lagou/servers/redis-master/bin/

    vim redis.conf

    # 将`daemonize`由`no`改为`yes`设置为后台启动
    daemonize yes
    # 默认绑定的是回环地址,不能被其他机器访问;表示允许所有远程访问
    bind 0.0.0.0
    # 是否开启保护模式,由yes该为no,关闭保护模式,可以外部访问
    protected-mode no
    # 日志地址生成后在 bin 目录下可找到
    logfile ./redis.log
    # redis的连接密码
    requirepass 12345678
    # slave 服务连接 master 的密码(跟redis密码一致)
    masterauth 12345678
  • Redis-Slaver1:127.0.0.1 6380

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    mkdir /opt/lagou/servers/redis-slaver1

    cp -r /opt/lagou/servers/redis-master/* /opt/lagou/servers/redis-slaver1

    cd /opt/lagou/servers/redis-slaver1/bin

    vim redis.conf

    port 6380
    # 主redis服务器ip 主redis端口
    replicaof 192.168.91.121 6379
  • Redis-Slaver2:127.0.0.1 6381

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    mkdir /opt/lagou/servers/redis-slaver2

    cp -r /opt/lagou/servers/redis-master/* /opt/lagou/servers/redis-slaver2

    cd /opt/lagou/servers/redis-slaver2/bin

    vim redis.conf

    port 6381
    replicaof 192.168.91.121 6379
  • Redis-Sentinel1:127.0.0.1 26379

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    29
    30
    31
    32
    33
    34
    35
    36
    37
    mkdir /opt/lagou/servers/redis-sentinel1

    cp -r /opt/lagou/servers/redis-master/* /opt/lagou/servers/redis-sentinel1

    cp /opt/lagou/software/redis-5.0.5/sentinel.conf /opt/lagou/servers/redis-sentinel1/bin

    cd /opt/lagou/servers/redis-sentinel1/bin

    vim sentinel.conf

    # 哨兵sentinel实例运行的端口 默认26379
    port 26379
    # 将`daemonize`由`no`改为`yes`
    daemonize yes
    # 哨兵sentinel监控的redis主节点的 ip port
    # master-name 可以自己命名的主节点名字 只能由字母A-z、数字0-9 、这三个字符".-_"组成。
    # quorum 当这些quorum个数sentinel哨兵认为master主节点失联 那么这时 客观上认为主节点失联了
    # sentinel monitor <master-name> <ip> <redis-port> <quorum>
    sentinel monitor mymaster 192.168.91.121 6379 2
    # 当在Redis实例中开启了requirepass foobared 授权密码 这样所有连接Redis实例的客户端都要提 供密码
    # 设置哨兵sentinel 连接主从的密码 注意必须为主从设置一样的验证密码
    # sentinel auth-pass <master-name> <password>
    sentinel auth-pass mymaster MySUPER--secret-0123passw0rd
    # 指定多少毫秒之后 主节点没有应答哨兵sentinel 此时 哨兵主观上认为主节点下线 默认30秒,改成3秒
    # sentinel down-after-milliseconds <master-name> <milliseconds>
    sentinel down-after-milliseconds mymaster 3000
    # 这个配置项指定了在发生failover主备切换时最多可以有多少个slave同时对新的master进行 同步, 这个数字越小,完成failover所需的时间就越长, 但是如果这个数字越大,就意味着越多的slave因为replication而不可用。 可以通过将这个值设为 1 来保证每次只有一个slave 处于不能处理命令请求的状态。
    # sentinel parallel-syncs <master-name> <numslaves>
    sentinel parallel-syncs mymaster 1
    # 故障转移的超时时间 failover-timeout 可以用在以下这些方面:
    #1. 同一个sentinel对同一个master两次failover之间的间隔时间。
    #2. 当一个slave从一个错误的master那里同步数据开始计算时间。直到slave被纠正为向正确的 master那里同步数据时。
    #3.当想要取消一个正在进行的failover所需要的时间。
    #4.当进行failover时,配置所有slaves指向新的master所需的最大时间。不过,即使过了这个超时, slaves依然会被正确配置为指向master,但是就不按parallel-syncs所配置的规则来了
    # 默认三分钟
    # sentinel failover-timeout <master-name> <milliseconds>
    sentinel failover-timeout mymaster 180000
  • Redis-Sentinel2:127.0.0.1 26380

    1
    2
    3
    4
    5
    6
    7
    8
    9
    mkdir /opt/lagou/servers/redis-sentinel2

    cp -r /opt/lagou/servers/redis-sentinel1/* /opt/lagou/servers/redis-sentinel2

    cd /opt/lagou/servers/redis-sentinel2/bin

    vim sentinel.conf

    port 26380
  • Redis-Sentinel3:127.0.0.1 26381

    1
    2
    3
    4
    5
    6
    7
    8
    9
    mkdir /opt/lagou/servers/redis-sentinel3

    cp -r /opt/lagou/servers/redis-sentinel1/* /opt/lagou/servers/redis-sentinel3

    cd /opt/lagou/servers/redis-sentinel3/bin

    vim sentinel.conf

    port 26381
  • 配置好后依次执行

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    29
    30
    31
    #启动redis-master
    cd /opt/lagou/servers/redis-master/bin
    ./redis-server redis.conf

    #启动redis-slaver1
    cd /opt/lagou/servers/redis-slaver1/bin
    ./redis-server redis.conf

    #启动redis-sentinel2
    cd /opt/lagou/servers/redis-slaver2/bin
    ./redis-server redis.conf

    #启动redis-sentinel1
    cd /opt/lagou/servers/redis-sentinel1/bin
    ./redis-sentinel sentinel.conf

    #启动redis-sentinel2
    ./redis-sentinel sentinel.conf

    #启动redis-sentinel3
    ./redis-sentinel sentinel.conf

    #查看启动状态
    ps -ef |grep redis
    root 3602 1 0 01:33 ? 00:00:00 ./redis-server *:6379
    root 3647 1 0 01:37 ? 00:00:00 ./redis-server *:6380
    root 3717 1 0 01:40 ? 00:00:00 ./redis-server *:6381
    root 3760 1 0 01:42 ? 00:00:00 ./redis-sentinel *:26379 [sentinel]
    root 3765 1 0 01:42 ? 00:00:00 ./redis-sentinel *:26380 [sentinel]
    root 3770 1 0 01:42 ? 00:00:00 ./redis-sentinel *:26381 [sentinel]
    root 3783 2261 0 01:42 pts/0 00:00:00 grep --color=auto redis

执行流程

  • 启动并初始化Sentinel

    Sentinel是一个特殊的Redis服务器,不会进行持久化

    Sentinel实例启动后,每个Sentinel会创建2个连向主服务器的网络连接

    命令连接:用于向主服务器发送命令,并接收响应;

    订阅连接:用于订阅主服务器的—sentinel—:hello频道。


  • 获取主服务器信息

    Sentinel默认每10s一次,向被监控的主服务器发送info命令,获取主服务器和其下属从服务器的信息。

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    127.0.0.1:6379> info
    # Server
    redis_version:5.0.5
    os:Linux 3.10.0-229.el7.x86_64 x86_64
    run_id:a4e06ab61b4116660aa37b85079ed482b0b695b1

    # Replication
    role:master
    connected_slaves:2
    slave0:ip=127.0.0.1,port=6380,state=online,offset=1571684,lag=1
    slave1:ip=127.0.0.1,port=6381,state=online,offset=1571551,lag=1
    master_replid:366322125dd7dc9bc95ed3467cfec841c112e207
    master_replid2:0000000000000000000000000000000000000000
    master_repl_offset:1571684
    second_repl_offset:-1
    repl_backlog_active:1
    repl_backlog_size:1048576
    repl_backlog_first_byte_offset:523109
    repl_backlog_histlen:1048576
  • 获取从服务器信息

    当Sentinel发现主服务器有新的从服务器出现时,Sentinel还会向从服务器建立命令连接和订阅连接。

    在命令连接建立之后,Sentinel还是默认10s一次,向从服务器发送info命令,并记录从服务器的信息。


    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    # Server
    redis_version:5.0.5
    os:Linux 3.10.0-229.el7.x86_64 x86_64
    run_id:e289b3286352aaf8cc9f1ac7ebcc6d36131b8321

    # Replication
    role:slave
    master_host:127.0.0.1
    master_port:6379
    master_link_status:up
    master_last_io_seconds_ago:0
    master_sync_in_progress:0
    slave_repl_offset:1699595
    slave_priority:100
    slave_read_only:1
    connected_slaves:0
    master_replid:366322125dd7dc9bc95ed3467cfec841c112e207
    master_replid2:0000000000000000000000000000000000000000
    master_repl_offset:1699595
    second_repl_offset:-1
    repl_backlog_active:1
    repl_backlog_size:1048576
    repl_backlog_first_byte_offset:651020
    repl_backlog_histlen:1048576
  • 向主服务器和从服务器发送消息(以订阅的方式)

    默认情况下,Sentinel每2s一次,向所有被监视的主服务器和从服务器所订阅的—sentinel—:hello频道上发送消息,消息中会携带Sentinel自身的信息和主服务器的信息。

    1
    PUBLISH _sentinel_:hello "< s_ip > < s_port >< s_runid >< s_epoch > < m_name > < m_ip >< m_port ><m_epoch>"
  • 接收来自主服务器和从服务器的频道信息

    当Sentinel与主服务器或者从服务器建立起订阅连接之后,Sentinel就会通过订阅连接,向服务器发送以下命令:

    1
    subscribe —sentinel—:hello

    Sentinel彼此之间只创建命令连接,而不创建订阅连接,因为Sentinel通过订阅主服务器或从服务器,就可以感知到新的Sentinel的加入,而一旦新Sentinel加入后,相互感知的Sentinel通过命令连接来通信就可以了。

  • 检测主观下线状态

    Sentinel每秒一次向所有与它建立了命令连接的实例(主服务器、从服务器和其他Sentinel)发送PING命令

    实例在down-after-milliseconds毫秒内返回无效回复(除了+PONG、-LOADING、-MASTERDOWN外)

    实例在down-after-milliseconds毫秒内无回复(超时)

    Sentinel就会认为该实例主观下线(SDown)

  • 检查客观下线状态

    当一个Sentinel将一个主服务器判断为主观下线后Sentinel会向同时监控这个主服务器的所有其他Sentinel发送查询命令

    1
    SENTINEL is-master-down-by-addr <ip> <port> <current_epoch> <runid>

    其他Sentinel回复

    1
    <down_state>< leader_runid >< leader_epoch >

    判断它们是否也认为主服务器下线。如果达到Sentinel配置中的quorum数量的Sentinel实例都判断主服务器为主观下线,则该主服务器就会被判定为客观下线(ODown)。

  • 选举Leader Sentinel

    当一个主服务器被判定为客观下线后,监视这个主服务器的所有Sentinel会通过选举算法(raft),选出一个Leader Sentinel去执行failover(故障转移)操作。

哨兵leader选举

Raft

Raft协议是用来解决分布式系统一致性问题的协议。

Raft协议描述的节点共有三种状态:Leader, Follower, Candidate。

term:Raft协议将时间切分为一个个的Term(任期),可以认为是一种“逻辑时间”。

选举流程:

  • Raft采用心跳机制触发Leader选举

  • 系统启动后,全部节点初始化为Follower,term为0。

  • 节点如果收到了RequestVote或者AppendEntries,就会保持自己的Follower身份

  • 节点如果一段时间内没收到AppendEntries消息,在该节点的超时时间内还没发现Leader,Follower就会转换成Candidate,自己开始竞选Leader。

    一旦转化为Candidate,该节点立即开始下面几件事情:

    • 增加自己的term。

    • 启动一个新的定时器。

    • 给自己投一票。

    • 向所有其他节点发送RequestVote,并等待其他节点的回复。

  • 如果在计时器超时前,节点收到多数节点的同意投票,就转换成Leader。同时向所有其他节点发送AppendEntries,告知自己成为了Leader。

  • 每个节点在一个term内只能投一票,采取先到先得的策略,Candidate前面说到已经投给了自己,Follower会投给第一个收到RequestVote的节点。Raft协议的定时器采取随机超时时间,这是选举Leader的关键。在同一个term内,先转为Candidate的节点会先发起投票,从而获得多数票。

Sentinel的leader选举流程

  • 1、某Sentinel认定master客观下线后,该Sentinel会先看看自己有没有投过票,如果自己已经投过票给其他Sentinel了,在一定时间内自己就不会成为Leader。

  • 2、如果该Sentinel还没投过票,那么它就成为Candidate。

  • 3、Sentinel需要完成几件事情:

    • 更新故障转移状态为start

    • 当前epoch加1,相当于进入一个新term,在Sentinel中epoch就是Raft协议中的term。

    • 向其他节点发送 is-master-down-by-addr 命令请求投票。命令会带上自己的epoch。

    -给自己投一票(leader、leader_epoch)

  • 4、当其它哨兵收到此命令时,可以同意或者拒绝它成为领导者;(通过判断epoch)

  • 5、Candidate会不断的统计自己的票数,直到他发现认同他成为Leader的票数超过一半而且超过它配置的quorum,这时它就成为了Leader。

  • 6、其他Sentinel等待Leader从slave选出master后,检测到新的master正常工作后,就会去掉客观下线的标识。

故障转移

当选举出Leader Sentinel后,Leader Sentinel会对下线的主服务器执行故障转移操作,主要有三个步骤:

  1. 它会将失效 Master 的其中一个 Slave 升级为新的 Master , 并让失效 Master 的其他 Slave 改为复制新的 Master ;

  2. 当客户端试图连接失效的 Master 时,集群也会向客户端返回新 Master 的地址,使得集群可以使用现在的 Master 替换失效 Master 。

  3. Master 和 Slave 服务器切换后, Master 的 redis.conf 、 Slave 的 redis.conf 和 sentinel.conf 的配置文件的内容都会发生相应的改变,即, Master 主服务器的 redis.conf 配置文件中会多一行 replicaof 的配置, sentinel.conf 的监控目标会随之调换。

主服务器的选择

哨兵leader根据以下规则从客观下线的主服务器的从服务器中选择出新的主服务器。

  1. 过滤掉主观下线的节点

  2. 选择slave-priority最高的节点,如果由则返回没有就继续选择

  3. 选择出复制偏移量最大的系节点,因为复制偏移量越大则数据复制的越完整,如果由就返回了,没有就继续

  4. 选择run_id最小的节点,因为run_id越小说明重启次数越少