TDengine/source/libs/sync/src/syncVoteMgr.c

290 lines
9.7 KiB
C
Raw Normal View History

2022-02-26 18:24:50 +00:00
/*
* 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/>.
*/
#include "syncVoteMgr.h"
2022-03-08 09:07:29 +00:00
#include "syncUtil.h"
// SVotesGranted -----------------------------
static void voteGrantedClearVotes(SVotesGranted *pVotesGranted) {
memset(pVotesGranted->isGranted, 0, sizeof(pVotesGranted->isGranted));
pVotesGranted->votes = 0;
}
2022-03-05 07:03:49 +00:00
SVotesGranted *voteGrantedCreate(SSyncNode *pSyncNode) {
2022-03-25 16:29:53 +00:00
SVotesGranted *pVotesGranted = taosMemoryMalloc(sizeof(SVotesGranted));
2022-06-21 08:02:36 +00:00
ASSERT(pVotesGranted != NULL);
2022-03-05 07:03:49 +00:00
memset(pVotesGranted, 0, sizeof(SVotesGranted));
2022-03-08 09:07:29 +00:00
pVotesGranted->replicas = &(pSyncNode->replicasId);
pVotesGranted->replicaNum = pSyncNode->replicaNum;
voteGrantedClearVotes(pVotesGranted);
2022-03-05 07:03:49 +00:00
pVotesGranted->term = 0;
2022-03-08 09:07:29 +00:00
pVotesGranted->quorum = pSyncNode->quorum;
2022-03-05 07:03:49 +00:00
pVotesGranted->toLeader = false;
pVotesGranted->pSyncNode = pSyncNode;
return pVotesGranted;
}
void voteGrantedDestroy(SVotesGranted *pVotesGranted) {
if (pVotesGranted != NULL) {
2022-03-25 16:29:53 +00:00
taosMemoryFree(pVotesGranted);
2022-03-05 07:03:49 +00:00
}
}
2022-05-24 12:23:11 +00:00
void voteGrantedUpdate(SVotesGranted *pVotesGranted, SSyncNode *pSyncNode) {
pVotesGranted->replicas = &(pSyncNode->replicasId);
pVotesGranted->replicaNum = pSyncNode->replicaNum;
voteGrantedClearVotes(pVotesGranted);
pVotesGranted->term = 0;
pVotesGranted->quorum = pSyncNode->quorum;
pVotesGranted->toLeader = false;
pVotesGranted->pSyncNode = pSyncNode;
}
2022-03-05 07:03:49 +00:00
bool voteGrantedMajority(SVotesGranted *pVotesGranted) {
bool ret = pVotesGranted->votes >= pVotesGranted->quorum;
return ret;
}
void voteGrantedVote(SVotesGranted *pVotesGranted, SyncRequestVoteReply *pMsg) {
2022-06-21 08:02:36 +00:00
ASSERT(pMsg->voteGranted == true);
ASSERT(pMsg->term == pVotesGranted->term);
ASSERT(syncUtilSameId(&pVotesGranted->pSyncNode->myRaftId, &pMsg->destId));
2022-03-08 09:07:29 +00:00
int j = -1;
for (int i = 0; i < pVotesGranted->replicaNum; ++i) {
if (syncUtilSameId(&((*(pVotesGranted->replicas))[i]), &(pMsg->srcId))) {
j = i;
break;
}
}
2022-06-21 08:02:36 +00:00
ASSERT(j != -1);
ASSERT(j >= 0 && j < pVotesGranted->replicaNum);
2022-03-08 09:07:29 +00:00
if (pVotesGranted->isGranted[j] != true) {
++(pVotesGranted->votes);
pVotesGranted->isGranted[j] = true;
}
2022-06-21 08:02:36 +00:00
ASSERT(pVotesGranted->votes <= pVotesGranted->replicaNum);
2022-03-05 07:03:49 +00:00
}
void voteGrantedReset(SVotesGranted *pVotesGranted, SyncTerm term) {
pVotesGranted->term = term;
2022-03-08 09:07:29 +00:00
voteGrantedClearVotes(pVotesGranted);
2022-03-05 07:03:49 +00:00
pVotesGranted->toLeader = false;
}
2022-03-08 09:07:29 +00:00
cJSON *voteGranted2Json(SVotesGranted *pVotesGranted) {
char u64buf[128] = {0};
2022-03-08 09:07:29 +00:00
cJSON *pRoot = cJSON_CreateObject();
2022-03-16 02:54:06 +00:00
if (pVotesGranted != NULL) {
cJSON_AddNumberToObject(pRoot, "replicaNum", pVotesGranted->replicaNum);
cJSON *pReplicas = cJSON_CreateArray();
cJSON_AddItemToObject(pRoot, "replicas", pReplicas);
for (int i = 0; i < pVotesGranted->replicaNum; ++i) {
cJSON_AddItemToArray(pReplicas, syncUtilRaftId2Json(&(*(pVotesGranted->replicas))[i]));
}
2022-03-25 16:29:53 +00:00
int *arr = (int *)taosMemoryMalloc(sizeof(int) * pVotesGranted->replicaNum);
2022-03-16 02:54:06 +00:00
for (int i = 0; i < pVotesGranted->replicaNum; ++i) {
arr[i] = pVotesGranted->isGranted[i];
}
cJSON *pIsGranted = cJSON_CreateIntArray(arr, pVotesGranted->replicaNum);
2022-03-25 16:29:53 +00:00
taosMemoryFree(arr);
2022-03-16 02:54:06 +00:00
cJSON_AddItemToObject(pRoot, "isGranted", pIsGranted);
cJSON_AddNumberToObject(pRoot, "votes", pVotesGranted->votes);
2022-07-08 11:41:23 +00:00
snprintf(u64buf, sizeof(u64buf), "%" PRIu64, pVotesGranted->term);
2022-03-16 02:54:06 +00:00
cJSON_AddStringToObject(pRoot, "term", u64buf);
cJSON_AddNumberToObject(pRoot, "quorum", pVotesGranted->quorum);
cJSON_AddNumberToObject(pRoot, "toLeader", pVotesGranted->toLeader);
snprintf(u64buf, sizeof(u64buf), "%p", pVotesGranted->pSyncNode);
cJSON_AddStringToObject(pRoot, "pSyncNode", u64buf);
bool majority = voteGrantedMajority(pVotesGranted);
cJSON_AddNumberToObject(pRoot, "majority", majority);
2022-03-08 12:22:31 +00:00
}
2022-03-08 09:07:29 +00:00
cJSON *pJson = cJSON_CreateObject();
cJSON_AddItemToObject(pJson, "SVotesGranted", pRoot);
return pJson;
}
char *voteGranted2Str(SVotesGranted *pVotesGranted) {
cJSON *pJson = voteGranted2Json(pVotesGranted);
2022-06-25 17:16:11 +00:00
char * serialized = cJSON_Print(pJson);
2022-03-08 09:07:29 +00:00
cJSON_Delete(pJson);
return serialized;
}
2022-03-10 11:21:02 +00:00
// for debug -------------------
void voteGrantedPrint(SVotesGranted *pObj) {
char *serialized = voteGranted2Str(pObj);
2022-07-08 10:00:03 +00:00
printf("voteGrantedPrint | len:%" PRIu64 " | %s \n", strlen(serialized), serialized);
2022-03-10 11:21:02 +00:00
fflush(NULL);
2022-03-25 16:29:53 +00:00
taosMemoryFree(serialized);
2022-03-10 11:21:02 +00:00
}
void voteGrantedPrint2(char *s, SVotesGranted *pObj) {
char *serialized = voteGranted2Str(pObj);
2022-07-08 10:00:03 +00:00
printf("voteGrantedPrint2 | len:%" PRIu64 " | %s | %s \n", strlen(serialized), s, serialized);
2022-03-10 11:21:02 +00:00
fflush(NULL);
2022-03-25 16:29:53 +00:00
taosMemoryFree(serialized);
2022-03-10 11:21:02 +00:00
}
void voteGrantedLog(SVotesGranted *pObj) {
char *serialized = voteGranted2Str(pObj);
2022-07-08 10:00:03 +00:00
sTrace("voteGrantedLog | len:%" PRIu64 " | %s", strlen(serialized), serialized);
2022-03-25 16:29:53 +00:00
taosMemoryFree(serialized);
2022-03-10 11:21:02 +00:00
}
void voteGrantedLog2(char *s, SVotesGranted *pObj) {
char *serialized = voteGranted2Str(pObj);
2022-07-08 10:00:03 +00:00
sTrace("voteGrantedLog2 | len:%" PRIu64 " | %s | %s", strlen(serialized), s, serialized);
2022-03-25 16:29:53 +00:00
taosMemoryFree(serialized);
2022-03-10 11:21:02 +00:00
}
2022-03-08 09:07:29 +00:00
// SVotesRespond -----------------------------
2022-03-05 07:03:49 +00:00
SVotesRespond *votesRespondCreate(SSyncNode *pSyncNode) {
2022-03-25 16:29:53 +00:00
SVotesRespond *pVotesRespond = taosMemoryMalloc(sizeof(SVotesRespond));
2022-06-21 08:02:36 +00:00
ASSERT(pVotesRespond != NULL);
2022-03-05 07:03:49 +00:00
memset(pVotesRespond, 0, sizeof(SVotesRespond));
pVotesRespond->replicas = &(pSyncNode->replicasId);
pVotesRespond->replicaNum = pSyncNode->replicaNum;
pVotesRespond->term = 0;
pVotesRespond->pSyncNode = pSyncNode;
return pVotesRespond;
}
2022-03-08 09:07:29 +00:00
void votesRespondDestory(SVotesRespond *pVotesRespond) {
if (pVotesRespond != NULL) {
2022-03-25 16:29:53 +00:00
taosMemoryFree(pVotesRespond);
2022-03-08 09:07:29 +00:00
}
}
2022-05-24 12:23:11 +00:00
void votesRespondUpdate(SVotesRespond *pVotesRespond, SSyncNode *pSyncNode) {
pVotesRespond->replicas = &(pSyncNode->replicasId);
pVotesRespond->replicaNum = pSyncNode->replicaNum;
pVotesRespond->term = 0;
pVotesRespond->pSyncNode = pSyncNode;
}
2022-03-05 07:03:49 +00:00
bool votesResponded(SVotesRespond *pVotesRespond, const SRaftId *pRaftId) {
bool ret = false;
for (int i = 0; i < pVotesRespond->replicaNum; ++i) {
if (syncUtilSameId(&(*pVotesRespond->replicas)[i], pRaftId) && pVotesRespond->isRespond[i]) {
ret = true;
break;
}
}
return ret;
}
void votesRespondAdd(SVotesRespond *pVotesRespond, const SyncRequestVoteReply *pMsg) {
2022-06-21 08:02:36 +00:00
ASSERT(pVotesRespond->term == pMsg->term);
2022-03-05 07:03:49 +00:00
for (int i = 0; i < pVotesRespond->replicaNum; ++i) {
2022-03-09 02:15:40 +00:00
if (syncUtilSameId(&((*(pVotesRespond->replicas))[i]), &pMsg->srcId)) {
2022-06-21 08:02:36 +00:00
// ASSERT(pVotesRespond->isRespond[i] == false);
2022-03-05 07:03:49 +00:00
pVotesRespond->isRespond[i] = true;
return;
}
}
2022-06-21 08:02:36 +00:00
ASSERT(0);
2022-03-05 07:03:49 +00:00
}
2022-03-09 02:15:40 +00:00
void votesRespondReset(SVotesRespond *pVotesRespond, SyncTerm term) {
2022-03-05 07:03:49 +00:00
pVotesRespond->term = term;
2022-03-09 02:15:40 +00:00
memset(pVotesRespond->isRespond, 0, sizeof(pVotesRespond->isRespond));
/*
for (int i = 0; i < pVotesRespond->replicaNum; ++i) {
pVotesRespond->isRespond[i] = false;
}
*/
}
cJSON *votesRespond2Json(SVotesRespond *pVotesRespond) {
char u64buf[128] = {0};
2022-03-09 02:15:40 +00:00
cJSON *pRoot = cJSON_CreateObject();
2022-03-16 02:54:06 +00:00
if (pVotesRespond != NULL) {
cJSON_AddNumberToObject(pRoot, "replicaNum", pVotesRespond->replicaNum);
cJSON *pReplicas = cJSON_CreateArray();
cJSON_AddItemToObject(pRoot, "replicas", pReplicas);
for (int i = 0; i < pVotesRespond->replicaNum; ++i) {
cJSON_AddItemToArray(pReplicas, syncUtilRaftId2Json(&(*(pVotesRespond->replicas))[i]));
}
int respondNum = 0;
2022-03-25 16:29:53 +00:00
int *arr = (int *)taosMemoryMalloc(sizeof(int) * pVotesRespond->replicaNum);
2022-03-16 02:54:06 +00:00
for (int i = 0; i < pVotesRespond->replicaNum; ++i) {
arr[i] = pVotesRespond->isRespond[i];
if (pVotesRespond->isRespond[i]) {
respondNum++;
}
2022-03-09 02:15:40 +00:00
}
2022-03-16 02:54:06 +00:00
cJSON *pIsRespond = cJSON_CreateIntArray(arr, pVotesRespond->replicaNum);
2022-03-25 16:29:53 +00:00
taosMemoryFree(arr);
2022-03-16 02:54:06 +00:00
cJSON_AddItemToObject(pRoot, "isRespond", pIsRespond);
cJSON_AddNumberToObject(pRoot, "respondNum", respondNum);
2022-07-08 11:41:23 +00:00
snprintf(u64buf, sizeof(u64buf), "%" PRIu64, pVotesRespond->term);
2022-03-16 02:54:06 +00:00
cJSON_AddStringToObject(pRoot, "term", u64buf);
snprintf(u64buf, sizeof(u64buf), "%p", pVotesRespond->pSyncNode);
cJSON_AddStringToObject(pRoot, "pSyncNode", u64buf);
2022-03-05 07:03:49 +00:00
}
2022-03-09 02:15:40 +00:00
cJSON *pJson = cJSON_CreateObject();
cJSON_AddItemToObject(pJson, "SVotesRespond", pRoot);
return pJson;
}
char *votesRespond2Str(SVotesRespond *pVotesRespond) {
cJSON *pJson = votesRespond2Json(pVotesRespond);
2022-06-25 17:16:11 +00:00
char * serialized = cJSON_Print(pJson);
2022-03-09 02:15:40 +00:00
cJSON_Delete(pJson);
return serialized;
2022-03-10 11:21:02 +00:00
}
// for debug -------------------
void votesRespondPrint(SVotesRespond *pObj) {
char *serialized = votesRespond2Str(pObj);
2022-07-08 10:00:03 +00:00
printf("votesRespondPrint | len:%" PRIu64 " | %s \n", strlen(serialized), serialized);
2022-03-10 11:21:02 +00:00
fflush(NULL);
2022-03-25 16:29:53 +00:00
taosMemoryFree(serialized);
2022-03-10 11:21:02 +00:00
}
void votesRespondPrint2(char *s, SVotesRespond *pObj) {
char *serialized = votesRespond2Str(pObj);
2022-07-08 10:00:03 +00:00
printf("votesRespondPrint2 | len:%" PRIu64 " | %s | %s \n", strlen(serialized), s, serialized);
2022-03-10 11:21:02 +00:00
fflush(NULL);
2022-03-25 16:29:53 +00:00
taosMemoryFree(serialized);
2022-03-10 11:21:02 +00:00
}
void votesRespondLog(SVotesRespond *pObj) {
char *serialized = votesRespond2Str(pObj);
2022-07-08 10:00:03 +00:00
sTrace("votesRespondLog | len:%" PRIu64 " | %s", strlen(serialized), serialized);
2022-03-25 16:29:53 +00:00
taosMemoryFree(serialized);
2022-03-10 11:21:02 +00:00
}
void votesRespondLog2(char *s, SVotesRespond *pObj) {
char *serialized = votesRespond2Str(pObj);
2022-07-08 10:00:03 +00:00
sTrace("votesRespondLog2 | len:%" PRIu64 " | %s | %s", strlen(serialized), s, serialized);
2022-03-25 16:29:53 +00:00
taosMemoryFree(serialized);
}