You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

24 KiB

24 | 从哨兵Leader选举学习Raft协议实现

你好,我是蒋德钧。

上节课我给你介绍了Raft协议的基本流程以及哨兵实例工作的基本过程。哨兵是通过serverCron函数的周期性执行进而在serverCron中调用sentinelTimer函数实现周期性处理哨兵相关的时间事件。而sentinelTimer函数处理的时间事件就包括了对哨兵监听的每个主节点它会通过调用sentinelHandleRedisInstance函数来检查主节点的在线状态并在主节点客观下线时进行故障切换。

另外我还带你了解了sentinelHandleRedisInstance函数执行过程的前三步操作分别是重连断连的实例、周期性给实例发送检测命令检测实例是否主观下线这也分别对应了sentinelReconnectInstance、sentinelSendPeriodicCommands和sentinelCheckSubjectivelyDown这三个函数你可以再回顾下。

那么今天这节课我接着来给你介绍sentinelHandleRedisInstance函数执行过程中的剩余操作分别是检测主节点是否客观下线、判断是否需要执行故障切换以及需要故障切换时的哨兵Leader选举的具体过程。

学完这节课的内容你就可以对哨兵工作的过程有个全面了解了。并且你可以掌握如何在代码层面实现Raft协议来完成Leader选举。这样当你日后在分布式系统中实现分布式共识时这部分内容就能帮助指导你的代码设计与实现了。

接下来,我们先来看下主节点的客观下线判断。

主节点客观下线判断

现在我们知道哨兵在sentinelHandleRedisInstance函数中会调用sentinelCheckObjectivelyDown函数在sentinel.c文件中来检测主节点是否客观下线。

而sentinelCheckObjectivelyDown函数在执行时除了会检查当前哨兵对主节点主观下线的判断结果还需要结合监听相同主节点的其他哨兵对主节点主观下线的判断结果。它把这些判断结果综合起来才能做出主节点客观下线的最终判断。

从代码实现层面来看,在哨兵用来记录主节点信息的sentinelRedisInstance结构体中,本身已经用哈希表保存了监听同一主节点的其他哨兵实例,如下所示:

typedef struct sentinelRedisInstance {
…
dict *sentinels;
…
}

这样一来sentinelCheckObjectivelyDown函数通过遍历主节点记录的sentinels哈希表就可以获取其他哨兵实例对同一主节点主观下线的判断结果。这也是因为sentinels哈希表中保存的哨兵实例它们同样使用了sentinelRedisInstance这个结构体而这个结构体的成员变量flags会记录哨兵对主节点主观下线的判断结果。

具体来说sentinelCheckObjectivelyDown函数会使用quorum变量来记录判断主节点为主观下线的哨兵数量。如果当前哨兵已经判断主节点为主观下线那么它会先把quorum值置为1。然后它会依次判断其他哨兵的flags变量检查是否设置了SRI_MASTER_DOWN的标记。如果设置了它就会把quorum值加1。

当遍历完sentinels哈希表后sentinelCheckObjectivelyDown函数会判断quorum值是否大于等于预设定的quorum阈值这个阈值保存在了主节点的数据结构中也就是master->quorum而这个阈值是在sentinel.conf配置文件中设置的。

如果实际的quorum值大于等于预设的quorum阈值sentinelCheckObjectivelyDown函数就判断主节点为客观下线并**设置变量odown为1**而这个变量就是用来表示当前哨兵对主节点客观下线的判断结果的。

这部分的判断逻辑如下代码所示,你可以看下:

void sentinelCheckObjectivelyDown(sentinelRedisInstance *master) {
…
//当前主节点已经被当前哨兵判断为主观下线
if (master->flags & SRI_S_DOWN) {
   quorum = 1; //当前哨兵将quorum值置为1
   
   di = dictGetIterator(master->sentinels);
   while((de = dictNext(di)) != NULL) {  //遍历监听同一主节点的其他哨兵
      sentinelRedisInstance *ri = dictGetVal(de);
      if (ri->flags & SRI_MASTER_DOWN) quorum++;
   }
   dictReleaseIterator(di);
   //如果quorum值大于预设的quorum阈值那么设置odown为1。
   if (quorum >= master->quorum) odown = 1;
}

另外,这里我也画了一张图,展示了该判断逻辑,你可以再来回顾下。
图片

那么一旦sentinelCheckObjectivelyDown函数判断主节点客观下线了它就会调用sentinelEvent函数发送+odown事件消息然后在主节点的flags变量中设置SRI_O_DOWN标记,如下所示:

//判断主节点为客观下线
if (odown) {
   //如果没有设置SRI_O_DOWN标记
   if ((master->flags & SRI_O_DOWN) == 0) {
    sentinelEvent(LL_WARNING,"+odown",master,"%@ #quorum %d/%d",
                quorum, master->quorum); //发送+odown事件消息
    master->flags |= SRI_O_DOWN;  //在主节点的flags中记录SRI_O_DOWN标记
    master->o_down_since_time = mstime(); //记录判断客观下线的时间
   }
}

也就是说,sentinelCheckObjectivelyDown函数是通过遍历监听同一主节点的其他哨兵的flags变量来判断主节点是否客观下线的。

不过你看完刚才的代码可能会有一个疑问在上节课学习的sentinelCheckSubjectivelyDown函数中如果哨兵判断主节点为主观下线是会在主节点的flags变量中设置SRI_S_DOWN标记,如下所示:

//哨兵已判断主节点为主观下线
…
//对应主节点的sentinelRedisInstance结构中flags没有记录主观下线
if ((ri->flags & SRI_S_DOWN) == 0) {
   …
   ri->flags |= SRI_S_DOWN;  //在主节点的flags中记录主观下线的标记
}

但是sentinelCheckObjectivelyDown函数是检查监听同一主节点的其他哨兵flags变量中的SRI_MASTER_DOWN标记那么其他哨兵的SRI_MASTER_DOWN标记是如何设置的呢?

这就和sentinelAskMasterStateToOtherSentinels函数在sentinel.c文件中有关系了下面我们来具体了解下这个函数。

sentinelAskMasterStateToOtherSentinels函数

sentinelAskMasterStateToOtherSentinels函数的主要目的是向监听同一主节点的其他哨兵发送is-master-down-by-addr命令进而询问其他哨兵对主节点的状态判断。

它会调用redisAsyncCommand函数async.c文件中依次向其他哨兵发送sentinel is-master-down-by-addr命令同时它设置了收到该命令返回结果的处理函数为sentinelReceiveIsMasterDownReply在sentinel.c文件中如下所示

void sentinelAskMasterStateToOtherSentinels(sentinelRedisInstance *master, int flags) {
…
di = dictGetIterator(master->sentinels);
//遍历监听同一主节点的其他哨兵
while((de = dictNext(di)) != NULL) {
   sentinelRedisInstance *ri = dictGetVal(de);
   …
   //发送sentinel is-master-down-by-addr命令
   retval = redisAsyncCommand(ri->link->cc,
             sentinelReceiveIsMasterDownReply, ri,
             "%s is-master-down-by-addr %s %s %llu %s",
             sentinelInstanceMapCommand(ri,"SENTINEL"),
             master->addr->ip, port,
             sentinel.current_epoch,
             (master->failover_state > SENTINEL_FAILOVER_STATE_NONE) ?
                 sentinel.myid : "*");
}
…
}

另外从代码中我们可以看到sentinel is-master-down-by-addr命令中还包括主节点IP、主节点端口号、当前纪元sentinel.current_epoch和实例ID。下面展示的就是这个命令的格式

sentinel is-master-down-by-addr 主节点IP 主节点端口 当前epoch 实例ID

在这其中哨兵会根据当前主节点所处的状态来设置实例ID。如果主节点已经要开始进行故障切换了那么实例ID会被设置为当前哨兵自身的ID否则就会被设置为*号。

这里你需要注意的是,主节点的数据结构是使用了master->failover_state来记录故障切换的状态其初始值为SENTINEL_FAILOVER_STATE_NONE对应的数值为0当主节点开始故障切换时这个状态值就会大于SENTINEL_FAILOVER_STATE_NONE了。

好了在了解了sentinelAskMasterStateToOtherSentinels函数的基本执行过程之后我们还需要知道sentinelAskMasterStateToOtherSentinels函数向其他哨兵发出了sentinel is-master-down-by-addr命令后其他哨兵是如何处理的呢

sentinel is-master-down-by-addr命令的处理

其实哨兵对于sentinel开头的命令都是在sentinelCommand函数在sentinel.c文件中进行处理的。sentinelCommand函数会根据sentinel命令后面跟的不同子命令来执行不同的分支而is-master-down-by-addr就是一条子命令。

在is-master-down-by-addr子命令对应的代码分支中sentinelCommand函数会根据命令中的主节点IP和端口号来获取主节点对应的sentinelRedisInstance结构体。

紧接着它会判断主节点的flags变量中是否有SRI_S_DOWN和SRI_MASTER标记也就是说sentinelCommand函数会检查当前节点是否的确是主节点以及哨兵是否已经将该节点标记为主观下线了。如果条件符合那么它会设置isdown变量为1而这个变量表示的就是哨兵对主节点主观下线的判断结果。

然后sentinelCommand函数会把当前哨兵对主节点主观下线的判断结果返回给发送sentinel命令的哨兵。它返回的结果主要包含三部分内容分别是当前哨兵对主节点主观下线的判断结果哨兵Leader的ID,以及哨兵Leader所属的纪元

sentinelCommand函数对sentinel命令处理的基本过程如下所示

void sentinelCommand(client *c) {
…
// is-master-down-by-addr子命令对应的分支
else if (!strcasecmp(c->argv[1]->ptr,"is-master-down-by-addr")) {
…
//当前哨兵判断主节点为主观下线
if (!sentinel.tilt && ri && (ri->flags & SRI_S_DOWN) && (ri->flags & SRI_MASTER))
   isdown = 1;
…
addReplyMultiBulkLen(c,3); //哨兵返回的sentinel命令处理结果中包含三部分内容
addReply(c, isdown ? shared.cone : shared.czero); //如果哨兵判断主节点为主观下线第一部分为1否则为0
addReplyBulkCString(c, leader ? leader : "*"); //第二部分是Leader ID或者是*
addReplyLongLong(c, (long long)leader_epoch); //第三部分是Leader的纪元
…}
…}

你也可以参考下图:
图片

好了到这里你就已经知道哨兵会通过sentinelAskMasterStateToOtherSentinels函数向监听同一节点的其他哨兵发送sentinel is-master-down-by-addr命令来获取其他哨兵对主节点主观下线的判断结果。而其他哨兵是使用sentinelCommand函数来处理sentinel is-master-down-by-addr命令并在命令处理的返回结果中包含自己对主节点主观下线的判断结果。

不过从刚才的代码中你也可以看到在其他哨兵返回的sentinel命令处理结果中会包含哨兵Leader的信息。其实这是因为sentinelAskMasterStateToOtherSentinels函数发送的sentinel is-master-down-by-addr命令本身也可以用来触发哨兵Leader选举。这个我稍后会给你介绍。

那么我们再回到前面讲主节点客观下线判断时提出的问题sentinelCheckObjectivelyDown函数要检查监听同一主节点的其他哨兵flags变量中的SRI_MASTER_DOWN标记但是其他哨兵的SRI_MASTER_DOWN标记是如何设置的呢

这实际上是和哨兵在sentinelAskMasterStateToOtherSentinels函数中向其他哨兵发送sentinel is-master-down-by-addr命令时设置的命令结果处理函数sentinelReceiveIsMasterDownReply有关。

sentinelReceiveIsMasterDownReply函数

在sentinelReceiveIsMasterDownReply函数中它会判断其他哨兵返回的回复结果。回复结果会包含我刚才介绍的三部分内容分别是当前哨兵对主节点主观下线的判断结果、哨兵Leader的ID以及哨兵Leader所属的纪元。这个函数会进一步检查其中第一部分内容“当前哨兵对主节点主观下线的判断结果”是否为1。

如果是的话这就表明对应的哨兵已经判断主节点为主观下线了那么当前哨兵就会把自己记录的对应哨兵的flags设置为SRI_MASTER_DOWN。

下面的代码就展示了sentinelReceiveIsMasterDownReply函数判断其他哨兵回复结果的执行逻辑你可以看下。

//r是当前哨兵收到的其他哨兵的命令处理结果
//如果返回结果包含三部分内容,并且第一,二,三部分内容的类型分别是整数、字符串和整数
if (r->type == REDIS_REPLY_ARRAY && r->elements == 3 &&
        r->element[0]->type == REDIS_REPLY_INTEGER &&
        r->element[1]->type == REDIS_REPLY_STRING &&
        r->element[2]->type == REDIS_REPLY_INTEGER)
{
        ri->last_master_down_reply_time = mstime();
        //如果返回结果第一部分的值为1则在对应哨兵的flags中设置SRI_MASTER_DOWN标记
        if (r->element[0]->integer == 1) {
            ri->flags |= SRI_MASTER_DOWN;
        }

所以到这里你就可以知道一个哨兵调用sentinelCheckObjectivelyDown函数是直接检查其他哨兵的flags是否有SRI_MASTER_DOWN标记而哨兵又是通过sentinelAskMasterStateToOtherSentinels函数向其他哨兵发送sentinel is-master-down-by-addr命令从而询问其他哨兵对主节点主观下线的判断结果的并且会根据命令回复结果在结果处理函数sentinelReceiveIsMasterDownReply中设置其他哨兵的flags为SRI_MASTER_DOWN。下图也展示了这个执行逻辑你可以再来整体回顾下。

图片

那么,掌握了这个执行逻辑后,我们再来看下,哨兵选举是什么时候开始执行的。

哨兵选举

这里为了了解哨兵选举的触发我们先来复习下在上节课我讲过的sentinelHandleRedisInstance函数中针对主节点的调用关系如下图所示

图片

从图中可以看到sentinelHandleRedisInstance会先调用sentinelCheckObjectivelyDown函数再调用sentinelStartFailoverIfNeeded函数判断是否要开始故障切换如果sentinelStartFailoverIfNeeded函数的返回值为非0值那么sentinelAskMasterStateToOtherSentinels函数会被调用。否则的话sentinelHandleRedisInstance就直接调用sentinelFailoverStateMachine函数并再次调用sentinelAskMasterStateToOtherSentinels函数。

那么在这个调用关系中sentinelStartFailoverIfNeeded会判断是否要进行故障切换它的判断条件有三个,分别是:

  • 主节点的flags已经标记了SRI_O_DOWN
  • 当前没有在执行故障切换;
  • 如果已经开始故障切换那么开始时间距离当前时间需要超过sentinel.conf文件中的sentinel failover-timeout配置项的2倍。

这三个条件都满足后sentinelStartFailoverIfNeeded就会调用sentinelStartFailover函数开始启动故障切换而sentinelStartFailover会将主节点的failover_state设置为SENTINEL_FAILOVER_STATE_WAIT_START同时在主节点的flags设置SRI_FAILOVER_IN_PROGRESS标记表示已经开始故障切换如下所示

void sentinelStartFailover(sentinelRedisInstance *master) {
…
master->failover_state = SENTINEL_FAILOVER_STATE_WAIT_START;
master->flags |= SRI_FAILOVER_IN_PROGRESS;
…
}

而一旦sentinelStartFailover函数将主节点的failover_state设置为SENTINEL_FAILOVER_STATE_WAIT_START后接下来sentinelFailoverStateMachine函数就会执行状态机来完成实际的切换。不过在实际切换前sentinelAskMasterStateToOtherSentinels函数会被调用。

看到这个调用关系你可能会有个疑问sentinelAskMasterStateToOtherSentinels函数是用来向其他哨兵询问对主节点主观下线的判断结果的如果sentinelStartFailoverIfNeeded判断要开始执行故障切换那么为什么还要调用sentinelAskMasterStateToOtherSentinels函数呢

其实这就和sentinelAskMasterStateToOtherSentinels函数的另一个作用有关了这个函数除了会用来向其他哨兵询问对主节点状态的判断它还可以用来向其他哨兵发起Leader选举

在刚才给你介绍这个函数时我提到它会给其他哨兵发送sentinel is-master-down-by-addr命令这个命令包括主节点IP、主节点端口号、当前纪元sentinel.current_epoch和实例ID。其中如果主节点的failover_state已经不再是SENTINEL_FAILOVER_STATE_NONE那么实例ID会被设置为当前哨兵的ID。

而在sentinel命令处理函数中如果检测到sentinel命令中的实例ID不为*号,那么就会调用sentinelVoteLeader函数来进行Leader选举。

//当前实例为主节点并且sentinel命令的实例ID不等于*号
if (ri && ri->flags & SRI_MASTER && strcasecmp(c->argv[5]->ptr,"*")) {
   //调用sentinelVoteLeader进行哨兵Leader选举
   leader = sentinelVoteLeader(ri,(uint64_t)req_epoch, c->argv[5]->ptr,
                                            &leader_epoch);
}

下面我们来具体了解下这个sentinelVoteLeader函数。

sentinelVoteLeader函数

sentinelVoteLeader函数会实际执行投票逻辑这里我通过一个例子来给你说明。

假设哨兵A判断主节点master客观下线了它现在向哨兵B发起投票请求哨兵A的ID是req_runid。那么哨兵B在执行sentinelVoteLeader函数时这个函数会判断哨兵A的纪元req_epoch、哨兵B的纪元sentinel.current_epoch以及master记录的Leader的纪元master->leader_epoch。按照Raft协议的定义哨兵A就是Candidate节点而哨兵B就是Follower节点。

我在上节课给你介绍Raft协议时有提到过Candidate发起投票都是有轮次记录的Follower在一轮投票中只能投一票。这里的纪元正是起到了轮次记录的作用。而sentinelVoteLeader函数判断纪元也是按照Raft协议的要求让Follower在一轮中只能投一票。

那么,sentinelVoteLeader函数让哨兵B投票的条件是master记录的Leader的纪元小于哨兵A的纪元同时哨兵A的纪元要大于或等于哨兵B的纪元。这两个条件保证了哨兵B还没有投过票否则的话sentinelVoteLeader函数就直接返回当前master中记录的Leader ID了这也是哨兵B之前投过票后记录下来的。

下面的代码展示了刚才介绍的这部分逻辑,你可以看下。

if (req_epoch > sentinel.current_epoch) {
   sentinel.current_epoch = req_epoch;
   …
   sentinelEvent(LL_WARNING,"+new-epoch",master,"%llu",
            (unsigned long long) sentinel.current_epoch);
}
 
if (master->leader_epoch < req_epoch && sentinel.current_epoch <= req_epoch)
{
        sdsfree(master->leader);
        master->leader = sdsnew(req_runid);
        master->leader_epoch = sentinel.current_epoch;
        …
}
return master->leader ? sdsnew(master->leader) : NULL;

那么现在你就了解了sentinelVoteLeader函数是如何使用纪元判断来按照Raft协议完成哨兵Leader选举的了。

接下来发起投票的哨兵仍然是通过sentinelReceiveIsMasterDownReply函数来处理其他哨兵对Leader投票的返回结果。这个返回结果就像刚才给你介绍的它的第二、三部分内容是哨兵Leader的ID和哨兵Leader所属的纪元。发起投票的哨兵就可以从这个结果中获得其他哨兵对Leader的投票结果了。

最后发起投票的哨兵在调用了sentinelAskMasterStateToOtherSentinels函数让其他哨兵投票后会执行sentinelFailoverStateMachine函数。

如果主节点开始执行故障切换了那么主节点的failover_state会被设置成SENTINEL_FAILOVER_STATE_WAIT_START。在这种状态下sentinelFailoverStateMachine函数会调用sentinelFailoverWaitStart函数。而sentinelFailoverWaitStart函数又会调用sentinelGetLeader函数来判断发起投票的哨兵是否为哨兵Leader。发起投票的哨兵要想成为Leader必须满足两个条件

  • 一是,获得超过半数的其他哨兵的赞成票
  • 二是获得超过预设的quorum阈值的赞成票数。

这两个条件也可以从sentinelGetLeader函数中的代码片段看到如下所示。

//voters是所有哨兵的个数max_votes是获得的票数
 voters_quorum = voters/2+1;  //赞成票的数量必须是超过半数以上的哨兵个数
//如果赞成票数不到半数的哨兵个数或者少于quorum阈值那么Leader就为NULL
 if (winner && (max_votes < voters_quorum || max_votes < master->quorum))
        winner = NULL;
//确定最终的Leader
winner = winner ? sdsnew(winner) : NULL;

下图就展示了刚才介绍的确认哨兵Leader时的调用关系你可以看下。

图片

好了到这里最终的哨兵Leader就能被确定了。

小结

好了,今天这节课的内容就到这里,我们来小结下。

今天这节课我在上节课的基础上重点给你介绍了哨兵工作过程中的客观下线判断以及Leader选举。因为这个过程涉及哨兵之间的交互询问所以并不容易掌握你需要好好关注以下我提到的重点内容。

首先客观下线的判断涉及三个标记的判断分别是主节点flags中的SRI_S_DOWN和SRI_O_DOWN以及哨兵实例flags中的SRI_MASTER_DOWN我画了下面这张表展示了这三个标记的设置函数和条件你可以再整体回顾下。

图片

而一旦哨兵判断主节点客观下线了,那么哨兵就会调用sentinelAskMasterStateToOtherSentinels函数进行哨兵Leader选举。这里你需要注意的是向其他哨兵询问主节点主观下线状态以及向其他哨兵发起Leader投票都是通过sentinel is-master-down-by-addr命令实现的而Redis源码是用了同一个函数sentinelAskMasterStateToOtherSentinels来发送该命令所以你在阅读源码时要注意区分sentinelAskMasterStateToOtherSentinels发送的命令是查询主节点主观下线状态还是进行投票

最后哨兵Leader选举的投票是在sentinelVoteLeader函数中完成的为了符合Raft协议的规定sentinelVoteLeader函数在执行时主要是要比较哨兵的纪元以及master记录的Leader纪元这样才能满足Raft协议对Follower在一轮投票中只能投一票的要求。

好了到今天这节课我们就了解了哨兵Leader选举的过程你可以看到虽然哨兵选举的最后执行逻辑就是在一个函数中但是哨兵选举的触发逻辑是包含在了哨兵的整个工作过程中的所以我们也需要掌握这个过程中的其他操作比如主观下线判断、客观下线判断等。

每课一问

哨兵在sentinelTimer函数中调用sentinelHandleDictOfRedisInstances函数对每个主节点都执行sentinelHandleRedisInstance函数并且还会对主节点的所有从节点也执行sentinelHandleRedisInstance函数那么哨兵会判断从节点的主观下线和客观下线吗