c0cb274659
* Fix incorrect usage of begin() when genesis block is requested in "protx diff" .begin() on mapBlockIndex does NOT return the genesis block, but just the block with lowest hash. The fix is to use chainActive[0] to get the genesis block. * Update src/evo/simplifiedmns.cpp Co-Authored-By: codablock <ablock84@gmail.com>
172 lines
5.8 KiB
C++
172 lines
5.8 KiB
C++
// Copyright (c) 2017-2019 The Dash Core developers
|
|
// Distributed under the MIT software license, see the accompanying
|
|
// file COPYING or http://www.opensource.org/licenses/mit-license.php.
|
|
|
|
#include "cbtx.h"
|
|
#include "core_io.h"
|
|
#include "deterministicmns.h"
|
|
#include "simplifiedmns.h"
|
|
#include "specialtx.h"
|
|
|
|
#include "base58.h"
|
|
#include "chainparams.h"
|
|
#include "consensus/merkle.h"
|
|
#include "univalue.h"
|
|
#include "validation.h"
|
|
|
|
CSimplifiedMNListEntry::CSimplifiedMNListEntry(const CDeterministicMN& dmn) :
|
|
proRegTxHash(dmn.proTxHash),
|
|
confirmedHash(dmn.pdmnState->confirmedHash),
|
|
service(dmn.pdmnState->addr),
|
|
pubKeyOperator(dmn.pdmnState->pubKeyOperator),
|
|
keyIDVoting(dmn.pdmnState->keyIDVoting),
|
|
isValid(dmn.pdmnState->nPoSeBanHeight == -1)
|
|
{
|
|
}
|
|
|
|
uint256 CSimplifiedMNListEntry::CalcHash() const
|
|
{
|
|
CHashWriter hw(SER_GETHASH, CLIENT_VERSION);
|
|
hw << *this;
|
|
return hw.GetHash();
|
|
}
|
|
|
|
std::string CSimplifiedMNListEntry::ToString() const
|
|
{
|
|
return strprintf("CSimplifiedMNListEntry(proRegTxHash=%s, confirmedHash=%s, service=%s, pubKeyOperator=%s, votingAddress=%s, isValie=%d)",
|
|
proRegTxHash.ToString(), confirmedHash.ToString(), service.ToString(false), pubKeyOperator.ToString(), CBitcoinAddress(keyIDVoting).ToString(), isValid);
|
|
}
|
|
|
|
void CSimplifiedMNListEntry::ToJson(UniValue& obj) const
|
|
{
|
|
obj.clear();
|
|
obj.setObject();
|
|
obj.push_back(Pair("proRegTxHash", proRegTxHash.ToString()));
|
|
obj.push_back(Pair("confirmedHash", confirmedHash.ToString()));
|
|
obj.push_back(Pair("service", service.ToString(false)));
|
|
obj.push_back(Pair("pubKeyOperator", pubKeyOperator.ToString()));
|
|
obj.push_back(Pair("votingAddress", CBitcoinAddress(keyIDVoting).ToString()));
|
|
obj.push_back(Pair("isValid", isValid));
|
|
}
|
|
|
|
CSimplifiedMNList::CSimplifiedMNList(const std::vector<CSimplifiedMNListEntry>& smlEntries)
|
|
{
|
|
mnList = smlEntries;
|
|
|
|
std::sort(mnList.begin(), mnList.end(), [&](const CSimplifiedMNListEntry& a, const CSimplifiedMNListEntry& b) {
|
|
return a.proRegTxHash.Compare(b.proRegTxHash) < 0;
|
|
});
|
|
}
|
|
|
|
CSimplifiedMNList::CSimplifiedMNList(const CDeterministicMNList& dmnList)
|
|
{
|
|
mnList.reserve(dmnList.GetAllMNsCount());
|
|
|
|
dmnList.ForEachMN(false, [this](const CDeterministicMNCPtr& dmn) {
|
|
mnList.emplace_back(*dmn);
|
|
});
|
|
|
|
std::sort(mnList.begin(), mnList.end(), [&](const CSimplifiedMNListEntry& a, const CSimplifiedMNListEntry& b) {
|
|
return a.proRegTxHash.Compare(b.proRegTxHash) < 0;
|
|
});
|
|
}
|
|
|
|
uint256 CSimplifiedMNList::CalcMerkleRoot(bool* pmutated) const
|
|
{
|
|
std::vector<uint256> leaves;
|
|
leaves.reserve(mnList.size());
|
|
for (const auto& e : mnList) {
|
|
leaves.emplace_back(e.CalcHash());
|
|
}
|
|
return ComputeMerkleRoot(leaves, pmutated);
|
|
}
|
|
|
|
void CSimplifiedMNListDiff::ToJson(UniValue& obj) const
|
|
{
|
|
obj.setObject();
|
|
|
|
obj.push_back(Pair("baseBlockHash", baseBlockHash.ToString()));
|
|
obj.push_back(Pair("blockHash", blockHash.ToString()));
|
|
|
|
CDataStream ssCbTxMerkleTree(SER_NETWORK, PROTOCOL_VERSION);
|
|
ssCbTxMerkleTree << cbTxMerkleTree;
|
|
obj.push_back(Pair("cbTxMerkleTree", HexStr(ssCbTxMerkleTree.begin(), ssCbTxMerkleTree.end())));
|
|
|
|
obj.push_back(Pair("cbTx", EncodeHexTx(*cbTx)));
|
|
|
|
UniValue deletedMNsArr(UniValue::VARR);
|
|
for (const auto& h : deletedMNs) {
|
|
deletedMNsArr.push_back(h.ToString());
|
|
}
|
|
obj.push_back(Pair("deletedMNs", deletedMNsArr));
|
|
|
|
UniValue mnListArr(UniValue::VARR);
|
|
for (const auto& e : mnList) {
|
|
UniValue eObj;
|
|
e.ToJson(eObj);
|
|
mnListArr.push_back(eObj);
|
|
}
|
|
obj.push_back(Pair("mnList", mnListArr));
|
|
|
|
CCbTx cbTxPayload;
|
|
if (GetTxPayload(*cbTx, cbTxPayload)) {
|
|
obj.push_back(Pair("merkleRootMNList", cbTxPayload.merkleRootMNList.ToString()));
|
|
}
|
|
}
|
|
|
|
bool BuildSimplifiedMNListDiff(const uint256& baseBlockHash, const uint256& blockHash, CSimplifiedMNListDiff& mnListDiffRet, std::string& errorRet)
|
|
{
|
|
AssertLockHeld(cs_main);
|
|
mnListDiffRet = CSimplifiedMNListDiff();
|
|
|
|
const CBlockIndex* baseBlockIndex = chainActive.Genesis();
|
|
if (!baseBlockHash.IsNull()) {
|
|
auto it = mapBlockIndex.find(baseBlockHash);
|
|
if (it == mapBlockIndex.end()) {
|
|
errorRet = strprintf("block %s not found", baseBlockHash.ToString());
|
|
return false;
|
|
}
|
|
baseBlockIndex = it->second;
|
|
}
|
|
auto blockIt = mapBlockIndex.find(blockHash);
|
|
if (blockIt == mapBlockIndex.end()) {
|
|
errorRet = strprintf("block %s not found", blockHash.ToString());
|
|
return false;
|
|
}
|
|
const CBlockIndex* blockIndex = blockIt->second;
|
|
|
|
if (!chainActive.Contains(baseBlockIndex) || !chainActive.Contains(blockIndex)) {
|
|
errorRet = strprintf("block %s and %s are not in the same chain", baseBlockHash.ToString(), blockHash.ToString());
|
|
return false;
|
|
}
|
|
if (baseBlockIndex->nHeight > blockIndex->nHeight) {
|
|
errorRet = strprintf("base block %s is higher then block %s", baseBlockHash.ToString(), blockHash.ToString());
|
|
return false;
|
|
}
|
|
|
|
LOCK(deterministicMNManager->cs);
|
|
|
|
auto baseDmnList = deterministicMNManager->GetListForBlock(baseBlockHash);
|
|
auto dmnList = deterministicMNManager->GetListForBlock(blockHash);
|
|
mnListDiffRet = baseDmnList.BuildSimplifiedDiff(dmnList);
|
|
|
|
// TODO store coinbase TX in CBlockIndex
|
|
CBlock block;
|
|
if (!ReadBlockFromDisk(block, blockIndex, Params().GetConsensus())) {
|
|
errorRet = strprintf("failed to read block %s from disk", blockHash.ToString());
|
|
return false;
|
|
}
|
|
|
|
mnListDiffRet.cbTx = block.vtx[0];
|
|
|
|
std::vector<uint256> vHashes;
|
|
std::vector<bool> vMatch(block.vtx.size(), false);
|
|
for (const auto& tx : block.vtx) {
|
|
vHashes.emplace_back(tx->GetHash());
|
|
}
|
|
vMatch[0] = true; // only coinbase matches
|
|
mnListDiffRet.cbTxMerkleTree = CPartialMerkleTree(vHashes, vMatch);
|
|
|
|
return true;
|
|
}
|