syncReplication.c 4.5 KB
Newer Older
M
Minghao Li 已提交
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15
/*
 * Copyright (c) 2019 TAOS Data, Inc. <jhtao@taosdata.com>
 *
 * This program is free software: you can use, redistribute, and/or modify
 * it under the terms of the GNU Affero General Public License, version 3
 * or later ("AGPL"), as published by the Free Software Foundation.
 *
 * This program is distributed in the hope that it will be useful, but WITHOUT
 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
 * FITNESS FOR A PARTICULAR PURPOSE.
 *
 * You should have received a copy of the GNU Affero General Public License
 * along with this program. If not, see <http://www.gnu.org/licenses/>.
 */

M
Minghao Li 已提交
16
#include "syncReplication.h"
M
Minghao Li 已提交
17
#include "syncIndexMgr.h"
M
Minghao Li 已提交
18
#include "syncMessage.h"
M
Minghao Li 已提交
19
#include "syncRaftEntry.h"
M
Minghao Li 已提交
20
#include "syncRaftLog.h"
M
Minghao Li 已提交
21
#include "syncRaftStore.h"
M
Minghao Li 已提交
22
#include "syncUtil.h"
M
Minghao Li 已提交
23

M
Minghao Li 已提交
24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47
// TLA+ Spec
// AppendEntries(i, j) ==
//    /\ i /= j
//    /\ state[i] = Leader
//    /\ LET prevLogIndex == nextIndex[i][j] - 1
//           prevLogTerm == IF prevLogIndex > 0 THEN
//                              log[i][prevLogIndex].term
//                          ELSE
//                              0
//           \* Send up to 1 entry, constrained by the end of the log.
//           lastEntry == Min({Len(log[i]), nextIndex[i][j]})
//           entries == SubSeq(log[i], nextIndex[i][j], lastEntry)
//       IN Send([mtype          |-> AppendEntriesRequest,
//                mterm          |-> currentTerm[i],
//                mprevLogIndex  |-> prevLogIndex,
//                mprevLogTerm   |-> prevLogTerm,
//                mentries       |-> entries,
//                \* mlog is used as a history variable for the proof.
//                \* It would not exist in a real implementation.
//                mlog           |-> log[i],
//                mcommitIndex   |-> Min({commitIndex[i], lastEntry}),
//                msource        |-> i,
//                mdest          |-> j])
//    /\ UNCHANGED <<serverVars, candidateVars, leaderVars, logVars>>
M
Minghao Li 已提交
48
//
M
Minghao Li 已提交
49
int32_t syncNodeAppendEntriesPeers(SSyncNode* pSyncNode) {
M
Minghao Li 已提交
50 51
  assert(pSyncNode->state == TAOS_SYNC_STATE_LEADER);

M
Minghao Li 已提交
52 53 54 55
  syncIndexMgrLog2("==syncNodeAppendEntriesPeers== pNextIndex", pSyncNode->pNextIndex);
  syncIndexMgrLog2("==syncNodeAppendEntriesPeers== pMatchIndex", pSyncNode->pMatchIndex);
  logStoreSimpleLog2("==syncNodeAppendEntriesPeers==", pSyncNode->pLogStore);

M
Minghao Li 已提交
56
  int32_t ret = 0;
M
Minghao Li 已提交
57
  for (int i = 0; i < pSyncNode->peersNum; ++i) {
M
Minghao Li 已提交
58
    SRaftId* pDestId = &(pSyncNode->peersId[i]);
M
Minghao Li 已提交
59

M
Minghao Li 已提交
60 61
    // set prevLogIndex
    SyncIndex nextIndex = syncIndexMgrGetIndex(pSyncNode->pNextIndex, pDestId);
M
Minghao Li 已提交
62
    SyncIndex preLogIndex = nextIndex - 1;
M
Minghao Li 已提交
63

M
Minghao Li 已提交
64
    // set preLogTerm
M
Minghao Li 已提交
65 66
    SyncTerm preLogTerm = 0;
    if (preLogIndex >= SYNC_INDEX_BEGIN) {
M
Minghao Li 已提交
67
      SSyncRaftEntry* pPreEntry = pSyncNode->pLogStore->getEntry(pSyncNode->pLogStore, preLogIndex);
M
Minghao Li 已提交
68 69
      assert(pPreEntry != NULL);

M
Minghao Li 已提交
70
      preLogTerm = pPreEntry->term;
M
Minghao Li 已提交
71
      syncEntryDestory(pPreEntry);
M
Minghao Li 已提交
72
    }
M
Minghao Li 已提交
73

M
Minghao Li 已提交
74 75 76 77 78 79
    // batch optimized
    // SyncIndex lastIndex = syncUtilMinIndex(pSyncNode->pLogStore->getLastIndex(pSyncNode->pLogStore), nextIndex);

    SyncAppendEntries* pMsg = NULL;
    SSyncRaftEntry*    pEntry = logStoreGetEntry(pSyncNode->pLogStore, nextIndex);
    if (pEntry != NULL) {
M
Minghao Li 已提交
80 81
      pMsg = syncAppendEntriesBuild(pEntry->bytes);
      assert(pMsg != NULL);
M
Minghao Li 已提交
82

M
Minghao Li 已提交
83 84 85 86 87 88
      // add pEntry into msg
      uint32_t len;
      char*    serialized = syncEntrySerialize(pEntry, &len);
      assert(len == pEntry->bytes);
      memcpy(pMsg->data, serialized, len);

wafwerar's avatar
wafwerar 已提交
89
      taosMemoryFree(serialized);
M
Minghao Li 已提交
90 91 92 93
      syncEntryDestory(pEntry);

    } else {
      // maybe overflow, send empty record
M
Minghao Li 已提交
94 95
      pMsg = syncAppendEntriesBuild(0);
      assert(pMsg != NULL);
M
Minghao Li 已提交
96
    }
M
Minghao Li 已提交
97

M
Minghao Li 已提交
98
    assert(pMsg != NULL);
M
Minghao Li 已提交
99 100
    pMsg->srcId = pSyncNode->myRaftId;
    pMsg->destId = *pDestId;
M
Minghao Li 已提交
101
    pMsg->term = pSyncNode->pRaftStore->currentTerm;
M
Minghao Li 已提交
102 103 104 105
    pMsg->prevLogIndex = preLogIndex;
    pMsg->prevLogTerm = preLogTerm;
    pMsg->commitIndex = pSyncNode->commitIndex;

M
Minghao Li 已提交
106 107
    syncAppendEntriesLog2("==syncNodeAppendEntriesPeers==", pMsg);

M
Minghao Li 已提交
108
    // send AppendEntries
M
Minghao Li 已提交
109
    syncNodeAppendEntries(pSyncNode, pDestId, pMsg);
M
Minghao Li 已提交
110
    syncAppendEntriesDestroy(pMsg);
M
Minghao Li 已提交
111 112
  }

M
Minghao Li 已提交
113 114 115 116 117 118 119 120
  return ret;
}

int32_t syncNodeReplicate(SSyncNode* pSyncNode) {
  // start replicate
  int32_t ret = syncNodeAppendEntriesPeers(pSyncNode);
  return ret;
}
M
Minghao Li 已提交
121 122 123 124 125 126 127 128 129 130

int32_t syncNodeAppendEntries(SSyncNode* pSyncNode, const SRaftId* destRaftId, const SyncAppendEntries* pMsg) {
  sTrace("syncNodeAppendEntries pSyncNode:%p ", pSyncNode);
  int32_t ret = 0;

  SRpcMsg rpcMsg;
  syncAppendEntries2RpcMsg(pMsg, &rpcMsg);
  syncNodeSendMsgById(destRaftId, pSyncNode, &rpcMsg);
  return ret;
}