Few mn payments fixes: (#1183)
- avoid processing same vote multiple times - do not relay votes until synced - do not ban for wrong signature of old votes - do not check masternode ranks for old votes on regular (non-MN) nodes
This commit is contained in:
parent
b847642428
commit
d9b1e7734a
@ -5112,7 +5112,7 @@ void static ProcessGetData(CNode* pfrom, const Consensus::Params& consensusParam
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (!pushed && inv.type == MSG_MASTERNODE_PAYMENT_VOTE) {
|
if (!pushed && inv.type == MSG_MASTERNODE_PAYMENT_VOTE) {
|
||||||
if(mnpayments.mapMasternodePaymentVotes.count(inv.hash)) {
|
if(mnpayments.HasVerifiedPaymentVote(inv.hash)) {
|
||||||
CDataStream ss(SER_NETWORK, PROTOCOL_VERSION);
|
CDataStream ss(SER_NETWORK, PROTOCOL_VERSION);
|
||||||
ss.reserve(1000);
|
ss.reserve(1000);
|
||||||
ss << mnpayments.mapMasternodePaymentVotes[inv.hash];
|
ss << mnpayments.mapMasternodePaymentVotes[inv.hash];
|
||||||
@ -5128,7 +5128,7 @@ void static ProcessGetData(CNode* pfrom, const Consensus::Params& consensusParam
|
|||||||
BOOST_FOREACH(CMasternodePayee& payee, mnpayments.mapMasternodeBlocks[mi->second->nHeight].vecPayees) {
|
BOOST_FOREACH(CMasternodePayee& payee, mnpayments.mapMasternodeBlocks[mi->second->nHeight].vecPayees) {
|
||||||
std::vector<uint256> vecVoteHashes = payee.GetVoteHashes();
|
std::vector<uint256> vecVoteHashes = payee.GetVoteHashes();
|
||||||
BOOST_FOREACH(uint256& hash, vecVoteHashes) {
|
BOOST_FOREACH(uint256& hash, vecVoteHashes) {
|
||||||
if(mnpayments.mapMasternodePaymentVotes.count(hash)) {
|
if(mnpayments.HasVerifiedPaymentVote(hash)) {
|
||||||
CDataStream ss(SER_NETWORK, PROTOCOL_VERSION);
|
CDataStream ss(SER_NETWORK, PROTOCOL_VERSION);
|
||||||
ss.reserve(1000);
|
ss.reserve(1000);
|
||||||
ss << mnpayments.mapMasternodePaymentVotes[hash];
|
ss << mnpayments.mapMasternodePaymentVotes[hash];
|
||||||
|
@ -341,9 +341,18 @@ void CMasternodePayments::ProcessMessage(CNode* pfrom, std::string& strCommand,
|
|||||||
|
|
||||||
if(!pCurrentBlockIndex) return;
|
if(!pCurrentBlockIndex) return;
|
||||||
|
|
||||||
if(mapMasternodePaymentVotes.count(vote.GetHash())) {
|
{
|
||||||
LogPrint("mnpayments", "MASTERNODEPAYMENTVOTE -- hash=%s, nHeight=%d seen\n", vote.GetHash().ToString(), pCurrentBlockIndex->nHeight);
|
LOCK(cs_mapMasternodePaymentVotes);
|
||||||
return;
|
if(mapMasternodePaymentVotes.count(vote.GetHash())) {
|
||||||
|
LogPrint("mnpayments", "MASTERNODEPAYMENTVOTE -- hash=%s, nHeight=%d seen\n", vote.GetHash().ToString(), pCurrentBlockIndex->nHeight);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Avoid processing same vote multiple times
|
||||||
|
mapMasternodePaymentVotes[vote.GetHash()] = vote;
|
||||||
|
// but first mark vote as non-verified,
|
||||||
|
// AddPaymentVote() below should take care of it if vote is actually ok
|
||||||
|
mapMasternodePaymentVotes[vote.GetHash()].MarkAsNotVerified();
|
||||||
}
|
}
|
||||||
|
|
||||||
int nFirstBlock = pCurrentBlockIndex->nHeight - GetStorageLimit();
|
int nFirstBlock = pCurrentBlockIndex->nHeight - GetStorageLimit();
|
||||||
@ -363,17 +372,32 @@ void CMasternodePayments::ProcessMessage(CNode* pfrom, std::string& strCommand,
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if(!vote.CheckSignature()) {
|
masternode_info_t mnInfo = mnodeman.GetMasternodeInfo(vote.vinMasternode);
|
||||||
// do not ban for old mnw, MN simply might be not active anymore
|
if(!mnInfo.fInfoValid) {
|
||||||
if(masternodeSync.IsSynced() && vote.nBlockHeight > pCurrentBlockIndex->nHeight) {
|
// mn was not found, so we can't check vote, some info is probably missing
|
||||||
LogPrintf("MASTERNODEPAYMENTVOTE -- invalid signature\n");
|
LogPrintf("MASTERNODEPAYMENTVOTE -- masternode is missing %s\n", vote.vinMasternode.prevout.ToStringShort());
|
||||||
Misbehaving(pfrom->GetId(), 20);
|
|
||||||
}
|
|
||||||
// it could just be a non-synced masternode
|
|
||||||
mnodeman.AskForMN(pfrom, vote.vinMasternode);
|
mnodeman.AskForMN(pfrom, vote.vinMasternode);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int nDos = 0;
|
||||||
|
if(!vote.CheckSignature(mnInfo.pubKeyMasternode, pCurrentBlockIndex->nHeight, nDos)) {
|
||||||
|
if(nDos) {
|
||||||
|
LogPrintf("MASTERNODEPAYMENTVOTE -- ERROR: invalid signature\n");
|
||||||
|
Misbehaving(pfrom->GetId(), nDos);
|
||||||
|
} else {
|
||||||
|
// only warn about anything non-critical (i.e. nDos == 0) in debug mode
|
||||||
|
LogPrint("mnpayments", "MASTERNODEPAYMENTVOTE -- WARNING: invalid signature\n");
|
||||||
|
}
|
||||||
|
// Either our info or vote info could be outdated.
|
||||||
|
// In case our info is outdated, ask for an update,
|
||||||
|
mnodeman.AskForMN(pfrom, vote.vinMasternode);
|
||||||
|
// but there is nothing we can do if vote info itself is outdated
|
||||||
|
// (i.e. it was signed by a mn which changed its key),
|
||||||
|
// so just quit here.
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
CTxDestination address1;
|
CTxDestination address1;
|
||||||
ExtractDestination(vote.payee, address1);
|
ExtractDestination(vote.payee, address1);
|
||||||
CBitcoinAddress address2(address1);
|
CBitcoinAddress address2(address1);
|
||||||
@ -443,9 +467,9 @@ bool CMasternodePayments::AddPaymentVote(const CMasternodePaymentVote& vote)
|
|||||||
uint256 blockHash = uint256();
|
uint256 blockHash = uint256();
|
||||||
if(!GetBlockHash(blockHash, vote.nBlockHeight - 101)) return false;
|
if(!GetBlockHash(blockHash, vote.nBlockHeight - 101)) return false;
|
||||||
|
|
||||||
LOCK2(cs_mapMasternodeBlocks, cs_mapMasternodePaymentVotes);
|
if(HasVerifiedPaymentVote(vote.GetHash())) return false;
|
||||||
|
|
||||||
if(mapMasternodePaymentVotes.count(vote.GetHash())) return false;
|
LOCK2(cs_mapMasternodeBlocks, cs_mapMasternodePaymentVotes);
|
||||||
|
|
||||||
mapMasternodePaymentVotes[vote.GetHash()] = vote;
|
mapMasternodePaymentVotes[vote.GetHash()] = vote;
|
||||||
|
|
||||||
@ -459,6 +483,13 @@ bool CMasternodePayments::AddPaymentVote(const CMasternodePaymentVote& vote)
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool CMasternodePayments::HasVerifiedPaymentVote(uint256 hashIn)
|
||||||
|
{
|
||||||
|
LOCK(cs_mapMasternodePaymentVotes);
|
||||||
|
std::map<uint256, CMasternodePaymentVote>::iterator it = mapMasternodePaymentVotes.find(hashIn);
|
||||||
|
return it != mapMasternodePaymentVotes.end() && it->second.IsVerified();
|
||||||
|
}
|
||||||
|
|
||||||
void CMasternodeBlockPayees::AddPayee(const CMasternodePaymentVote& vote)
|
void CMasternodeBlockPayees::AddPayee(const CMasternodePaymentVote& vote)
|
||||||
{
|
{
|
||||||
LOCK(cs_vecPayees);
|
LOCK(cs_vecPayees);
|
||||||
@ -634,7 +665,7 @@ bool CMasternodePaymentVote::IsValid(CNode* pnode, int nValidationHeight, std::s
|
|||||||
}
|
}
|
||||||
|
|
||||||
int nMinRequiredProtocol;
|
int nMinRequiredProtocol;
|
||||||
if(nBlockHeight > nValidationHeight) {
|
if(nBlockHeight >= nValidationHeight) {
|
||||||
// new votes must comply SPORK_10_MASTERNODE_PAY_UPDATED_NODES rules
|
// new votes must comply SPORK_10_MASTERNODE_PAY_UPDATED_NODES rules
|
||||||
nMinRequiredProtocol = mnpayments.GetMinMasternodePaymentsProto();
|
nMinRequiredProtocol = mnpayments.GetMinMasternodePaymentsProto();
|
||||||
} else {
|
} else {
|
||||||
@ -647,6 +678,10 @@ bool CMasternodePaymentVote::IsValid(CNode* pnode, int nValidationHeight, std::s
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Only masternodes should try to check masternode rank for old votes - they need to pick the right winner for future blocks.
|
||||||
|
// Regular clients (miners included) need to verify masternode rank for future block votes only.
|
||||||
|
if(!fMasterNode && nBlockHeight < nValidationHeight) return true;
|
||||||
|
|
||||||
int nRank = mnodeman.GetMasternodeRank(vinMasternode, nBlockHeight - 101, nMinRequiredProtocol, false);
|
int nRank = mnodeman.GetMasternodeRank(vinMasternode, nBlockHeight - 101, nMinRequiredProtocol, false);
|
||||||
|
|
||||||
if(nRank > MNPAYMENTS_SIGNATURES_TOTAL) {
|
if(nRank > MNPAYMENTS_SIGNATURES_TOTAL) {
|
||||||
@ -733,23 +768,29 @@ bool CMasternodePayments::ProcessBlock(int nBlockHeight)
|
|||||||
|
|
||||||
void CMasternodePaymentVote::Relay()
|
void CMasternodePaymentVote::Relay()
|
||||||
{
|
{
|
||||||
|
// do not relay until synced
|
||||||
|
if (!masternodeSync.IsSynced()) return;
|
||||||
CInv inv(MSG_MASTERNODE_PAYMENT_VOTE, GetHash());
|
CInv inv(MSG_MASTERNODE_PAYMENT_VOTE, GetHash());
|
||||||
RelayInv(inv);
|
RelayInv(inv);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool CMasternodePaymentVote::CheckSignature()
|
bool CMasternodePaymentVote::CheckSignature(const CPubKey& pubKeyMasternode, int nValidationHeight, int &nDos)
|
||||||
{
|
{
|
||||||
|
// do not ban by default
|
||||||
CMasternode* pmn = mnodeman.Find(vinMasternode);
|
nDos = 0;
|
||||||
|
|
||||||
if (!pmn) return false;
|
|
||||||
|
|
||||||
std::string strMessage = vinMasternode.prevout.ToStringShort() +
|
std::string strMessage = vinMasternode.prevout.ToStringShort() +
|
||||||
boost::lexical_cast<std::string>(nBlockHeight) +
|
boost::lexical_cast<std::string>(nBlockHeight) +
|
||||||
ScriptToAsmStr(payee);
|
ScriptToAsmStr(payee);
|
||||||
|
|
||||||
std::string strError = "";
|
std::string strError = "";
|
||||||
if (!darkSendSigner.VerifyMessage(pmn->pubKeyMasternode, vchSig, strMessage, strError)) {
|
if (!darkSendSigner.VerifyMessage(pubKeyMasternode, vchSig, strMessage, strError)) {
|
||||||
|
// Only ban for future block vote when we are already synced.
|
||||||
|
// Otherwise it could be the case when MN which signed this vote is using another key now
|
||||||
|
// and we have no idea about the old one.
|
||||||
|
if(masternodeSync.IsSynced() && nBlockHeight > nValidationHeight) {
|
||||||
|
nDos = 20;
|
||||||
|
}
|
||||||
return error("CMasternodePaymentVote::CheckSignature -- Got bad Masternode payment signature, masternode=%s, error: %s", vinMasternode.prevout.ToStringShort().c_str(), strError);
|
return error("CMasternodePaymentVote::CheckSignature -- Got bad Masternode payment signature, masternode=%s, error: %s", vinMasternode.prevout.ToStringShort().c_str(), strError);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -791,6 +832,7 @@ void CMasternodePayments::Sync(CNode* pnode, int nCountNeeded)
|
|||||||
BOOST_FOREACH(CMasternodePayee& payee, mapMasternodeBlocks[h].vecPayees) {
|
BOOST_FOREACH(CMasternodePayee& payee, mapMasternodeBlocks[h].vecPayees) {
|
||||||
std::vector<uint256> vecVoteHashes = payee.GetVoteHashes();
|
std::vector<uint256> vecVoteHashes = payee.GetVoteHashes();
|
||||||
BOOST_FOREACH(uint256& hash, vecVoteHashes) {
|
BOOST_FOREACH(uint256& hash, vecVoteHashes) {
|
||||||
|
if(!HasVerifiedPaymentVote(hash)) continue;
|
||||||
pnode->PushInventory(CInv(MSG_MASTERNODE_PAYMENT_VOTE, hash));
|
pnode->PushInventory(CInv(MSG_MASTERNODE_PAYMENT_VOTE, hash));
|
||||||
nInvCount++;
|
nInvCount++;
|
||||||
}
|
}
|
||||||
|
@ -79,8 +79,14 @@ public:
|
|||||||
int nBlockHeight;
|
int nBlockHeight;
|
||||||
std::vector<CMasternodePayee> vecPayees;
|
std::vector<CMasternodePayee> vecPayees;
|
||||||
|
|
||||||
CMasternodeBlockPayees() : nBlockHeight(0) {}
|
CMasternodeBlockPayees() :
|
||||||
CMasternodeBlockPayees(int nBlockHeightIn) : nBlockHeight(nBlockHeightIn) {}
|
nBlockHeight(0),
|
||||||
|
vecPayees()
|
||||||
|
{}
|
||||||
|
CMasternodeBlockPayees(int nBlockHeightIn) :
|
||||||
|
nBlockHeight(nBlockHeightIn),
|
||||||
|
vecPayees()
|
||||||
|
{}
|
||||||
|
|
||||||
ADD_SERIALIZE_METHODS;
|
ADD_SERIALIZE_METHODS;
|
||||||
|
|
||||||
@ -142,11 +148,14 @@ public:
|
|||||||
}
|
}
|
||||||
|
|
||||||
bool Sign();
|
bool Sign();
|
||||||
bool CheckSignature();
|
bool CheckSignature(const CPubKey& pubKeyMasternode, int nValidationHeight, int &nDos);
|
||||||
|
|
||||||
bool IsValid(CNode* pnode, int nValidationHeight, std::string& strError);
|
bool IsValid(CNode* pnode, int nValidationHeight, std::string& strError);
|
||||||
void Relay();
|
void Relay();
|
||||||
|
|
||||||
|
bool IsVerified() { return !vchSig.empty(); }
|
||||||
|
void MarkAsNotVerified() { vchSig.clear(); }
|
||||||
|
|
||||||
std::string ToString() const;
|
std::string ToString() const;
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -184,6 +193,7 @@ public:
|
|||||||
void Clear();
|
void Clear();
|
||||||
|
|
||||||
bool AddPaymentVote(const CMasternodePaymentVote& vote);
|
bool AddPaymentVote(const CMasternodePaymentVote& vote);
|
||||||
|
bool HasVerifiedPaymentVote(uint256 hashIn);
|
||||||
bool ProcessBlock(int nBlockHeight);
|
bool ProcessBlock(int nBlockHeight);
|
||||||
|
|
||||||
void Sync(CNode* node, int nCountNeeded);
|
void Sync(CNode* node, int nCountNeeded);
|
||||||
|
Loading…
Reference in New Issue
Block a user