71acdb839c
originally from BvbFan
2532 lines
98 KiB
C++
2532 lines
98 KiB
C++
#include <claimtrie.h>
|
|
#include <coins.h>
|
|
#include <hash.h>
|
|
#include <logging.h>
|
|
#include <util.h>
|
|
|
|
#include <iostream>
|
|
#include <algorithm>
|
|
|
|
#include <boost/scoped_ptr.hpp>
|
|
|
|
static const std::string rootClaimName = "";
|
|
static const std::string rootClaimHash = "0000000000000000000000000000000000000000000000000000000000000001";
|
|
|
|
std::vector<unsigned char> heightToVch(int n)
|
|
{
|
|
std::vector<unsigned char> vchHeight;
|
|
vchHeight.resize(8);
|
|
vchHeight[0] = 0;
|
|
vchHeight[1] = 0;
|
|
vchHeight[2] = 0;
|
|
vchHeight[3] = 0;
|
|
vchHeight[4] = n >> 24;
|
|
vchHeight[5] = n >> 16;
|
|
vchHeight[6] = n >> 8;
|
|
vchHeight[7] = n;
|
|
return vchHeight;
|
|
}
|
|
|
|
uint256 getValueHash(COutPoint outPoint, int nHeightOfLastTakeover)
|
|
{
|
|
CHash256 hasher;
|
|
auto hash = Hash(outPoint.hash.begin(), outPoint.hash.end());
|
|
hasher.Write(hash.begin(), hash.size());
|
|
|
|
auto snOut = std::to_string(outPoint.n);
|
|
hash = Hash(snOut.begin(), snOut.end());
|
|
hasher.Write(hash.begin(), hash.size());
|
|
|
|
auto vchHash = heightToVch(nHeightOfLastTakeover);
|
|
hash = Hash(vchHash.begin(), vchHash.end());
|
|
hasher.Write(hash.begin(), hash.size());
|
|
|
|
uint256 result;
|
|
hasher.Finalize(result.begin());
|
|
return result;
|
|
}
|
|
|
|
bool CClaimTrieNode::insertClaim(CClaimValue claim)
|
|
{
|
|
claims.push_back(claim);
|
|
return true;
|
|
}
|
|
|
|
bool CClaimTrieNode::removeClaim(const COutPoint& outPoint, CClaimValue& claim)
|
|
{
|
|
std::vector<CClaimValue>::iterator itClaims;
|
|
for (itClaims = claims.begin(); itClaims != claims.end(); ++itClaims)
|
|
{
|
|
if (itClaims->outPoint == outPoint)
|
|
{
|
|
std::swap(claim, *itClaims);
|
|
break;
|
|
}
|
|
}
|
|
if (itClaims != claims.end())
|
|
{
|
|
claims.erase(itClaims);
|
|
}
|
|
else
|
|
{
|
|
LogPrintf("CClaimTrieNode::%s() : asked to remove a claim that doesn't exist\n", __func__);
|
|
LogPrintf("CClaimTrieNode::%s() : claims that do exist:\n", __func__);
|
|
for (unsigned int i = 0; i < claims.size(); i++)
|
|
{
|
|
LogPrintf("\ttxhash: %s, nOut: %d:\n", claims[i].outPoint.hash.ToString(), claims[i].outPoint.n);
|
|
}
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
bool CClaimTrieNode::getBestClaim(CClaimValue& claim) const
|
|
{
|
|
if (claims.empty())
|
|
return false;
|
|
|
|
claim = claims.front();
|
|
return true;
|
|
}
|
|
|
|
bool CClaimTrieNode::haveClaim(const COutPoint& outPoint) const
|
|
{
|
|
for (std::vector<CClaimValue>::const_iterator itclaim = claims.begin(); itclaim != claims.end(); ++itclaim)
|
|
{
|
|
if (itclaim->outPoint == outPoint)
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
void CClaimTrieNode::reorderClaims(supportMapEntryType& supports)
|
|
{
|
|
std::vector<CClaimValue>::iterator itclaim;
|
|
|
|
for (itclaim = claims.begin(); itclaim != claims.end(); ++itclaim)
|
|
itclaim->nEffectiveAmount = itclaim->nAmount;
|
|
|
|
for (supportMapEntryType::iterator itsupport = supports.begin(); itsupport != supports.end(); ++itsupport)
|
|
{
|
|
for (itclaim = claims.begin(); itclaim != claims.end(); ++itclaim)
|
|
{
|
|
if (itsupport->supportedClaimId == itclaim->claimId)
|
|
{
|
|
itclaim->nEffectiveAmount += itsupport->nAmount;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
std::make_heap(claims.begin(), claims.end());
|
|
}
|
|
|
|
uint256 CClaimTrie::getMerkleHash()
|
|
{
|
|
return root.hash;
|
|
}
|
|
|
|
bool CClaimTrie::empty() const
|
|
{
|
|
return root.empty();
|
|
}
|
|
|
|
template<typename K> bool CClaimTrie::keyTypeEmpty(char keyType, K& /* throwaway */) const
|
|
{
|
|
boost::scoped_ptr<CDBIterator> pcursor(const_cast<CDBWrapper*>(&db)->NewIterator());
|
|
pcursor->SeekToFirst();
|
|
|
|
while (pcursor->Valid())
|
|
{
|
|
std::pair<char, K> key;
|
|
if (!pcursor->GetKey(key))
|
|
break;
|
|
|
|
if (key.first == keyType)
|
|
return false;
|
|
|
|
pcursor->Next();
|
|
}
|
|
return true;
|
|
}
|
|
|
|
bool CClaimTrie::queueEmpty() const
|
|
{
|
|
for (claimQueueType::const_iterator itRow = dirtyQueueRows.begin(); itRow != dirtyQueueRows.end(); ++itRow)
|
|
{
|
|
if (!itRow->second.empty())
|
|
return false;
|
|
}
|
|
int throwaway;
|
|
return keyTypeEmpty(CLAIM_QUEUE_ROW, throwaway);
|
|
}
|
|
|
|
bool CClaimTrie::expirationQueueEmpty() const
|
|
{
|
|
for (expirationQueueType::const_iterator itRow = dirtyExpirationQueueRows.begin(); itRow != dirtyExpirationQueueRows.end(); ++itRow)
|
|
{
|
|
if (!itRow->second.empty())
|
|
return false;
|
|
}
|
|
int throwaway;
|
|
return keyTypeEmpty(EXP_QUEUE_ROW, throwaway);
|
|
}
|
|
|
|
bool CClaimTrie::supportEmpty() const
|
|
{
|
|
for (supportMapType::const_iterator itNode = dirtySupportNodes.begin(); itNode != dirtySupportNodes.end(); ++itNode)
|
|
{
|
|
if (!itNode->second.empty())
|
|
return false;
|
|
}
|
|
std::string throwaway;
|
|
return keyTypeEmpty(SUPPORT, throwaway);
|
|
}
|
|
|
|
bool CClaimTrie::supportQueueEmpty() const
|
|
{
|
|
for (supportQueueType::const_iterator itRow = dirtySupportQueueRows.begin(); itRow != dirtySupportQueueRows.end(); ++itRow)
|
|
{
|
|
if (!itRow->second.empty())
|
|
return false;
|
|
}
|
|
int throwaway;
|
|
return keyTypeEmpty(SUPPORT_QUEUE_ROW, throwaway);
|
|
}
|
|
|
|
void CClaimTrie::setExpirationTime(int t)
|
|
{
|
|
nExpirationTime = t;
|
|
LogPrintf("%s: Expiration time is now %d\n", __func__, nExpirationTime);
|
|
}
|
|
|
|
void CClaimTrie::clear()
|
|
{
|
|
clear(&root);
|
|
}
|
|
|
|
void CClaimTrie::clear(CClaimTrieNode* current)
|
|
{
|
|
for (nodeMapType::const_iterator itchildren = current->children.begin(); itchildren != current->children.end(); ++itchildren)
|
|
{
|
|
clear(itchildren->second);
|
|
delete itchildren->second;
|
|
}
|
|
current->children.clear();
|
|
}
|
|
|
|
bool CClaimTrie::haveClaim(const std::string& name, const COutPoint& outPoint) const
|
|
{
|
|
const CClaimTrieNode* current = &root;
|
|
for (std::string::const_iterator itname = name.begin(); itname != name.end(); ++itname)
|
|
{
|
|
nodeMapType::const_iterator itchildren = current->children.find(*itname);
|
|
if (itchildren == current->children.end())
|
|
return false;
|
|
current = itchildren->second;
|
|
}
|
|
return current->haveClaim(outPoint);
|
|
}
|
|
|
|
bool CClaimTrie::haveSupport(const std::string& name, const COutPoint& outPoint) const
|
|
{
|
|
supportMapEntryType node;
|
|
if (!getSupportNode(name, node))
|
|
return false;
|
|
|
|
for (supportMapEntryType::const_iterator itnode = node.begin(); itnode != node.end(); ++itnode)
|
|
{
|
|
if (itnode->outPoint == outPoint)
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
bool CClaimTrie::haveClaimInQueue(const std::string& name, const COutPoint& outPoint, int& nValidAtHeight) const
|
|
{
|
|
queueNameRowType nameRow;
|
|
if (!getQueueNameRow(name, nameRow))
|
|
{
|
|
return false;
|
|
}
|
|
queueNameRowType::const_iterator itNameRow;
|
|
for (itNameRow = nameRow.begin(); itNameRow != nameRow.end(); ++itNameRow)
|
|
{
|
|
if (itNameRow->outPoint == outPoint)
|
|
{
|
|
nValidAtHeight = itNameRow->nHeight;
|
|
break;
|
|
}
|
|
}
|
|
if (itNameRow == nameRow.end())
|
|
{
|
|
return false;
|
|
}
|
|
claimQueueRowType row;
|
|
if (getQueueRow(nValidAtHeight, row))
|
|
{
|
|
for (claimQueueRowType::const_iterator itRow = row.begin(); itRow != row.end(); ++itRow)
|
|
{
|
|
if (itRow->first == name && itRow->second.outPoint == outPoint)
|
|
{
|
|
if (itRow->second.nValidAtHeight != nValidAtHeight)
|
|
{
|
|
LogPrintf("%s: An inconsistency was found in the claim queue. Please report this to the developers:\nDifferent nValidAtHeight between named queue and height queue\n: name: %s, txid: %s, nOut: %d, nValidAtHeight in named queue: %d, nValidAtHeight in height queue: %d current height: %d\n", __func__, name, outPoint.hash.GetHex(), outPoint.n, nValidAtHeight, itRow->second.nValidAtHeight, nCurrentHeight);
|
|
}
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
LogPrintf("%s: An inconsistency was found in the claim queue. Please report this to the developers:\nFound in named queue but not in height queue: name: %s, txid: %s, nOut: %d, nValidAtHeight: %d, current height: %d\n", __func__, name, outPoint.hash.GetHex(), outPoint.n, nValidAtHeight, nCurrentHeight);
|
|
return false;
|
|
}
|
|
|
|
bool CClaimTrie::haveSupportInQueue(const std::string& name, const COutPoint& outPoint, int& nValidAtHeight) const
|
|
{
|
|
queueNameRowType nameRow;
|
|
if (!getSupportQueueNameRow(name, nameRow))
|
|
return false;
|
|
|
|
queueNameRowType::const_iterator itNameRow;
|
|
for (itNameRow = nameRow.begin(); itNameRow != nameRow.end(); ++itNameRow)
|
|
{
|
|
if (itNameRow->outPoint == outPoint)
|
|
{
|
|
nValidAtHeight = itNameRow->nHeight;
|
|
break;
|
|
}
|
|
}
|
|
if (itNameRow == nameRow.end())
|
|
return false;
|
|
|
|
supportQueueRowType row;
|
|
if (getSupportQueueRow(nValidAtHeight, row))
|
|
{
|
|
for (supportQueueRowType::const_iterator itRow = row.begin(); itRow != row.end(); ++itRow)
|
|
{
|
|
if (itRow->first == name && itRow->second.outPoint == outPoint)
|
|
{
|
|
if (itRow->second.nValidAtHeight != nValidAtHeight)
|
|
{
|
|
LogPrintf("%s: An inconsistency was found in the support queue. Please report this to the developers:\nDifferent nValidAtHeight between named queue and height queue\n: name: %s, txid: %s, nOut: %d, nValidAtHeight in named queue: %d, nValidAtHeight in height queue: %d current height: %d\n", __func__, name, outPoint.hash.GetHex(), outPoint.n, nValidAtHeight, itRow->second.nValidAtHeight, nCurrentHeight);
|
|
}
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
LogPrintf("%s: An inconsistency was found in the claim queue. Please report this to the developers:\nFound in named queue but not in height queue: name: %s, txid: %s, nOut: %d, nValidAtHeight: %d, current height: %d\n", __func__, name, outPoint.hash.GetHex(), outPoint.n, nValidAtHeight, nCurrentHeight);
|
|
return false;
|
|
}
|
|
|
|
unsigned int CClaimTrie::getTotalNamesInTrie() const
|
|
{
|
|
if (empty())
|
|
return 0;
|
|
const CClaimTrieNode* current = &root;
|
|
return getTotalNamesRecursive(current);
|
|
}
|
|
|
|
unsigned int CClaimTrie::getTotalNamesRecursive(const CClaimTrieNode* current) const
|
|
{
|
|
unsigned int names_in_subtrie = 0;
|
|
if (!(current->claims.empty()))
|
|
names_in_subtrie += 1;
|
|
for (nodeMapType::const_iterator it = current->children.begin(); it != current->children.end(); ++it)
|
|
{
|
|
names_in_subtrie += getTotalNamesRecursive(it->second);
|
|
}
|
|
return names_in_subtrie;
|
|
}
|
|
|
|
unsigned int CClaimTrie::getTotalClaimsInTrie() const
|
|
{
|
|
if (empty())
|
|
return 0;
|
|
const CClaimTrieNode* current = &root;
|
|
return getTotalClaimsRecursive(current);
|
|
}
|
|
|
|
unsigned int CClaimTrie::getTotalClaimsRecursive(const CClaimTrieNode* current) const
|
|
{
|
|
unsigned int claims_in_subtrie = current->claims.size();
|
|
for (nodeMapType::const_iterator it = current->children.begin(); it != current->children.end(); ++it)
|
|
{
|
|
claims_in_subtrie += getTotalClaimsRecursive(it->second);
|
|
}
|
|
return claims_in_subtrie;
|
|
}
|
|
|
|
CAmount CClaimTrie::getTotalValueOfClaimsInTrie(bool fControllingOnly) const
|
|
{
|
|
if (empty())
|
|
return 0;
|
|
const CClaimTrieNode* current = &root;
|
|
return getTotalValueOfClaimsRecursive(current, fControllingOnly);
|
|
}
|
|
|
|
CAmount CClaimTrie::getTotalValueOfClaimsRecursive(const CClaimTrieNode* current, bool fControllingOnly) const
|
|
{
|
|
CAmount value_in_subtrie = 0;
|
|
for (std::vector<CClaimValue>::const_iterator itclaim = current->claims.begin(); itclaim != current->claims.end(); ++itclaim)
|
|
{
|
|
value_in_subtrie += itclaim->nAmount;
|
|
if (fControllingOnly)
|
|
break;
|
|
}
|
|
for (nodeMapType::const_iterator itchild = current->children.begin(); itchild != current->children.end(); ++itchild)
|
|
{
|
|
value_in_subtrie += getTotalValueOfClaimsRecursive(itchild->second, fControllingOnly);
|
|
}
|
|
return value_in_subtrie;
|
|
}
|
|
|
|
const CClaimTrieNode* CClaimTrie::getNodeForName(const std::string& name) const
|
|
{
|
|
auto current = const_cast<CClaimTrieNode*>(&root);
|
|
for (std::string::const_iterator itname = name.begin(); itname != name.end(); ++itname)
|
|
{
|
|
nodeMapType::const_iterator itchildren = current->children.find(*itname);
|
|
if (itchildren == current->children.end())
|
|
return nullptr;
|
|
current = itchildren->second;
|
|
}
|
|
return current;
|
|
}
|
|
|
|
bool CClaimTrie::getInfoForName(const std::string& name, CClaimValue& claim) const
|
|
{
|
|
const CClaimTrieNode* current = getNodeForName(name);
|
|
return current ? current->getBestClaim(claim) : false;
|
|
}
|
|
|
|
bool CClaimTrie::getLastTakeoverForName(const std::string& name, int& lastTakeoverHeight) const
|
|
{
|
|
const CClaimTrieNode* current = getNodeForName(name);
|
|
if (current && !current->claims.empty())
|
|
{
|
|
lastTakeoverHeight = current->nHeightOfLastTakeover;
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
std::vector<CClaimValue> CClaimTrie::getClaimsForName(const std::string& name) const
|
|
{
|
|
static const std::vector<CClaimValue> empty{};
|
|
const CClaimTrieNode* current = getNodeForName(name);
|
|
return current == nullptr ? empty : current->claims;
|
|
}
|
|
|
|
bool CClaimTrie::checkConsistency() const
|
|
{
|
|
return empty() ? true : recursiveCheckConsistency(&root);
|
|
}
|
|
|
|
bool CClaimTrie::recursiveCheckConsistency(const CClaimTrieNode* node) const
|
|
{
|
|
std::vector<unsigned char> vchToHash;
|
|
|
|
for (nodeMapType::const_iterator it = node->children.begin(); it != node->children.end(); ++it)
|
|
{
|
|
if (!recursiveCheckConsistency(it->second))
|
|
return false;
|
|
|
|
vchToHash.push_back(it->first);
|
|
vchToHash.insert(vchToHash.end(), it->second->hash.begin(), it->second->hash.end());
|
|
}
|
|
|
|
CClaimValue claim;
|
|
if (node->getBestClaim(claim)) // hasClaim
|
|
{
|
|
uint256 valueHash = getValueHash(claim.outPoint, node->nHeightOfLastTakeover);
|
|
vchToHash.insert(vchToHash.end(), valueHash.begin(), valueHash.end());
|
|
}
|
|
|
|
auto calculatedHash = Hash(vchToHash.begin(), vchToHash.end());
|
|
return calculatedHash == node->hash;
|
|
}
|
|
|
|
void CClaimTrie::addToClaimIndex(const std::string& name, const CClaimValue& claim)
|
|
{
|
|
CClaimIndexElement element = { name, claim };
|
|
db.Write(std::make_pair(CLAIM_BY_ID, claim.claimId), element);
|
|
}
|
|
|
|
void CClaimTrie::removeFromClaimIndex(const CClaimValue& claim)
|
|
{
|
|
db.Erase(std::make_pair(CLAIM_BY_ID, claim.claimId));
|
|
}
|
|
|
|
bool CClaimTrie::getClaimById(const uint160 claimId, std::string& name, CClaimValue& claim) const
|
|
{
|
|
CClaimIndexElement element;
|
|
if (db.Read(std::make_pair(CLAIM_BY_ID, claimId), element))
|
|
{
|
|
if (element.claim.claimId == claimId)
|
|
{
|
|
name = element.name;
|
|
claim = element.claim;
|
|
return true;
|
|
}
|
|
|
|
LogPrintf("%s: ClaimIndex[%s] returned unmatched claimId %s when looking for %s\n",
|
|
__func__, claimId.GetHex(), element.claim.claimId.GetHex(), name);
|
|
}
|
|
return false;
|
|
}
|
|
|
|
bool CClaimTrie::getQueueRow(int nHeight, claimQueueRowType& row) const
|
|
{
|
|
claimQueueType::const_iterator itQueueRow = dirtyQueueRows.find(nHeight);
|
|
if (itQueueRow != dirtyQueueRows.end())
|
|
{
|
|
row = itQueueRow->second;
|
|
return true;
|
|
}
|
|
return db.Read(std::make_pair(CLAIM_QUEUE_ROW, nHeight), row);
|
|
}
|
|
|
|
bool CClaimTrie::getQueueNameRow(const std::string& name, queueNameRowType& row) const
|
|
{
|
|
queueNameType::const_iterator itQueueNameRow = dirtyQueueNameRows.find(name);
|
|
if (itQueueNameRow != dirtyQueueNameRows.end())
|
|
{
|
|
row = itQueueNameRow->second;
|
|
return true;
|
|
}
|
|
return db.Read(std::make_pair(CLAIM_QUEUE_NAME_ROW, name), row);
|
|
}
|
|
|
|
bool CClaimTrie::getExpirationQueueRow(int nHeight, expirationQueueRowType& row) const
|
|
{
|
|
expirationQueueType::const_iterator itQueueRow = dirtyExpirationQueueRows.find(nHeight);
|
|
if (itQueueRow != dirtyExpirationQueueRows.end())
|
|
{
|
|
row = itQueueRow->second;
|
|
return true;
|
|
}
|
|
return db.Read(std::make_pair(EXP_QUEUE_ROW, nHeight), row);
|
|
}
|
|
|
|
void CClaimTrie::updateQueueRow(int nHeight, claimQueueRowType& row)
|
|
{
|
|
claimQueueType::iterator itQueueRow = dirtyQueueRows.find(nHeight);
|
|
if (itQueueRow == dirtyQueueRows.end())
|
|
{
|
|
claimQueueRowType newRow;
|
|
std::pair<claimQueueType::iterator, bool> ret;
|
|
ret = dirtyQueueRows.insert(std::pair<int, claimQueueRowType >(nHeight, newRow));
|
|
assert(ret.second);
|
|
itQueueRow = ret.first;
|
|
}
|
|
itQueueRow->second.swap(row);
|
|
}
|
|
|
|
void CClaimTrie::updateQueueNameRow(const std::string& name, queueNameRowType& row)
|
|
{
|
|
queueNameType::iterator itQueueRow = dirtyQueueNameRows.find(name);
|
|
if (itQueueRow == dirtyQueueNameRows.end())
|
|
{
|
|
queueNameRowType newRow;
|
|
std::pair<queueNameType::iterator, bool> ret;
|
|
ret = dirtyQueueNameRows.insert(std::pair<std::string, queueNameRowType>(name, newRow));
|
|
assert(ret.second);
|
|
itQueueRow = ret.first;
|
|
}
|
|
itQueueRow->second.swap(row);
|
|
}
|
|
|
|
void CClaimTrie::updateExpirationRow(int nHeight, expirationQueueRowType& row)
|
|
{
|
|
expirationQueueType::iterator itQueueRow = dirtyExpirationQueueRows.find(nHeight);
|
|
if (itQueueRow == dirtyExpirationQueueRows.end())
|
|
{
|
|
expirationQueueRowType newRow;
|
|
std::pair<expirationQueueType::iterator, bool> ret;
|
|
ret = dirtyExpirationQueueRows.insert(std::pair<int, expirationQueueRowType >(nHeight, newRow));
|
|
assert(ret.second);
|
|
itQueueRow = ret.first;
|
|
}
|
|
itQueueRow->second.swap(row);
|
|
}
|
|
|
|
void CClaimTrie::updateSupportMap(const std::string& name, supportMapEntryType& node)
|
|
{
|
|
supportMapType::iterator itNode = dirtySupportNodes.find(name);
|
|
if (itNode == dirtySupportNodes.end())
|
|
{
|
|
supportMapEntryType newNode;
|
|
std::pair<supportMapType::iterator, bool> ret;
|
|
ret = dirtySupportNodes.insert(std::pair<std::string, supportMapEntryType>(name, newNode));
|
|
assert(ret.second);
|
|
itNode = ret.first;
|
|
}
|
|
itNode->second.swap(node);
|
|
}
|
|
|
|
void CClaimTrie::updateSupportQueue(int nHeight, supportQueueRowType& row)
|
|
{
|
|
supportQueueType::iterator itQueueRow = dirtySupportQueueRows.find(nHeight);
|
|
if (itQueueRow == dirtySupportQueueRows.end())
|
|
{
|
|
supportQueueRowType newRow;
|
|
std::pair<supportQueueType::iterator, bool> ret;
|
|
ret = dirtySupportQueueRows.insert(std::pair<int, supportQueueRowType >(nHeight, newRow));
|
|
assert(ret.second);
|
|
itQueueRow = ret.first;
|
|
}
|
|
itQueueRow->second.swap(row);
|
|
}
|
|
|
|
void CClaimTrie::updateSupportNameQueue(const std::string& name, queueNameRowType& row)
|
|
{
|
|
queueNameType::iterator itQueueRow = dirtySupportQueueNameRows.find(name);
|
|
if (itQueueRow == dirtySupportQueueNameRows.end())
|
|
{
|
|
queueNameRowType newRow;
|
|
std::pair<queueNameType::iterator, bool> ret;
|
|
ret = dirtySupportQueueNameRows.insert(std::pair<std::string, queueNameRowType>(name, newRow));
|
|
assert(ret.second);
|
|
itQueueRow = ret.first;
|
|
}
|
|
itQueueRow->second.swap(row);
|
|
}
|
|
|
|
void CClaimTrie::updateSupportExpirationQueue(int nHeight, expirationQueueRowType& row)
|
|
{
|
|
expirationQueueType::iterator itQueueRow = dirtySupportExpirationQueueRows.find(nHeight);
|
|
if (itQueueRow == dirtySupportExpirationQueueRows.end())
|
|
{
|
|
expirationQueueRowType newRow;
|
|
std::pair<expirationQueueType::iterator, bool> ret;
|
|
ret = dirtySupportExpirationQueueRows.insert(std::pair<int, expirationQueueRowType >(nHeight, newRow));
|
|
assert(ret.second);
|
|
itQueueRow = ret.first;
|
|
}
|
|
itQueueRow->second.swap(row);
|
|
}
|
|
|
|
bool CClaimTrie::getSupportNode(std::string name, supportMapEntryType& node) const
|
|
{
|
|
supportMapType::const_iterator itNode = dirtySupportNodes.find(name);
|
|
if (itNode != dirtySupportNodes.end())
|
|
{
|
|
node = itNode->second;
|
|
return true;
|
|
}
|
|
return db.Read(std::make_pair(SUPPORT, name), node);
|
|
}
|
|
|
|
bool CClaimTrie::getSupportQueueRow(int nHeight, supportQueueRowType& row) const
|
|
{
|
|
supportQueueType::const_iterator itQueueRow = dirtySupportQueueRows.find(nHeight);
|
|
if (itQueueRow != dirtySupportQueueRows.end())
|
|
{
|
|
row = itQueueRow->second;
|
|
return true;
|
|
}
|
|
return db.Read(std::make_pair(SUPPORT_QUEUE_ROW, nHeight), row);
|
|
}
|
|
|
|
bool CClaimTrie::getSupportQueueNameRow(const std::string& name, queueNameRowType& row) const
|
|
{
|
|
queueNameType::const_iterator itQueueNameRow = dirtySupportQueueNameRows.find(name);
|
|
if (itQueueNameRow != dirtySupportQueueNameRows.end())
|
|
{
|
|
row = itQueueNameRow->second;
|
|
return true;
|
|
}
|
|
return db.Read(std::make_pair(SUPPORT_QUEUE_NAME_ROW, name), row);
|
|
}
|
|
|
|
bool CClaimTrie::getSupportExpirationQueueRow(int nHeight, expirationQueueRowType& row) const
|
|
{
|
|
expirationQueueType::const_iterator itQueueRow = dirtySupportExpirationQueueRows.find(nHeight);
|
|
if (itQueueRow != dirtySupportExpirationQueueRows.end())
|
|
{
|
|
row = itQueueRow->second;
|
|
return true;
|
|
}
|
|
return db.Read(std::make_pair(SUPPORT_EXP_QUEUE_ROW, nHeight), row);
|
|
}
|
|
|
|
bool CClaimTrie::update(nodeCacheType& cache, hashMapType& hashes, std::map<std::string, int>& takeoverHeights, const uint256& hashBlockIn, claimQueueType& queueCache, queueNameType& queueNameCache, expirationQueueType& expirationQueueCache, int nNewHeight, supportMapType& supportCache, supportQueueType& supportQueueCache, queueNameType& supportQueueNameCache, expirationQueueType& supportExpirationQueueCache)
|
|
{
|
|
for (nodeCacheType::iterator itcache = cache.begin(); itcache != cache.end(); ++itcache)
|
|
{
|
|
if (!updateName(itcache->first, itcache->second))
|
|
{
|
|
LogPrintf("%s: Failed to update name for: %s\n", __func__, itcache->first);
|
|
return false;
|
|
}
|
|
}
|
|
for (hashMapType::iterator ithash = hashes.begin(); ithash != hashes.end(); ++ithash)
|
|
{
|
|
if (!updateHash(ithash->first, ithash->second))
|
|
{
|
|
LogPrintf("%s: Failed to update hash for: %s\n", __func__, ithash->first);
|
|
return false;
|
|
}
|
|
}
|
|
for (std::map<std::string, int>::iterator itheight = takeoverHeights.begin(); itheight != takeoverHeights.end(); ++itheight)
|
|
{
|
|
if (!updateTakeoverHeight(itheight->first, itheight->second))
|
|
{
|
|
LogPrintf("%s: Failed to update takeover height for: %s\n", __func__, itheight->first);
|
|
return false;
|
|
}
|
|
}
|
|
for (claimQueueType::iterator itQueueCacheRow = queueCache.begin(); itQueueCacheRow != queueCache.end(); ++itQueueCacheRow)
|
|
{
|
|
updateQueueRow(itQueueCacheRow->first, itQueueCacheRow->second);
|
|
}
|
|
for (queueNameType::iterator itQueueNameCacheRow = queueNameCache.begin(); itQueueNameCacheRow != queueNameCache.end(); ++itQueueNameCacheRow)
|
|
{
|
|
updateQueueNameRow(itQueueNameCacheRow->first, itQueueNameCacheRow->second);
|
|
}
|
|
for (expirationQueueType::iterator itExpirationRow = expirationQueueCache.begin(); itExpirationRow != expirationQueueCache.end(); ++itExpirationRow)
|
|
{
|
|
updateExpirationRow(itExpirationRow->first, itExpirationRow->second);
|
|
}
|
|
for (supportMapType::iterator itSupportCache = supportCache.begin(); itSupportCache != supportCache.end(); ++itSupportCache)
|
|
{
|
|
updateSupportMap(itSupportCache->first, itSupportCache->second);
|
|
}
|
|
for (supportQueueType::iterator itSupportQueue = supportQueueCache.begin(); itSupportQueue != supportQueueCache.end(); ++itSupportQueue)
|
|
{
|
|
updateSupportQueue(itSupportQueue->first, itSupportQueue->second);
|
|
}
|
|
for (queueNameType::iterator itSupportNameQueue = supportQueueNameCache.begin(); itSupportNameQueue != supportQueueNameCache.end(); ++itSupportNameQueue)
|
|
{
|
|
updateSupportNameQueue(itSupportNameQueue->first, itSupportNameQueue->second);
|
|
}
|
|
for (expirationQueueType::iterator itSupportExpirationQueue = supportExpirationQueueCache.begin(); itSupportExpirationQueue != supportExpirationQueueCache.end(); ++itSupportExpirationQueue)
|
|
{
|
|
updateSupportExpirationQueue(itSupportExpirationQueue->first, itSupportExpirationQueue->second);
|
|
}
|
|
hashBlock = hashBlockIn;
|
|
nCurrentHeight = nNewHeight;
|
|
return true;
|
|
}
|
|
|
|
void CClaimTrie::markNodeDirty(const std::string &name, CClaimTrieNode* node)
|
|
{
|
|
std::pair<nodeCacheType::iterator, bool> ret;
|
|
ret = dirtyNodes.insert(std::pair<std::string, CClaimTrieNode*>(name, node));
|
|
if (ret.second == false)
|
|
ret.first->second = node;
|
|
}
|
|
|
|
bool CClaimTrie::updateName(const std::string &name, CClaimTrieNode* updatedNode)
|
|
{
|
|
CClaimTrieNode* current = &root;
|
|
for (std::string::const_iterator itname = name.begin(); itname != name.end(); ++itname)
|
|
{
|
|
nodeMapType::iterator itchild = current->children.find(*itname);
|
|
if (itchild == current->children.end())
|
|
{
|
|
if (itname + 1 != name.end())
|
|
return false;
|
|
|
|
CClaimTrieNode* newNode = new CClaimTrieNode();
|
|
current->children[*itname] = newNode;
|
|
current = newNode;
|
|
}
|
|
else
|
|
{
|
|
current = itchild->second;
|
|
}
|
|
}
|
|
assert(current != nullptr);
|
|
current->claims.swap(updatedNode->claims);
|
|
markNodeDirty(name, current);
|
|
for (nodeMapType::iterator itchild = current->children.begin(); itchild != current->children.end();)
|
|
{
|
|
nodeMapType::iterator itupdatechild = updatedNode->children.find(itchild->first);
|
|
if (itupdatechild == updatedNode->children.end())
|
|
{
|
|
// This character has apparently been deleted, so delete
|
|
// all descendents from this child.
|
|
std::stringstream nextName;
|
|
nextName << name << itchild->first;
|
|
if (!recursiveNullify(itchild->second, nextName.str()))
|
|
return false;
|
|
current->children.erase(itchild++);
|
|
}
|
|
else
|
|
{
|
|
++itchild;
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
bool CClaimTrie::recursiveNullify(CClaimTrieNode* node, const std::string& name)
|
|
{
|
|
assert(node != nullptr);
|
|
for (nodeMapType::iterator itchild = node->children.begin(); itchild != node->children.end(); ++itchild)
|
|
{
|
|
std::stringstream nextName;
|
|
nextName << name << itchild->first;
|
|
if (!recursiveNullify(itchild->second, nextName.str()))
|
|
return false;
|
|
}
|
|
node->children.clear();
|
|
markNodeDirty(name, nullptr);
|
|
delete node;
|
|
|
|
return true;
|
|
}
|
|
|
|
bool CClaimTrie::updateHash(const std::string& name, uint256& hash)
|
|
{
|
|
CClaimTrieNode* current = &root;
|
|
for (std::string::const_iterator itname = name.begin(); itname != name.end(); ++itname)
|
|
{
|
|
nodeMapType::iterator itchild = current->children.find(*itname);
|
|
if (itchild == current->children.end())
|
|
return false;
|
|
current = itchild->second;
|
|
}
|
|
assert(current != nullptr);
|
|
assert(!hash.IsNull());
|
|
current->hash = hash;
|
|
markNodeDirty(name, current);
|
|
return true;
|
|
}
|
|
|
|
bool CClaimTrie::updateTakeoverHeight(const std::string& name, int nTakeoverHeight)
|
|
{
|
|
CClaimTrieNode* current = &root;
|
|
for (std::string::const_iterator itname = name.begin(); itname != name.end(); ++itname)
|
|
{
|
|
nodeMapType::iterator itchild = current->children.find(*itname);
|
|
if (itchild == current->children.end())
|
|
return false;
|
|
current = itchild->second;
|
|
}
|
|
assert(current != nullptr);
|
|
current->nHeightOfLastTakeover = nTakeoverHeight;
|
|
markNodeDirty(name, current);
|
|
return true;
|
|
}
|
|
|
|
void CClaimTrie::BatchWriteNode(CDBBatch& batch, const std::string& name, const CClaimTrieNode* pNode) const
|
|
{
|
|
if (pNode)
|
|
batch.Write(std::make_pair(TRIE_NODE, name), *pNode);
|
|
else
|
|
batch.Erase(std::make_pair(TRIE_NODE, name));
|
|
}
|
|
|
|
void CClaimTrie::BatchWriteQueueRows(CDBBatch& batch)
|
|
{
|
|
for (claimQueueType::iterator itQueue = dirtyQueueRows.begin(); itQueue != dirtyQueueRows.end(); ++itQueue)
|
|
{
|
|
if (itQueue->second.empty())
|
|
{
|
|
batch.Erase(std::make_pair(CLAIM_QUEUE_ROW, itQueue->first));
|
|
}
|
|
else
|
|
{
|
|
batch.Write(std::make_pair(CLAIM_QUEUE_ROW, itQueue->first), itQueue->second);
|
|
}
|
|
}
|
|
}
|
|
|
|
void CClaimTrie::BatchWriteQueueNameRows(CDBBatch& batch)
|
|
{
|
|
for (queueNameType::iterator itQueue = dirtyQueueNameRows.begin(); itQueue != dirtyQueueNameRows.end(); ++itQueue)
|
|
{
|
|
if (itQueue->second.empty())
|
|
{
|
|
batch.Erase(std::make_pair(CLAIM_QUEUE_NAME_ROW, itQueue->first));
|
|
}
|
|
else
|
|
{
|
|
batch.Write(std::make_pair(CLAIM_QUEUE_NAME_ROW, itQueue->first), itQueue->second);
|
|
}
|
|
}
|
|
}
|
|
|
|
void CClaimTrie::BatchWriteExpirationQueueRows(CDBBatch& batch)
|
|
{
|
|
for (expirationQueueType::iterator itQueue = dirtyExpirationQueueRows.begin(); itQueue != dirtyExpirationQueueRows.end(); ++itQueue)
|
|
{
|
|
if (itQueue->second.empty())
|
|
{
|
|
batch.Erase(std::make_pair(EXP_QUEUE_ROW, itQueue->first));
|
|
}
|
|
else
|
|
{
|
|
batch.Write(std::make_pair(EXP_QUEUE_ROW, itQueue->first), itQueue->second);
|
|
}
|
|
}
|
|
}
|
|
|
|
void CClaimTrie::BatchWriteSupportNodes(CDBBatch& batch)
|
|
{
|
|
for (supportMapType::iterator itSupport = dirtySupportNodes.begin(); itSupport != dirtySupportNodes.end(); ++itSupport)
|
|
{
|
|
if (itSupport->second.empty())
|
|
{
|
|
batch.Erase(std::make_pair(SUPPORT, itSupport->first));
|
|
}
|
|
else
|
|
{
|
|
batch.Write(std::make_pair(SUPPORT, itSupport->first), itSupport->second);
|
|
}
|
|
}
|
|
}
|
|
|
|
void CClaimTrie::BatchWriteSupportQueueRows(CDBBatch& batch)
|
|
{
|
|
for (supportQueueType::iterator itQueue = dirtySupportQueueRows.begin(); itQueue != dirtySupportQueueRows.end(); ++itQueue)
|
|
{
|
|
if (itQueue->second.empty())
|
|
{
|
|
batch.Erase(std::make_pair(SUPPORT_QUEUE_ROW, itQueue->first));
|
|
}
|
|
else
|
|
{
|
|
batch.Write(std::make_pair(SUPPORT_QUEUE_ROW, itQueue->first), itQueue->second);
|
|
}
|
|
}
|
|
}
|
|
|
|
void CClaimTrie::BatchWriteSupportQueueNameRows(CDBBatch& batch)
|
|
{
|
|
for (queueNameType::iterator itQueue = dirtySupportQueueNameRows.begin(); itQueue != dirtySupportQueueNameRows.end(); ++itQueue)
|
|
{
|
|
if (itQueue->second.empty())
|
|
{
|
|
batch.Erase(std::make_pair(SUPPORT_QUEUE_NAME_ROW, itQueue->first));
|
|
}
|
|
else
|
|
{
|
|
batch.Write(std::make_pair(SUPPORT_QUEUE_NAME_ROW, itQueue->first), itQueue->second);
|
|
}
|
|
}
|
|
}
|
|
|
|
void CClaimTrie::BatchWriteSupportExpirationQueueRows(CDBBatch& batch)
|
|
{
|
|
for (expirationQueueType::iterator itQueue = dirtySupportExpirationQueueRows.begin(); itQueue != dirtySupportExpirationQueueRows.end(); ++itQueue)
|
|
{
|
|
if (itQueue->second.empty())
|
|
{
|
|
batch.Erase(std::make_pair(SUPPORT_EXP_QUEUE_ROW, itQueue->first));
|
|
}
|
|
else
|
|
{
|
|
batch.Write(std::make_pair(SUPPORT_EXP_QUEUE_ROW, itQueue->first), itQueue->second);
|
|
}
|
|
}
|
|
}
|
|
|
|
bool CClaimTrie::WriteToDisk()
|
|
{
|
|
CDBBatch batch(db);
|
|
for (nodeCacheType::iterator itcache = dirtyNodes.begin(); itcache != dirtyNodes.end(); ++itcache)
|
|
BatchWriteNode(batch, itcache->first, itcache->second);
|
|
dirtyNodes.clear();
|
|
BatchWriteQueueRows(batch);
|
|
dirtyQueueRows.clear();
|
|
BatchWriteQueueNameRows(batch);
|
|
dirtyQueueNameRows.clear();
|
|
BatchWriteExpirationQueueRows(batch);
|
|
dirtyExpirationQueueRows.clear();
|
|
BatchWriteSupportNodes(batch);
|
|
dirtySupportNodes.clear();
|
|
BatchWriteSupportQueueRows(batch);
|
|
dirtySupportQueueRows.clear();
|
|
BatchWriteSupportQueueNameRows(batch);
|
|
dirtySupportQueueNameRows.clear();
|
|
BatchWriteSupportExpirationQueueRows(batch);
|
|
dirtySupportExpirationQueueRows.clear();
|
|
batch.Write(HASH_BLOCK, hashBlock);
|
|
batch.Write(CURRENT_HEIGHT, nCurrentHeight);
|
|
return db.WriteBatch(batch);
|
|
}
|
|
|
|
bool CClaimTrie::InsertFromDisk(const std::string& name, CClaimTrieNode* node)
|
|
{
|
|
if (name.empty())
|
|
{
|
|
root = *node;
|
|
return true;
|
|
}
|
|
CClaimTrieNode* current = &root;
|
|
for (std::string::const_iterator itname = name.begin(); itname + 1 != name.end(); ++itname)
|
|
{
|
|
nodeMapType::iterator itchild = current->children.find(*itname);
|
|
if (itchild == current->children.end())
|
|
return false;
|
|
current = itchild->second;
|
|
}
|
|
current->children[name[name.size() - 1]] = node;
|
|
return true;
|
|
}
|
|
|
|
bool CClaimTrie::ReadFromDisk(bool check)
|
|
{
|
|
if (!db.Read(HASH_BLOCK, hashBlock))
|
|
LogPrintf("%s: Couldn't read the best block's hash\n", __func__);
|
|
if (!db.Read(CURRENT_HEIGHT, nCurrentHeight))
|
|
LogPrintf("%s: Couldn't read the current height\n", __func__);
|
|
setExpirationTime(Params().GetConsensus().GetExpirationTime(nCurrentHeight-1));
|
|
|
|
boost::scoped_ptr<CDBIterator> pcursor(db.NewIterator());
|
|
|
|
pcursor->SeekToFirst();
|
|
while (pcursor->Valid())
|
|
{
|
|
std::pair<char, std::string> key;
|
|
if (pcursor->GetKey(key))
|
|
{
|
|
if (key.first == TRIE_NODE)
|
|
{
|
|
CClaimTrieNode* node = new CClaimTrieNode();
|
|
if (pcursor->GetValue(*node))
|
|
{
|
|
if (!InsertFromDisk(key.second, node))
|
|
{
|
|
return error("%s(): error restoring claim trie from disk", __func__);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
return error("%s(): error reading claim trie from disk", __func__);
|
|
}
|
|
}
|
|
}
|
|
pcursor->Next();
|
|
}
|
|
|
|
if (check)
|
|
{
|
|
LogPrintf("Checking Claim trie consistency...");
|
|
if (checkConsistency())
|
|
{
|
|
LogPrintf("consistent\n");
|
|
return true;
|
|
}
|
|
LogPrintf("inconsistent!\n");
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
bool CClaimTrieCacheBase::recursiveComputeMerkleHash(CClaimTrieNode* tnCurrent, const std::string& sPos, bool forceCompute) const
|
|
{
|
|
if ((sPos == rootClaimName) && tnCurrent->empty())
|
|
{
|
|
cacheHashes[rootClaimName] = uint256S(rootClaimHash);
|
|
return true;
|
|
}
|
|
std::vector<unsigned char> vchToHash;
|
|
nodeCacheType::iterator cachedNode;
|
|
|
|
for (nodeMapType::iterator it = tnCurrent->children.begin(); it != tnCurrent->children.end(); ++it)
|
|
{
|
|
std::stringstream ss;
|
|
ss << sPos << it->first;
|
|
const std::string& sNextPos = ss.str();
|
|
if ((dirtyHashes.count(sNextPos) != 0) || forceCompute)
|
|
{
|
|
// the child might be in the cache, so look for it there
|
|
cachedNode = cache.find(sNextPos);
|
|
if (cachedNode != cache.end())
|
|
recursiveComputeMerkleHash(cachedNode->second, sNextPos, forceCompute);
|
|
else
|
|
recursiveComputeMerkleHash(it->second, sNextPos, forceCompute);
|
|
}
|
|
vchToHash.push_back(sNextPos[sNextPos.size() - 1]);
|
|
|
|
hashMapType::iterator ithash = cacheHashes.find(sNextPos);
|
|
if (ithash != cacheHashes.end())
|
|
{
|
|
assert(!ithash->second.IsNull()); // fast call if it's not actually null (as the first byte will bail)
|
|
vchToHash.insert(vchToHash.end(), ithash->second.begin(), ithash->second.end());
|
|
}
|
|
else
|
|
{
|
|
assert(!it->second->hash.IsNull());
|
|
vchToHash.insert(vchToHash.end(), it->second->hash.begin(), it->second->hash.end());
|
|
}
|
|
}
|
|
|
|
CClaimValue claim;
|
|
if (tnCurrent->getBestClaim(claim)) // hasClaim
|
|
{
|
|
int nHeightOfLastTakeover;
|
|
assert(getLastTakeoverForName(sPos, nHeightOfLastTakeover));
|
|
uint256 valueHash = getValueHash(claim.outPoint, nHeightOfLastTakeover);
|
|
vchToHash.insert(vchToHash.end(), valueHash.begin(), valueHash.end());
|
|
}
|
|
|
|
CHash256 hasher;
|
|
hasher.Write(vchToHash.data(), vchToHash.size());
|
|
std::vector<unsigned char> vchHash(hasher.OUTPUT_SIZE);
|
|
hasher.Finalize(&(vchHash[0]));
|
|
cacheHashes[sPos] = uint256(vchHash);
|
|
dirtyHashes.erase(sPos);
|
|
|
|
return true;
|
|
}
|
|
|
|
uint256 CClaimTrieCacheBase::getMerkleHash(bool forceCompute) const
|
|
{
|
|
if (empty())
|
|
{
|
|
static const uint256 one(uint256S(rootClaimHash));
|
|
return one;
|
|
}
|
|
if (forceCompute || dirty())
|
|
{
|
|
CClaimTrieNode* root = getRoot();
|
|
recursiveComputeMerkleHash(root, rootClaimName, forceCompute);
|
|
dirtyHashes.clear();
|
|
}
|
|
hashMapType::iterator ithash = cacheHashes.find(rootClaimName);
|
|
return ((ithash != cacheHashes.end()) ? ithash->second : base->root.hash);
|
|
}
|
|
|
|
bool CClaimTrieCacheBase::empty() const
|
|
{
|
|
return base->empty() && cache.empty();
|
|
}
|
|
|
|
// "position" has already been normalized if needed
|
|
CClaimTrieNode* CClaimTrieCacheBase::addNodeToCache(const std::string& position, CClaimTrieNode* original) const
|
|
{
|
|
// create a copy of the node in the cache, if new node, create empty node
|
|
CClaimTrieNode* cacheCopy = (original ? new CClaimTrieNode(*original) : new CClaimTrieNode());
|
|
cache[position] = cacheCopy;
|
|
|
|
// check to see if there is the original node in block_originals,
|
|
// if not, add it to block_originals cache
|
|
nodeCacheType::const_iterator itOriginals = block_originals.find(position);
|
|
if (block_originals.end() == itOriginals)
|
|
block_originals[position] = (original ? new CClaimTrieNode(*original) : new CClaimTrieNode());
|
|
|
|
return cacheCopy;
|
|
}
|
|
|
|
bool CClaimTrieCacheBase::getOriginalInfoForName(const std::string& name, CClaimValue& claim) const
|
|
{
|
|
nodeCacheType::const_iterator itOriginalCache = block_originals.find(name);
|
|
return ((itOriginalCache == block_originals.end()) ? base->getInfoForName(name, claim) :
|
|
itOriginalCache->second->getBestClaim(claim));
|
|
}
|
|
|
|
bool CClaimTrieCacheBase::insertClaimIntoTrie(const std::string& name, CClaimValue claim, bool fCheckTakeover) const
|
|
{
|
|
CClaimTrieNode* currentNode = getRoot();
|
|
nodeCacheType::iterator cachedNode;
|
|
for (std::string::const_iterator itCur = name.begin(); itCur != name.end(); ++itCur)
|
|
{
|
|
const std::string sCurrentSubstring(name.begin(), itCur);
|
|
const std::string sNextSubstring(name.begin(), itCur + 1);
|
|
|
|
cachedNode = cache.find(sNextSubstring);
|
|
if (cachedNode != cache.end())
|
|
{
|
|
currentNode = cachedNode->second;
|
|
continue;
|
|
}
|
|
nodeMapType::iterator childNode = currentNode->children.find(*itCur);
|
|
if (childNode != currentNode->children.end())
|
|
{
|
|
currentNode = childNode->second;
|
|
continue;
|
|
}
|
|
|
|
// This next substring doesn't exist in the cache and the next
|
|
// character doesn't exist in current node's children, so check
|
|
// if the current node is in the cache, and if it's not, copy
|
|
// it and stick it in the cache, and then create a new node as
|
|
// its child and stick that in the cache. We have to have both
|
|
// this node and its child in the cache so that the current
|
|
// node's child map will contain the next letter, which will be
|
|
// used to find the child in the cache. This is necessary in
|
|
// order to calculate the merkle hash.
|
|
cachedNode = cache.find(sCurrentSubstring);
|
|
if (cachedNode != cache.end())
|
|
{
|
|
assert(cachedNode->second == currentNode);
|
|
}
|
|
else
|
|
{
|
|
currentNode = addNodeToCache(sCurrentSubstring, currentNode);
|
|
}
|
|
CClaimTrieNode* newNode = addNodeToCache(sNextSubstring, nullptr);
|
|
currentNode->children[*itCur] = newNode;
|
|
currentNode = newNode;
|
|
}
|
|
|
|
cachedNode = cache.find(name);
|
|
if (cachedNode != cache.end())
|
|
{
|
|
assert(cachedNode->second == currentNode);
|
|
}
|
|
else
|
|
{
|
|
currentNode = addNodeToCache(name, currentNode);
|
|
}
|
|
bool fChanged = false;
|
|
if (currentNode->claims.empty())
|
|
{
|
|
fChanged = true;
|
|
currentNode->insertClaim(claim);
|
|
}
|
|
else
|
|
{
|
|
CClaimValue currentTop = currentNode->claims.front();
|
|
currentNode->insertClaim(claim);
|
|
supportMapEntryType node;
|
|
getSupportsForName(name, node);
|
|
currentNode->reorderClaims(node);
|
|
if (currentTop != currentNode->claims.front())
|
|
fChanged = true;
|
|
}
|
|
|
|
if (fChanged)
|
|
{
|
|
for (std::string::const_iterator itCur = name.begin(); itCur != name.end(); ++itCur)
|
|
{
|
|
std::string sub(name.begin(), itCur);
|
|
dirtyHashes.insert(sub);
|
|
}
|
|
dirtyHashes.insert(name);
|
|
if (fCheckTakeover)
|
|
namesToCheckForTakeover.insert(name);
|
|
}
|
|
return true;
|
|
}
|
|
|
|
bool CClaimTrieCacheBase::removeClaimFromTrie(const std::string& name, const COutPoint& outPoint, CClaimValue& claim, bool fCheckTakeover) const
|
|
{
|
|
CClaimTrieNode* currentNode = getRoot();
|
|
nodeCacheType::iterator cachedNode;
|
|
assert(currentNode != nullptr); // If there is no root in either the trie or the cache, how can there be any names to remove?
|
|
for (std::string::const_iterator itCur = name.begin(); itCur != name.end(); ++itCur)
|
|
{
|
|
std::string sCurrentSubstring(name.begin(), itCur);
|
|
std::string sNextSubstring(name.begin(), itCur + 1);
|
|
|
|
cachedNode = cache.find(sNextSubstring);
|
|
if (cachedNode != cache.end())
|
|
{
|
|
currentNode = cachedNode->second;
|
|
continue;
|
|
}
|
|
nodeMapType::iterator childNode = currentNode->children.find(*itCur);
|
|
if (childNode != currentNode->children.end())
|
|
{
|
|
currentNode = childNode->second;
|
|
continue;
|
|
}
|
|
LogPrintf("%s: The name %s does not exist in the trie\n", __func__, name.c_str());
|
|
return false;
|
|
}
|
|
|
|
cachedNode = cache.find(name);
|
|
if (cachedNode != cache.end())
|
|
assert(cachedNode->second == currentNode);
|
|
else
|
|
currentNode = addNodeToCache(name, currentNode);
|
|
|
|
assert(currentNode != nullptr);
|
|
if (currentNode->claims.empty())
|
|
{
|
|
LogPrintf("%s: Asked to remove claim from node without claims\n", __func__);
|
|
return false;
|
|
}
|
|
|
|
bool success = currentNode->removeClaim(outPoint, claim);
|
|
if (!currentNode->claims.empty())
|
|
{
|
|
supportMapEntryType node;
|
|
getSupportsForName(name, node);
|
|
currentNode->reorderClaims(node);
|
|
}
|
|
|
|
if (!success)
|
|
{
|
|
LogPrintf("%s: Removing a claim was unsuccessful. name = %s, txhash = %s, nOut = %d\n",
|
|
__func__, name.c_str(), outPoint.hash.GetHex(), outPoint.n);
|
|
return false;
|
|
}
|
|
|
|
for (std::string::const_iterator itCur = name.begin(); itCur != name.end(); ++itCur)
|
|
{
|
|
std::string sub(name.begin(), itCur);
|
|
dirtyHashes.insert(sub);
|
|
}
|
|
dirtyHashes.insert(name);
|
|
if (fCheckTakeover)
|
|
namesToCheckForTakeover.insert(name);
|
|
|
|
return recursivePruneName(getRoot(), 0, name);
|
|
}
|
|
|
|
// sName has already been normalized if needed
|
|
bool CClaimTrieCacheBase::recursivePruneName(CClaimTrieNode* tnCurrent, unsigned int nPos, const std::string& sName, bool* pfNullified) const
|
|
{
|
|
// Recursively prune leaf node(s) without any claims in it and store
|
|
// the modified nodes in the cache
|
|
|
|
bool fNullified = false;
|
|
std::string sCurrentSubstring = sName.substr(0, nPos);
|
|
if (nPos < sName.size())
|
|
{
|
|
std::string sNextSubstring = sName.substr(0, nPos + 1);
|
|
unsigned char cNext = sName.at(nPos);
|
|
CClaimTrieNode* tnNext = nullptr;
|
|
nodeCacheType::iterator cachedNode = cache.find(sNextSubstring);
|
|
if (cachedNode != cache.end())
|
|
{
|
|
tnNext = cachedNode->second;
|
|
}
|
|
else
|
|
{
|
|
nodeMapType::iterator childNode = tnCurrent->children.find(cNext);
|
|
if (childNode != tnCurrent->children.end())
|
|
tnNext = childNode->second;
|
|
}
|
|
if (tnNext == nullptr)
|
|
return false;
|
|
bool fChildNullified = false;
|
|
if (!recursivePruneName(tnNext, nPos + 1, sName, &fChildNullified))
|
|
return false;
|
|
if (fChildNullified)
|
|
{
|
|
// If the child nullified itself, the child should already be
|
|
// out of the cache, and the character must now be removed
|
|
// from the current node's map of child nodes to ensure that
|
|
// it isn't found when calculating the merkle hash. But
|
|
// tnCurrent isn't necessarily in the cache. If it's not, it
|
|
// has to be added to the cache, so nothing is changed in the
|
|
// trie. If the current node is added to the cache, however,
|
|
// that does not imply that the parent node must be altered to
|
|
// reflect that its child is now in the cache, since it
|
|
// already has a character in its child map which will be used
|
|
// when calculating the merkle root.
|
|
|
|
// First, find out if this node is in the cache.
|
|
cachedNode = cache.find(sCurrentSubstring);
|
|
if (cachedNode == cache.end())
|
|
{
|
|
// it isn't, so make a copy, stick it in the cache,
|
|
// and make it the new current node
|
|
tnCurrent = addNodeToCache(sCurrentSubstring, tnCurrent);
|
|
}
|
|
// erase the character from the current node, which is
|
|
// now guaranteed to be in the cache
|
|
nodeMapType::iterator childNode = tnCurrent->children.find(cNext);
|
|
if (childNode != tnCurrent->children.end())
|
|
tnCurrent->children.erase(childNode);
|
|
else
|
|
return false;
|
|
}
|
|
}
|
|
if (!sCurrentSubstring.empty() && tnCurrent->empty())
|
|
{
|
|
// If the current node is in the cache, remove it from there
|
|
nodeCacheType::iterator cachedNode = cache.find(sCurrentSubstring);
|
|
if (cachedNode != cache.end())
|
|
{
|
|
assert(tnCurrent == cachedNode->second);
|
|
delete tnCurrent;
|
|
cache.erase(cachedNode);
|
|
}
|
|
fNullified = true;
|
|
}
|
|
if (pfNullified)
|
|
*pfNullified = fNullified;
|
|
return true;
|
|
}
|
|
|
|
claimQueueType::iterator CClaimTrieCacheBase::getQueueCacheRow(int nHeight, bool createIfNotExists) const
|
|
{
|
|
claimQueueType::iterator itQueueRow = claimQueueCache.find(nHeight);
|
|
if (itQueueRow == claimQueueCache.end())
|
|
{
|
|
// Have to make a new row it put in the cache, if createIfNotExists is true
|
|
claimQueueRowType queueRow;
|
|
// If the row exists in the base, copy its claims into the new row.
|
|
bool exists = base->getQueueRow(nHeight, queueRow);
|
|
if (!exists && !createIfNotExists)
|
|
return itQueueRow;
|
|
// Stick the new row in the cache
|
|
std::pair<claimQueueType::iterator, bool> ret;
|
|
ret = claimQueueCache.insert(std::pair<int, claimQueueRowType >(nHeight, queueRow));
|
|
assert(ret.second);
|
|
itQueueRow = ret.first;
|
|
}
|
|
return itQueueRow;
|
|
}
|
|
|
|
queueNameType::iterator CClaimTrieCacheBase::getQueueCacheNameRow(const std::string& name, bool createIfNotExists) const
|
|
{
|
|
queueNameType::iterator itQueueNameRow = claimQueueNameCache.find(name);
|
|
if (itQueueNameRow == claimQueueNameCache.end())
|
|
{
|
|
// Have to make a new name row and put it in the cache, if createIfNotExists is true
|
|
queueNameRowType queueNameRow;
|
|
// If the row exists in the base, copy its claims into the new row.
|
|
bool exists = base->getQueueNameRow(name, queueNameRow);
|
|
if (!exists && !createIfNotExists)
|
|
return itQueueNameRow;
|
|
// Stick the new row in the cache
|
|
std::pair<queueNameType::iterator, bool> ret;
|
|
ret = claimQueueNameCache.insert(std::pair<std::string, queueNameRowType>(name, queueNameRow));
|
|
assert(ret.second);
|
|
itQueueNameRow = ret.first;
|
|
}
|
|
return itQueueNameRow;
|
|
}
|
|
|
|
bool CClaimTrieCacheBase::addClaim(const std::string& name, const COutPoint& outPoint, uint160 claimId, CAmount nAmount, int nHeight) const
|
|
{
|
|
LogPrintf("%s: name: %s, txhash: %s, nOut: %d, claimId: %s, nAmount: %u, nHeight: %d, nCurrentHeight: %lu\n", __func__, name, outPoint.hash.GetHex(), outPoint.n, claimId.GetHex(), nAmount, nHeight, nCurrentHeight);
|
|
assert(nHeight == nCurrentHeight);
|
|
|
|
int delayForClaim = getDelayForName(name, claimId);
|
|
CClaimValue claim(outPoint, claimId, nAmount, nHeight, nHeight + delayForClaim);
|
|
addClaimToQueues(name, claim);
|
|
CClaimIndexElement element = { name, claim };
|
|
claimsToAdd.push_back(element);
|
|
return true;
|
|
}
|
|
|
|
bool CClaimTrieCacheBase::undoSpendClaim(const std::string& name, const COutPoint& outPoint, uint160 claimId, CAmount nAmount, int nHeight, int nValidAtHeight) const
|
|
{
|
|
LogPrintf("%s: name: %s, txhash: %s, nOut: %d, claimId: %s, nAmount: %d, nHeight: %d, nValidAtHeight: %d, nCurrentHeight: %d\n", __func__, name, outPoint.hash.GetHex(), outPoint.n, claimId.GetHex(), nAmount, nHeight, nValidAtHeight, nCurrentHeight);
|
|
CClaimValue claim(outPoint, claimId, nAmount, nHeight, nValidAtHeight);
|
|
if (nValidAtHeight < nCurrentHeight)
|
|
{
|
|
nameOutPointType entry(adjustNameForValidHeight(name, nValidAtHeight), claim.outPoint);
|
|
addToExpirationQueue(claim.nHeight + base->nExpirationTime, entry);
|
|
CClaimIndexElement element = {name, claim};
|
|
claimsToAdd.push_back(element);
|
|
return insertClaimIntoTrie(name, claim, false);
|
|
}
|
|
addClaimToQueues(name, claim);
|
|
CClaimIndexElement element = { name, claim };
|
|
claimsToAdd.push_back(element);
|
|
return true;
|
|
}
|
|
|
|
void CClaimTrieCacheBase::addClaimToQueues(const std::string& name, CClaimValue& claim) const
|
|
{
|
|
// name is not normalized and we always keep the original name data in the queues and index
|
|
claimQueueEntryType entry(name, claim);
|
|
claimQueueType::iterator itQueueRow = getQueueCacheRow(claim.nValidAtHeight, true);
|
|
queueNameType::iterator itQueueNameRow = getQueueCacheNameRow(name, true);
|
|
itQueueRow->second.push_back(entry);
|
|
itQueueNameRow->second.push_back(outPointHeightType(claim.outPoint, claim.nValidAtHeight));
|
|
|
|
nameOutPointType expireEntry(name, claim.outPoint);
|
|
addToExpirationQueue(claim.nHeight + base->nExpirationTime, expireEntry);
|
|
}
|
|
|
|
std::string CClaimTrieCacheBase::adjustNameForValidHeight(const std::string& name, int validHeight) const {
|
|
return name;
|
|
}
|
|
|
|
bool CClaimTrieCacheBase::removeClaimFromQueue(const std::string& name, const COutPoint& outPoint, CClaimValue& claim) const
|
|
{
|
|
queueNameType::iterator itQueueNameRow = getQueueCacheNameRow(name, false);
|
|
if (itQueueNameRow == claimQueueNameCache.end())
|
|
return false;
|
|
|
|
queueNameRowType::iterator itQueueName;
|
|
for (itQueueName = itQueueNameRow->second.begin(); itQueueName != itQueueNameRow->second.end(); ++itQueueName)
|
|
{
|
|
if (itQueueName->outPoint == outPoint)
|
|
break;
|
|
}
|
|
if (itQueueName == itQueueNameRow->second.end())
|
|
return false;
|
|
|
|
claimQueueType::iterator itQueueRow = getQueueCacheRow(itQueueName->nHeight, false);
|
|
if (itQueueRow != claimQueueCache.end())
|
|
{
|
|
claimQueueRowType::iterator itQueue;
|
|
// compare normalized names for the UPDATE_OP; we allow you to change the case on an update after the fork
|
|
for (itQueue = itQueueRow->second.begin(); itQueue != itQueueRow->second.end(); ++itQueue)
|
|
{
|
|
if (itQueue->second.outPoint == outPoint && name == itQueue->first)
|
|
break;
|
|
}
|
|
if (itQueue != itQueueRow->second.end())
|
|
{
|
|
claim = itQueue->second;
|
|
itQueueNameRow->second.erase(itQueueName);
|
|
itQueueRow->second.erase(itQueue);
|
|
return true;
|
|
}
|
|
}
|
|
LogPrintf("%s: An inconsistency was found in the claim queue. Please report this to the developers:\nFound in named queue but not in height queue: name: %s, txid: %s, nOut: %d, nValidAtHeight: %d, current height: %d\n", __func__, name, outPoint.hash.GetHex(), outPoint.n, itQueueName->nHeight, nCurrentHeight);
|
|
return false;
|
|
}
|
|
|
|
bool CClaimTrieCacheBase::undoAddClaim(const std::string& name, const COutPoint& outPoint, int nHeight) const
|
|
{
|
|
int throwaway;
|
|
return removeClaim(name, outPoint, nHeight, throwaway, false);
|
|
}
|
|
|
|
bool CClaimTrieCacheBase::spendClaim(const std::string& name, const COutPoint& outPoint, int nHeight, int& nValidAtHeight) const
|
|
{
|
|
return removeClaim(name, outPoint, nHeight, nValidAtHeight, true);
|
|
}
|
|
|
|
bool CClaimTrieCacheBase::removeClaim(const std::string& name, const COutPoint& outPoint, int nHeight, int& nValidAtHeight, bool fCheckTakeover) const
|
|
{
|
|
LogPrintf("%s: name: %s, txhash: %s, nOut: %s, nHeight: %s, nCurrentHeight: %s\n", __func__, name, outPoint.hash.GetHex(), outPoint.n, nHeight, nCurrentHeight);
|
|
bool removed = false;
|
|
CClaimValue claim;
|
|
nValidAtHeight = nHeight + getDelayForName(name);
|
|
std::string adjusted = adjustNameForValidHeight(name, nValidAtHeight);
|
|
|
|
if (removeClaimFromQueue(adjusted, outPoint, claim))
|
|
// assert(claim.nValidAtHeight == nValidAtHeight); probably better to leak than to crash
|
|
removed = true;
|
|
|
|
if (removed == false && removeClaimFromTrie(name, outPoint, claim, fCheckTakeover))
|
|
removed = true;
|
|
|
|
if (removed == true)
|
|
{
|
|
int expirationHeight = claim.nHeight + base->nExpirationTime;
|
|
removeFromExpirationQueue(adjusted, outPoint, expirationHeight);
|
|
claimsToDelete.insert(claim);
|
|
nValidAtHeight = claim.nValidAtHeight;
|
|
}
|
|
return removed;
|
|
}
|
|
|
|
void CClaimTrieCacheBase::addToExpirationQueue(int nExpirationHeight, nameOutPointType& entry) const
|
|
{
|
|
expirationQueueType::iterator itQueueRow = getExpirationQueueCacheRow(nExpirationHeight, true);
|
|
itQueueRow->second.push_back(entry);
|
|
}
|
|
|
|
void CClaimTrieCacheBase::removeFromExpirationQueue(const std::string& name, const COutPoint& outPoint, int expirationHeight) const
|
|
{
|
|
expirationQueueType::iterator itQueueRow = getExpirationQueueCacheRow(expirationHeight, false);
|
|
expirationQueueRowType::iterator itQueue;
|
|
if (itQueueRow != expirationQueueCache.end())
|
|
{
|
|
for (itQueue = itQueueRow->second.begin(); itQueue != itQueueRow->second.end(); ++itQueue)
|
|
{
|
|
if (outPoint == itQueue->outPoint && name == itQueue->name)
|
|
break;
|
|
}
|
|
|
|
if (itQueue != itQueueRow->second.end())
|
|
itQueueRow->second.erase(itQueue);
|
|
}
|
|
}
|
|
|
|
expirationQueueType::iterator CClaimTrieCacheBase::getExpirationQueueCacheRow(int nHeight, bool createIfNotExists) const
|
|
{
|
|
expirationQueueType::iterator itQueueRow = expirationQueueCache.find(nHeight);
|
|
if (itQueueRow == expirationQueueCache.end())
|
|
{
|
|
// Have to make a new row it put in the cache, if createIfNotExists is true
|
|
expirationQueueRowType queueRow;
|
|
// If the row exists in the base, copy its claims into the new row.
|
|
bool exists = base->getExpirationQueueRow(nHeight, queueRow);
|
|
if (!exists && !createIfNotExists)
|
|
return itQueueRow;
|
|
// Stick the new row in the cache
|
|
std::pair<expirationQueueType::iterator, bool> ret;
|
|
ret = expirationQueueCache.insert(std::pair<int, expirationQueueRowType >(nHeight, queueRow));
|
|
assert(ret.second);
|
|
itQueueRow = ret.first;
|
|
}
|
|
return itQueueRow;
|
|
}
|
|
|
|
// "name" is already normalized if needed
|
|
bool CClaimTrieCacheBase::reorderTrieNode(const std::string& name, bool fCheckTakeover) const
|
|
{
|
|
assert(base);
|
|
nodeCacheType::iterator cachedNode;
|
|
cachedNode = cache.find(name);
|
|
if (cachedNode == cache.end())
|
|
{
|
|
CClaimTrieNode* currentNode = getRoot();
|
|
for (std::string::const_iterator itCur = name.begin(); itCur != name.end(); ++itCur)
|
|
{
|
|
std::string sCurrentSubstring(name.begin(), itCur);
|
|
std::string sNextSubstring(name.begin(), itCur + 1);
|
|
|
|
cachedNode = cache.find(sNextSubstring);
|
|
if (cachedNode != cache.end())
|
|
{
|
|
currentNode = cachedNode->second;
|
|
continue;
|
|
}
|
|
nodeMapType::iterator childNode = currentNode->children.find(*itCur);
|
|
if (childNode != currentNode->children.end())
|
|
{
|
|
currentNode = childNode->second;
|
|
continue;
|
|
}
|
|
// The node doesn't exist, so it can't be reordered.
|
|
return true;
|
|
}
|
|
|
|
currentNode = new CClaimTrieNode(*currentNode);
|
|
std::pair<nodeCacheType::iterator, bool> ret;
|
|
ret = cache.insert(std::pair<std::string, CClaimTrieNode*>(name, currentNode));
|
|
assert(ret.second);
|
|
cachedNode = ret.first;
|
|
}
|
|
bool fChanged = false;
|
|
if (cachedNode->second->claims.empty())
|
|
{
|
|
// Nothing in there to reorder
|
|
return true;
|
|
}
|
|
else
|
|
{
|
|
CClaimValue currentTop = cachedNode->second->claims.front();
|
|
supportMapEntryType node;
|
|
getSupportsForName(name, node);
|
|
cachedNode->second->reorderClaims(node);
|
|
if (cachedNode->second->claims.front() != currentTop)
|
|
fChanged = true;
|
|
}
|
|
if (fChanged)
|
|
{
|
|
for (std::string::const_iterator itCur = name.begin(); itCur != name.end(); ++itCur)
|
|
{
|
|
std::string sub(name.begin(), itCur);
|
|
dirtyHashes.insert(sub);
|
|
}
|
|
dirtyHashes.insert(name);
|
|
if (fCheckTakeover)
|
|
namesToCheckForTakeover.insert(name);
|
|
}
|
|
return true;
|
|
}
|
|
|
|
// name has already been normalized if needed
|
|
bool CClaimTrieCacheBase::getSupportsForName(const std::string& name, supportMapEntryType& supports) const
|
|
{
|
|
const supportMapType::iterator cachedNode = supportCache.find(name);
|
|
if (cachedNode != supportCache.end())
|
|
{
|
|
supports = cachedNode->second;
|
|
return true;
|
|
}
|
|
|
|
return base->getSupportNode(name, supports);
|
|
}
|
|
|
|
bool CClaimTrieCacheBase::insertSupportIntoMap(const std::string& name, CSupportValue support, bool fCheckTakeover) const
|
|
{
|
|
supportMapType::iterator cachedNode;
|
|
// If this node is already in the cache, use that
|
|
cachedNode = supportCache.find(name);
|
|
// If not, copy the one from base if it exists, and use that
|
|
if (cachedNode == supportCache.end())
|
|
{
|
|
supportMapEntryType node;
|
|
base->getSupportNode(name, node);
|
|
std::pair<supportMapType::iterator, bool> ret;
|
|
ret = supportCache.insert(std::pair<std::string, supportMapEntryType>(name, node));
|
|
assert(ret.second);
|
|
cachedNode = ret.first;
|
|
}
|
|
cachedNode->second.push_back(support);
|
|
// See if this changed the biggest bid
|
|
return reorderTrieNode(name, fCheckTakeover);
|
|
}
|
|
|
|
bool CClaimTrieCacheBase::removeSupportFromMap(const std::string& name, const COutPoint& outPoint, CSupportValue& support, bool fCheckTakeover) const
|
|
{
|
|
supportMapType::iterator cachedNode;
|
|
cachedNode = supportCache.find(name);
|
|
if (cachedNode == supportCache.end())
|
|
{
|
|
supportMapEntryType node;
|
|
if (!base->getSupportNode(name, node)) // clearly, this support does not exist
|
|
return false;
|
|
|
|
std::pair<supportMapType::iterator, bool> ret;
|
|
ret = supportCache.insert(std::pair<std::string, supportMapEntryType>(name, node));
|
|
assert(ret.second);
|
|
cachedNode = ret.first;
|
|
}
|
|
supportMapEntryType::iterator itSupport;
|
|
for (itSupport = cachedNode->second.begin(); itSupport != cachedNode->second.end(); ++itSupport)
|
|
{
|
|
if (itSupport->outPoint == outPoint)
|
|
break;
|
|
}
|
|
if (itSupport != cachedNode->second.end())
|
|
{
|
|
std::swap(support, *itSupport);
|
|
cachedNode->second.erase(itSupport);
|
|
return reorderTrieNode(name, fCheckTakeover);
|
|
}
|
|
|
|
LogPrintf("CClaimTrieCacheBase::%s() : asked to remove a support that doesn't exist\n", __func__);
|
|
return false;
|
|
}
|
|
|
|
supportQueueType::iterator CClaimTrieCacheBase::getSupportQueueCacheRow(int nHeight, bool createIfNotExists) const
|
|
{
|
|
supportQueueType::iterator itQueueRow = supportQueueCache.find(nHeight);
|
|
if (itQueueRow == supportQueueCache.end())
|
|
{
|
|
supportQueueRowType queueRow;
|
|
bool exists = base->getSupportQueueRow(nHeight, queueRow);
|
|
if (!exists)
|
|
if (!createIfNotExists)
|
|
return itQueueRow;
|
|
// Stick the new row in the cache
|
|
std::pair<supportQueueType::iterator, bool> ret;
|
|
ret = supportQueueCache.insert(std::pair<int, supportQueueRowType >(nHeight, queueRow));
|
|
assert(ret.second);
|
|
itQueueRow = ret.first;
|
|
}
|
|
return itQueueRow;
|
|
}
|
|
|
|
queueNameType::iterator CClaimTrieCacheBase::getSupportQueueCacheNameRow(const std::string& name, bool createIfNotExists) const
|
|
{
|
|
queueNameType::iterator itQueueNameRow = supportQueueNameCache.find(name);
|
|
if (itQueueNameRow == supportQueueNameCache.end())
|
|
{
|
|
queueNameRowType queueNameRow;
|
|
bool exists = base->getSupportQueueNameRow(name, queueNameRow);
|
|
if (!exists && !createIfNotExists)
|
|
return itQueueNameRow;
|
|
// Stick the new row in the name cache
|
|
std::pair<queueNameType::iterator, bool> ret;
|
|
ret = supportQueueNameCache.insert(std::pair<std::string, queueNameRowType>(name, queueNameRow));
|
|
assert(ret.second);
|
|
itQueueNameRow = ret.first;
|
|
}
|
|
return itQueueNameRow;
|
|
}
|
|
|
|
bool CClaimTrieCacheBase::addSupportToQueues(const std::string& name, CSupportValue& support) const
|
|
{
|
|
LogPrintf("%s: nValidAtHeight: %d\n", __func__, support.nValidAtHeight);
|
|
supportQueueEntryType entry(name, support);
|
|
supportQueueType::iterator itQueueRow = getSupportQueueCacheRow(support.nValidAtHeight, true);
|
|
queueNameType::iterator itQueueNameRow = getSupportQueueCacheNameRow(name, true);
|
|
itQueueRow->second.push_back(entry);
|
|
itQueueNameRow->second.push_back(outPointHeightType(support.outPoint, support.nValidAtHeight));
|
|
nameOutPointType expireEntry(name, support.outPoint);
|
|
addSupportToExpirationQueue(support.nHeight + base->nExpirationTime, expireEntry);
|
|
return true;
|
|
}
|
|
|
|
bool CClaimTrieCacheBase::removeSupportFromQueue(const std::string& name, const COutPoint& outPoint, CSupportValue& support) const
|
|
{
|
|
queueNameType::iterator itQueueNameRow = getSupportQueueCacheNameRow(name, false);
|
|
if (itQueueNameRow == supportQueueNameCache.end())
|
|
return false;
|
|
|
|
queueNameRowType::iterator itQueueName;
|
|
for (itQueueName = itQueueNameRow->second.begin(); itQueueName != itQueueNameRow->second.end(); ++itQueueName)
|
|
{
|
|
if (itQueueName->outPoint == outPoint)
|
|
break;
|
|
}
|
|
if (itQueueName == itQueueNameRow->second.end())
|
|
return false;
|
|
|
|
supportQueueType::iterator itQueueRow = getSupportQueueCacheRow(itQueueName->nHeight, false);
|
|
if (itQueueRow != supportQueueCache.end())
|
|
{
|
|
supportQueueRowType::iterator itQueue;
|
|
for (itQueue = itQueueRow->second.begin(); itQueue != itQueueRow->second.end(); ++itQueue)
|
|
{
|
|
CSupportValue& support = itQueue->second;
|
|
if (support.outPoint == outPoint && name == itQueue->first)
|
|
break;
|
|
}
|
|
if (itQueue != itQueueRow->second.end())
|
|
{
|
|
std::swap(support, itQueue->second);
|
|
itQueueNameRow->second.erase(itQueueName);
|
|
itQueueRow->second.erase(itQueue);
|
|
return true;
|
|
}
|
|
}
|
|
LogPrintf("%s: An inconsistency was found in the claim queue. Please report this to the developers:\nFound in named support queue but not in height support queue: name: %s, txid: %s, nOut: %d, nValidAtHeight: %d, current height: %d\n", __func__, name, outPoint.hash.GetHex(), outPoint.n, itQueueName->nHeight, nCurrentHeight);
|
|
return false;
|
|
}
|
|
|
|
bool CClaimTrieCacheBase::addSupport(const std::string& name, const COutPoint& outPoint, CAmount nAmount, uint160 supportedClaimId, int nHeight) const
|
|
{
|
|
LogPrintf("%s: name: %s, txhash: %s, nOut: %d, nAmount: %d, supportedClaimId: %s, nHeight: %d, nCurrentHeight: %d\n", __func__, name, outPoint.hash.GetHex(), outPoint.n, nAmount, supportedClaimId.GetHex(), nHeight, nCurrentHeight);
|
|
assert(nHeight == nCurrentHeight);
|
|
|
|
int delayForSupport = getDelayForName(name, supportedClaimId);
|
|
CSupportValue support(outPoint, supportedClaimId, nAmount, nHeight, nHeight + delayForSupport);
|
|
return addSupportToQueues(name, support);
|
|
}
|
|
|
|
bool CClaimTrieCacheBase::undoSpendSupport(const std::string& name, const COutPoint& outPoint, uint160 supportedClaimId, CAmount nAmount, int nHeight, int nValidAtHeight) const
|
|
{
|
|
LogPrintf("%s: name: %s, txhash: %s, nOut: %d, nAmount: %d, supportedClaimId: %s, nHeight: %d, nCurrentHeight: %d\n", __func__, name, outPoint.hash.GetHex(), outPoint.n, nAmount, supportedClaimId.GetHex(), nHeight, nCurrentHeight);
|
|
CSupportValue support(outPoint, supportedClaimId, nAmount, nHeight, nValidAtHeight);
|
|
if (nValidAtHeight < nCurrentHeight)
|
|
{
|
|
nameOutPointType entry(adjustNameForValidHeight(name, nValidAtHeight), support.outPoint);
|
|
addSupportToExpirationQueue(support.nHeight + base->nExpirationTime, entry);
|
|
return insertSupportIntoMap(name, support, false);
|
|
}
|
|
|
|
return addSupportToQueues(name, support);
|
|
}
|
|
|
|
bool CClaimTrieCacheBase::removeSupport(const std::string& name, const COutPoint& outPoint, int nHeight, int& nValidAtHeight, bool fCheckTakeover) const
|
|
{
|
|
bool removed = false;
|
|
CSupportValue support;
|
|
nValidAtHeight = nHeight + getDelayForName(name);
|
|
std::string adjusted = adjustNameForValidHeight(name, nValidAtHeight);
|
|
|
|
if (removeSupportFromQueue(adjusted, outPoint, support))
|
|
removed = true;
|
|
if (removed == false && removeSupportFromMap(name, outPoint, support, fCheckTakeover))
|
|
removed = true;
|
|
if (removed)
|
|
{
|
|
int expirationHeight = nHeight + base->nExpirationTime;
|
|
removeSupportFromExpirationQueue(adjusted, outPoint, expirationHeight);
|
|
nValidAtHeight = support.nValidAtHeight; // might not actually match original computation if it got incorporated early
|
|
}
|
|
return removed;
|
|
}
|
|
|
|
void CClaimTrieCacheBase::addSupportToExpirationQueue(int nExpirationHeight, nameOutPointType& entry) const
|
|
{
|
|
expirationQueueType::iterator itQueueRow = getSupportExpirationQueueCacheRow(nExpirationHeight, true);
|
|
itQueueRow->second.push_back(entry);
|
|
}
|
|
|
|
void CClaimTrieCacheBase::removeSupportFromExpirationQueue(const std::string& name, const COutPoint& outPoint, int expirationHeight) const
|
|
{
|
|
expirationQueueType::iterator itQueueRow = getSupportExpirationQueueCacheRow(expirationHeight, false);
|
|
expirationQueueRowType::iterator itQueue;
|
|
if (itQueueRow != supportExpirationQueueCache.end())
|
|
{
|
|
for (itQueue = itQueueRow->second.begin(); itQueue != itQueueRow->second.end(); ++itQueue)
|
|
{
|
|
if (outPoint == itQueue->outPoint && name == itQueue->name)
|
|
break;
|
|
}
|
|
}
|
|
if (itQueue != itQueueRow->second.end())
|
|
itQueueRow->second.erase(itQueue);
|
|
}
|
|
|
|
expirationQueueType::iterator CClaimTrieCacheBase::getSupportExpirationQueueCacheRow(int nHeight, bool createIfNotExists) const
|
|
{
|
|
expirationQueueType::iterator itQueueRow = supportExpirationQueueCache.find(nHeight);
|
|
if (itQueueRow == supportExpirationQueueCache.end())
|
|
{
|
|
// Have to make a new row it put in the cache, if createIfNotExists is true
|
|
expirationQueueRowType queueRow;
|
|
// If the row exists in the base, copy its claims into the new row.
|
|
bool exists = base->getSupportExpirationQueueRow(nHeight, queueRow);
|
|
if (!exists && !createIfNotExists)
|
|
return itQueueRow;
|
|
// Stick the new row in the cache
|
|
std::pair<expirationQueueType::iterator, bool> ret;
|
|
ret = supportExpirationQueueCache.insert(std::pair<int, expirationQueueRowType >(nHeight, queueRow));
|
|
assert(ret.second);
|
|
itQueueRow = ret.first;
|
|
}
|
|
return itQueueRow;
|
|
}
|
|
|
|
bool CClaimTrieCacheBase::undoAddSupport(const std::string& name, const COutPoint& outPoint, int nHeight) const
|
|
{
|
|
LogPrintf("%s: name: %s, txhash: %s, nOut: %d, nHeight: %d, nCurrentHeight: %d\n", __func__, name, outPoint.hash.GetHex(), outPoint.n, nHeight, nCurrentHeight);
|
|
int throwaway;
|
|
return removeSupport(name, outPoint, nHeight, throwaway, false);
|
|
}
|
|
|
|
bool CClaimTrieCacheBase::spendSupport(const std::string& name, const COutPoint& outPoint, int nHeight, int& nValidAtHeight) const
|
|
{
|
|
LogPrintf("%s: name: %s, txhash: %s, nOut: %d, nHeight: %d, nCurrentHeight: %d\n", __func__, name, outPoint.hash.GetHex(), outPoint.n, nHeight, nCurrentHeight);
|
|
return removeSupport(name, outPoint, nHeight, nValidAtHeight, true);
|
|
}
|
|
|
|
bool CClaimTrieCacheBase::incrementBlock(insertUndoType& insertUndo, claimQueueRowType& expireUndo,
|
|
insertUndoType& insertSupportUndo, supportQueueRowType& expireSupportUndo,
|
|
std::vector<std::pair<std::string, int> >& takeoverHeightUndo)
|
|
{
|
|
// we don't actually modify the claimTrie here; that happens in flush
|
|
claimQueueType::iterator itQueueRow = getQueueCacheRow(nCurrentHeight, false);
|
|
if (itQueueRow != claimQueueCache.end())
|
|
{
|
|
// for each claim activating right now
|
|
for (claimQueueRowType::iterator itEntry = itQueueRow->second.begin(); itEntry != itQueueRow->second.end(); ++itEntry)
|
|
{
|
|
bool found = false;
|
|
queueNameType::iterator itQueueNameRow = getQueueCacheNameRow(itEntry->first, false);
|
|
if (itQueueNameRow != claimQueueNameCache.end())
|
|
{
|
|
for (queueNameRowType::iterator itQueueName = itQueueNameRow->second.begin(); itQueueName != itQueueNameRow->second.end(); ++itQueueName)
|
|
{
|
|
if (itQueueName->outPoint == itEntry->second.outPoint && itQueueName->nHeight == nCurrentHeight)
|
|
{
|
|
found = true;
|
|
itQueueNameRow->second.erase(itQueueName);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
if (!found)
|
|
{
|
|
LogPrintf("%s: An inconsistency was found in the claim queue. Please report this to the developers:\nFound in height queue but not in named queue: name: %s, txid: %s, nOut: %d, nValidAtHeight: %d, current height: %d\n", __func__, itEntry->first, itEntry->second.outPoint.hash.GetHex(), itEntry->second.outPoint.n, itEntry->second.nValidAtHeight, nCurrentHeight);
|
|
if (itQueueNameRow != claimQueueNameCache.end())
|
|
{
|
|
LogPrintf("Claims found for that name:\n");
|
|
for (queueNameRowType::iterator itQueueName = itQueueNameRow->second.begin(); itQueueName != itQueueNameRow->second.end(); ++itQueueName)
|
|
{
|
|
LogPrintf("\ttxid: %s, nOut: %d, nValidAtHeight: %d\n", itQueueName->outPoint.hash.GetHex(), itQueueName->outPoint.n, itQueueName->nHeight);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
LogPrintf("No claims found for that name\n");
|
|
}
|
|
}
|
|
assert(found);
|
|
|
|
// error triggered by claimtriecache_normalization:
|
|
// it will cause a double remove when rolling back -- something we don't handle correctly
|
|
// (same problem exists for the insertSupportUndo below)
|
|
//for (insertUndoType::reverse_iterator itInsertUndo = insertUndo.rbegin(); itInsertUndo != insertUndo.rend(); ++itInsertUndo)
|
|
// if (itInsertUndo->name == itEntry->first && itInsertUndo->outPoint == itEntry->second.outPoint)
|
|
// throw std::runtime_error("not expected");
|
|
|
|
insertClaimIntoTrie(itEntry->first, itEntry->second, true);
|
|
insertUndo.push_back(nameOutPointHeightType(itEntry->first, itEntry->second.outPoint, itEntry->second.nValidAtHeight));
|
|
}
|
|
itQueueRow->second.clear();
|
|
}
|
|
expirationQueueType::iterator itExpirationRow = getExpirationQueueCacheRow(nCurrentHeight, false);
|
|
if (itExpirationRow != expirationQueueCache.end())
|
|
{
|
|
// for every claim expiring right now
|
|
for (expirationQueueRowType::iterator itEntry = itExpirationRow->second.begin(); itEntry != itExpirationRow->second.end(); ++itEntry)
|
|
{
|
|
CClaimValue claim;
|
|
assert(removeClaimFromTrie(itEntry->name, itEntry->outPoint, claim, true));
|
|
claimsToDelete.insert(claim);
|
|
expireUndo.push_back(std::make_pair(itEntry->name, claim));
|
|
LogPrintf("Expiring claim %s: %s, nHeight: %d, nValidAtHeight: %d\n",
|
|
claim.claimId.GetHex(), itEntry->name, claim.nHeight, claim.nValidAtHeight);
|
|
}
|
|
itExpirationRow->second.clear();
|
|
}
|
|
supportQueueType::iterator itSupportRow = getSupportQueueCacheRow(nCurrentHeight, false);
|
|
if (itSupportRow != supportQueueCache.end())
|
|
{
|
|
for (supportQueueRowType::iterator itSupport = itSupportRow->second.begin(); itSupport != itSupportRow->second.end(); ++itSupport)
|
|
{
|
|
bool found = false;
|
|
queueNameType::iterator itSupportNameRow = getSupportQueueCacheNameRow(itSupport->first, false);
|
|
if (itSupportNameRow != supportQueueNameCache.end())
|
|
{
|
|
for (queueNameRowType::iterator itSupportName = itSupportNameRow->second.begin(); itSupportName != itSupportNameRow->second.end(); ++itSupportName)
|
|
{
|
|
if (itSupportName->outPoint == itSupport->second.outPoint && itSupportName->nHeight == itSupport->second.nValidAtHeight)
|
|
{
|
|
found = true;
|
|
itSupportNameRow->second.erase(itSupportName);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
if (!found)
|
|
{
|
|
LogPrintf("%s: An inconsistency was found in the support queue. Please report this to the developers:\nFound in height queue but not in named queue: %s, txid: %s, nOut: %d, nValidAtHeight: %d, current height: %d\n", __func__, itSupport->first, itSupport->second.outPoint.hash.GetHex(), itSupport->second.outPoint.n, itSupport->second.nValidAtHeight, nCurrentHeight);
|
|
if (itSupportNameRow != supportQueueNameCache.end())
|
|
{
|
|
LogPrintf("Supports found for that name:\n");
|
|
for (queueNameRowType::iterator itSupportName = itSupportNameRow->second.begin(); itSupportName != itSupportNameRow->second.end(); ++itSupportName)
|
|
{
|
|
LogPrintf("\ttxid: %s, nOut: %d, nValidAtHeight: %d\n", itSupportName->outPoint.hash.GetHex(), itSupportName->outPoint.n, itSupportName->nHeight);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
LogPrintf("No support found for that name\n");
|
|
}
|
|
}
|
|
insertSupportIntoMap(itSupport->first, itSupport->second, true);
|
|
insertSupportUndo.push_back(nameOutPointHeightType(itSupport->first, itSupport->second.outPoint, itSupport->second.nValidAtHeight));
|
|
}
|
|
itSupportRow->second.clear();
|
|
}
|
|
expirationQueueType::iterator itSupportExpirationRow = getSupportExpirationQueueCacheRow(nCurrentHeight, false);
|
|
if (itSupportExpirationRow != supportExpirationQueueCache.end())
|
|
{
|
|
for (expirationQueueRowType::iterator itEntry = itSupportExpirationRow->second.begin(); itEntry != itSupportExpirationRow->second.end(); ++itEntry)
|
|
{
|
|
CSupportValue support;
|
|
assert(removeSupportFromMap(itEntry->name, itEntry->outPoint, support, true));
|
|
expireSupportUndo.push_back(std::make_pair(itEntry->name, support));
|
|
LogPrintf("Expiring support %s: %s, nHeight: %d, nValidAtHeight: %d\n",
|
|
support.supportedClaimId.GetHex(), itEntry->name, support.nHeight, support.nValidAtHeight);
|
|
}
|
|
itSupportExpirationRow->second.clear();
|
|
}
|
|
|
|
checkNamesForTakeover(insertUndo, insertSupportUndo, takeoverHeightUndo);
|
|
|
|
for (nodeCacheType::const_iterator itOriginals = block_originals.begin(); itOriginals != block_originals.end(); ++itOriginals)
|
|
delete itOriginals->second;
|
|
block_originals.clear();
|
|
|
|
for (nodeCacheType::const_iterator itCache = cache.begin(); itCache != cache.end(); ++itCache)
|
|
block_originals[itCache->first] = new CClaimTrieNode(*itCache->second);
|
|
|
|
nCurrentHeight++;
|
|
|
|
return true;
|
|
}
|
|
|
|
void CClaimTrieCacheBase::checkNamesForTakeover(insertUndoType& insertUndo, insertUndoType& insertSupportUndo,
|
|
std::vector<std::pair<std::string, int> >& takeoverHeightUndo) const {
|
|
// check each potentially taken over name to see if a takeover occurred.
|
|
// if it did, then check the claim and support insertion queues for
|
|
// the names that have been taken over, immediately insert all claim and
|
|
// supports for those names, and stick them in the insertUndo or
|
|
// insertSupportUndo vectors, with the nValidAtHeight they had prior to
|
|
// this block.
|
|
// Run through all names that have been taken over
|
|
for (std::set<std::string>::iterator itNamesToCheck = namesToCheckForTakeover.begin(); itNamesToCheck != namesToCheckForTakeover.end(); ++itNamesToCheck)
|
|
{
|
|
const std::string& nameToCheck = *itNamesToCheck;
|
|
// Check if a takeover has occurred
|
|
nodeCacheType::iterator itCachedNode = cache.find(nameToCheck);
|
|
// many possibilities
|
|
// if this node is new, don't put it into the undo -- there will be nothing to restore, after all
|
|
// if all of this node's claims were deleted, it should be put into the undo -- there could be
|
|
// claims in the queue for that name and the takeover height should be the current height
|
|
// if the node is not in the cache, or getbestclaim fails, that means all of its claims were
|
|
// deleted
|
|
// if getOriginalInfoForName returns false, that means it's new and shouldn't go into the undo
|
|
// if both exist, and the current best claim is not the same as or the parent to the new best
|
|
// claim, then ownership has changed and the current height of last takeover should go into
|
|
// the queue
|
|
CClaimValue claimInCache;
|
|
CClaimValue claimInTrie;
|
|
bool haveClaimInCache;
|
|
bool haveClaimInTrie;
|
|
if (itCachedNode == cache.end())
|
|
{
|
|
haveClaimInCache = false;
|
|
}
|
|
else
|
|
{
|
|
haveClaimInCache = itCachedNode->second->getBestClaim(claimInCache);
|
|
}
|
|
haveClaimInTrie = getOriginalInfoForName(nameToCheck, claimInTrie);
|
|
bool takeoverHappened = false;
|
|
if (!haveClaimInTrie)
|
|
{
|
|
takeoverHappened = true;
|
|
}
|
|
else if (!haveClaimInCache)
|
|
{
|
|
takeoverHappened = true;
|
|
}
|
|
else if (claimInCache != claimInTrie)
|
|
{
|
|
if (claimInCache.claimId != claimInTrie.claimId)
|
|
{
|
|
takeoverHappened = true;
|
|
}
|
|
}
|
|
if (takeoverHappened)
|
|
{
|
|
// Get all pending claims for that name and activate them all in the case that our winner is defunct.
|
|
queueNameType::iterator itQueueNameRow = getQueueCacheNameRow(nameToCheck, false);
|
|
if (itQueueNameRow != claimQueueNameCache.end())
|
|
{
|
|
// for each of those claims
|
|
for (queueNameRowType::iterator itQueueName = itQueueNameRow->second.begin(); itQueueName != itQueueNameRow->second.end(); ++itQueueName)
|
|
{
|
|
bool found = false;
|
|
// find the matching claim data from the other cache (indexed by height)
|
|
claimQueueType::iterator itQueueRow = getQueueCacheRow(itQueueName->nHeight, false);
|
|
claimQueueRowType::iterator itQueue;
|
|
if (itQueueRow != claimQueueCache.end())
|
|
{
|
|
for (itQueue = itQueueRow->second.begin(); itQueue != itQueueRow->second.end(); ++itQueue)
|
|
{
|
|
if (nameToCheck == itQueue->first && itQueue->second.outPoint == itQueueName->outPoint && itQueue->second.nValidAtHeight == itQueueName->nHeight) {
|
|
found = true;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
if (found)
|
|
{
|
|
// Insert them into the queue undo with their previous nValidAtHeight
|
|
insertUndo.push_back(nameOutPointHeightType(itQueue->first, itQueue->second.outPoint, itQueue->second.nValidAtHeight));
|
|
// Insert them into the name trie with the new nValidAtHeight
|
|
itQueue->second.nValidAtHeight = nCurrentHeight;
|
|
insertClaimIntoTrie(itQueue->first, itQueue->second, false);
|
|
// Delete them from the height-based queue
|
|
itQueueRow->second.erase(itQueue);
|
|
}
|
|
else
|
|
{
|
|
LogPrintf("%s(): An inconsistency was found in the claim queue. Please report this to the developers:\nClaim found in name queue but not in height based queue:\nname: %s, txid: %s, nOut: %d, nValidAtHeight in name based queue: %d, current height: %d\n", __func__, nameToCheck, itQueueName->outPoint.hash.GetHex(), itQueueName->outPoint.n, itQueueName->nHeight, nCurrentHeight);
|
|
}
|
|
assert(found); // if this fails, you may have ended up with a duplicate in itQueueNameRow->second
|
|
}
|
|
// remove all claims from the queue for that name
|
|
itQueueNameRow->second.clear();
|
|
}
|
|
//
|
|
// Then, get all supports in the queue for that name
|
|
queueNameType::iterator itSupportQueueNameRow = getSupportQueueCacheNameRow(nameToCheck, false);
|
|
if (itSupportQueueNameRow != supportQueueNameCache.end())
|
|
{
|
|
for (queueNameRowType::iterator itSupportQueueName = itSupportQueueNameRow->second.begin(); itSupportQueueName != itSupportQueueNameRow->second.end(); ++itSupportQueueName)
|
|
{
|
|
// Pull those supports out of the height-based queue
|
|
supportQueueType::iterator itSupportQueueRow = getSupportQueueCacheRow(itSupportQueueName->nHeight, false);
|
|
if (itSupportQueueRow != supportQueueCache.end())
|
|
{
|
|
supportQueueRowType::iterator itSupportQueue;
|
|
for (itSupportQueue = itSupportQueueRow->second.begin(); itSupportQueue != itSupportQueueRow->second.end(); ++itSupportQueue)
|
|
{
|
|
if (nameToCheck == itSupportQueue->first && itSupportQueue->second.outPoint == itSupportQueueName->outPoint && itSupportQueue->second.nValidAtHeight == itSupportQueueName->nHeight) {
|
|
break;
|
|
}
|
|
}
|
|
if (itSupportQueue != itSupportQueueRow->second.end())
|
|
{
|
|
// Insert them into the support queue undo with the previous nValidAtHeight
|
|
insertSupportUndo.push_back(nameOutPointHeightType(itSupportQueue->first, itSupportQueue->second.outPoint, itSupportQueue->second.nValidAtHeight));
|
|
// Insert them into the support map with the new nValidAtHeight
|
|
itSupportQueue->second.nValidAtHeight = nCurrentHeight;
|
|
insertSupportIntoMap(itSupportQueue->first, itSupportQueue->second, false);
|
|
// Delete them from the height-based queue
|
|
itSupportQueueRow->second.erase(itSupportQueue);
|
|
}
|
|
else
|
|
{
|
|
// here be problems TODO: show error, assert false
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// here be problems
|
|
}
|
|
}
|
|
// remove all supports from the queue for that name
|
|
itSupportQueueNameRow->second.clear();
|
|
}
|
|
|
|
// save the old last height so that it can be restored if the block is undone
|
|
if (haveClaimInTrie)
|
|
{
|
|
int nHeightOfLastTakeover;
|
|
assert(getLastTakeoverForName(nameToCheck, nHeightOfLastTakeover));
|
|
takeoverHeightUndo.push_back(std::make_pair(nameToCheck, nHeightOfLastTakeover));
|
|
}
|
|
itCachedNode = cache.find(nameToCheck);
|
|
if (itCachedNode != cache.end())
|
|
{
|
|
cacheTakeoverHeights[nameToCheck] = nCurrentHeight;
|
|
}
|
|
}
|
|
}
|
|
namesToCheckForTakeover.clear();
|
|
}
|
|
|
|
bool CClaimTrieCacheBase::decrementBlock(insertUndoType& insertUndo, claimQueueRowType& expireUndo,
|
|
insertUndoType& insertSupportUndo, supportQueueRowType& expireSupportUndo,
|
|
std::vector<std::pair<std::string, int> >& takeoverHeightUndo)
|
|
{
|
|
nCurrentHeight--;
|
|
|
|
if (expireSupportUndo.begin() != expireSupportUndo.end())
|
|
{
|
|
expirationQueueType::iterator itSupportExpireRow = getSupportExpirationQueueCacheRow(nCurrentHeight, true);
|
|
for (supportQueueRowType::reverse_iterator itSupportExpireUndo = expireSupportUndo.rbegin(); itSupportExpireUndo != expireSupportUndo.rend(); ++itSupportExpireUndo)
|
|
{
|
|
insertSupportIntoMap(itSupportExpireUndo->first, itSupportExpireUndo->second, false);
|
|
if (nCurrentHeight == itSupportExpireUndo->second.nHeight + base->nExpirationTime)
|
|
itSupportExpireRow->second.push_back(nameOutPointType(itSupportExpireUndo->first, itSupportExpireUndo->second.outPoint));
|
|
}
|
|
}
|
|
|
|
for (insertUndoType::reverse_iterator itSupportUndo = insertSupportUndo.rbegin(); itSupportUndo != insertSupportUndo.rend(); ++itSupportUndo)
|
|
{
|
|
CSupportValue support;
|
|
assert(removeSupportFromMap(itSupportUndo->name, itSupportUndo->outPoint, support, false));
|
|
if (itSupportUndo->nHeight >= 0)
|
|
{
|
|
// support.nValidHeight may have been changed if this was inserted before activation height
|
|
// due to a triggered takeover, change it back to original nValidAtHeight
|
|
support.nValidAtHeight = itSupportUndo->nHeight;
|
|
supportQueueType::iterator itSupportRow = getSupportQueueCacheRow(itSupportUndo->nHeight, true);
|
|
queueNameType::iterator itSupportNameRow = getSupportQueueCacheNameRow(itSupportUndo->name, true);
|
|
itSupportRow->second.push_back(std::make_pair(itSupportUndo->name, support));
|
|
itSupportNameRow->second.push_back(outPointHeightType(support.outPoint, support.nValidAtHeight));
|
|
}
|
|
}
|
|
|
|
if (!expireUndo.empty())
|
|
{
|
|
expirationQueueType::iterator itExpireRow = getExpirationQueueCacheRow(nCurrentHeight, true);
|
|
for (claimQueueRowType::reverse_iterator itExpireUndo = expireUndo.rbegin(); itExpireUndo != expireUndo.rend(); ++itExpireUndo)
|
|
{
|
|
insertClaimIntoTrie(itExpireUndo->first, itExpireUndo->second, false);
|
|
CClaimIndexElement element = { itExpireUndo->first, itExpireUndo->second };
|
|
claimsToAdd.push_back(element);
|
|
// Check if it expired at this height rather than being a rename/normalization
|
|
if (nCurrentHeight == itExpireUndo->second.nHeight + base->nExpirationTime)
|
|
itExpireRow->second.push_back(nameOutPointType(itExpireUndo->first, itExpireUndo->second.outPoint));
|
|
}
|
|
}
|
|
|
|
for (insertUndoType::reverse_iterator itInsertUndo = insertUndo.rbegin(); itInsertUndo != insertUndo.rend(); ++itInsertUndo)
|
|
{
|
|
CClaimValue claim;
|
|
assert(removeClaimFromTrie(itInsertUndo->name, itInsertUndo->outPoint, claim, false));
|
|
if (itInsertUndo->nHeight >= 0) // aka it became valid at this height rather than being a rename/normalization
|
|
{
|
|
// valid height may have been changed if this was inserted because the winning claim was abandoned; reset it here:
|
|
claim.nValidAtHeight = itInsertUndo->nHeight;
|
|
claimQueueType::iterator itQueueRow = getQueueCacheRow(itInsertUndo->nHeight, true);
|
|
itQueueRow->second.push_back(std::make_pair(itInsertUndo->name, claim));
|
|
queueNameType::iterator itQueueNameRow = getQueueCacheNameRow(itInsertUndo->name, true);
|
|
itQueueNameRow->second.push_back(outPointHeightType(itInsertUndo->outPoint, claim.nValidAtHeight));
|
|
}
|
|
else
|
|
{
|
|
// no present way to delete claim from the index by name (but we read after the deletion anyway)
|
|
claimsToDelete.insert(claim);
|
|
}
|
|
}
|
|
|
|
for (std::vector<std::pair<std::string, int> >::reverse_iterator itTakeoverHeightUndo = takeoverHeightUndo.rbegin(); itTakeoverHeightUndo != takeoverHeightUndo.rend(); ++itTakeoverHeightUndo)
|
|
{
|
|
cacheTakeoverHeights[itTakeoverHeightUndo->first] = itTakeoverHeightUndo->second;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
bool CClaimTrieCacheBase::finalizeDecrement() const
|
|
{
|
|
for (nodeCacheType::iterator itOriginals = block_originals.begin(); itOriginals != block_originals.end(); ++itOriginals)
|
|
delete itOriginals->second;
|
|
block_originals.clear();
|
|
|
|
for (nodeCacheType::const_iterator itCache = cache.begin(); itCache != cache.end(); ++itCache)
|
|
block_originals[itCache->first] = new CClaimTrieNode(*itCache->second);
|
|
|
|
return true;
|
|
}
|
|
|
|
bool CClaimTrieCacheBase::getLastTakeoverForName(const std::string& name, int& nLastTakeoverForName) const
|
|
{
|
|
if (!fRequireTakeoverHeights)
|
|
{
|
|
nLastTakeoverForName = 0;
|
|
return true;
|
|
}
|
|
std::map<std::string, int>::iterator itHeights = cacheTakeoverHeights.find(name);
|
|
if (itHeights == cacheTakeoverHeights.end())
|
|
{
|
|
return base->getLastTakeoverForName(name, nLastTakeoverForName);
|
|
}
|
|
nLastTakeoverForName = itHeights->second;
|
|
return true;
|
|
}
|
|
|
|
int CClaimTrieCacheBase::getNumBlocksOfContinuousOwnership(const std::string& name) const
|
|
{
|
|
const CClaimTrieNode* node = getNodeForName(name);
|
|
if (!node || node->claims.empty())
|
|
return 0;
|
|
int nLastTakeoverHeight;
|
|
assert(getLastTakeoverForName(name, nLastTakeoverHeight));
|
|
return nCurrentHeight - nLastTakeoverHeight;
|
|
}
|
|
|
|
const CClaimTrieNode* CClaimTrieCacheBase::getNodeForName(const std::string& name) const
|
|
{
|
|
const CClaimTrieNode* node = nullptr;
|
|
nodeCacheType::const_iterator itCache = cache.find(name);
|
|
if (itCache != cache.end()) {
|
|
node = itCache->second;
|
|
}
|
|
if (!node) {
|
|
node = base->getNodeForName(name);
|
|
}
|
|
return node;
|
|
}
|
|
|
|
// "name" is already normalized if needed
|
|
int CClaimTrieCacheBase::getDelayForName(const std::string& name) const
|
|
{
|
|
if (!fRequireTakeoverHeights)
|
|
return 0;
|
|
|
|
int nBlocksOfContinuousOwnership = getNumBlocksOfContinuousOwnership(name);
|
|
return std::min(nBlocksOfContinuousOwnership / base->nProportionalDelayFactor, 4032);
|
|
}
|
|
|
|
int CClaimTrieCacheBase::getDelayForName(const std::string& name, const uint160& claimId) const
|
|
{
|
|
CClaimValue currentClaim;
|
|
int delayForClaim;
|
|
if (getOriginalInfoForName(name, currentClaim) && currentClaim.claimId == claimId)
|
|
delayForClaim = 0;
|
|
else
|
|
delayForClaim = getDelayForName(name);
|
|
return delayForClaim;
|
|
}
|
|
|
|
uint256 CClaimTrieCacheBase::getBestBlock()
|
|
{
|
|
if (hashBlock.IsNull() && base != nullptr)
|
|
hashBlock = base->hashBlock;
|
|
return hashBlock;
|
|
}
|
|
|
|
void CClaimTrieCacheBase::setBestBlock(const uint256& hashBlockIn)
|
|
{
|
|
hashBlock = hashBlockIn;
|
|
}
|
|
|
|
bool CClaimTrieCacheBase::clear() const
|
|
{
|
|
for (nodeCacheType::iterator itcache = cache.begin(); itcache != cache.end(); ++itcache)
|
|
delete itcache->second;
|
|
cache.clear();
|
|
|
|
for (nodeCacheType::iterator itOriginals = block_originals.begin(); itOriginals != block_originals.end(); ++itOriginals)
|
|
delete itOriginals->second;
|
|
block_originals.clear();
|
|
|
|
dirtyHashes.clear();
|
|
cacheHashes.clear();
|
|
claimQueueCache.clear();
|
|
claimQueueNameCache.clear();
|
|
expirationQueueCache.clear();
|
|
supportCache.clear();
|
|
supportQueueCache.clear();
|
|
supportQueueNameCache.clear();
|
|
supportExpirationQueueCache.clear();
|
|
namesToCheckForTakeover.clear();
|
|
cacheTakeoverHeights.clear();
|
|
return true;
|
|
}
|
|
|
|
bool CClaimTrieCacheBase::flush()
|
|
{
|
|
if (dirty())
|
|
getMerkleHash();
|
|
|
|
if (!claimsToDelete.empty()) {
|
|
for (claimIndexClaimListType::iterator it = claimsToDelete.begin(); it != claimsToDelete.end(); ++it)
|
|
base->removeFromClaimIndex(*it);
|
|
claimsToDelete.clear();
|
|
}
|
|
if (!claimsToAdd.empty()) {
|
|
for (claimIndexElementListType::iterator it = claimsToAdd.begin(); it != claimsToAdd.end(); ++it)
|
|
base->addToClaimIndex(it->name, it->claim);
|
|
claimsToAdd.clear();
|
|
}
|
|
|
|
bool success = base->update(cache, cacheHashes, cacheTakeoverHeights, getBestBlock(), claimQueueCache, claimQueueNameCache, expirationQueueCache, nCurrentHeight, supportCache, supportQueueCache, supportQueueNameCache, supportExpirationQueueCache);
|
|
return (success ? clear() : success);
|
|
}
|
|
|
|
uint256 CClaimTrieCacheBase::getLeafHashForProof(const std::string& currentPosition, const CClaimTrieNode* currentNode) const
|
|
{
|
|
hashMapType::iterator cachedHash = cacheHashes.find(currentPosition);
|
|
return cachedHash == cacheHashes.end() ? currentNode->hash : cachedHash->second;
|
|
}
|
|
|
|
void CClaimTrieCacheBase::recursiveIterateTrie(std::string& name, const CClaimTrieNode* current, CNodeCallback& callback) const
|
|
{
|
|
callback.visit(name, current);
|
|
|
|
nodeCacheType::const_iterator cachedNode;
|
|
for (nodeMapType::const_iterator it = current->children.begin(); it != current->children.end(); ++it) {
|
|
name.push_back(it->first);
|
|
cachedNode = cache.find(name);
|
|
if (cachedNode != cache.end())
|
|
recursiveIterateTrie(name, cachedNode->second, callback);
|
|
else
|
|
recursiveIterateTrie(name, it->second, callback);
|
|
name.erase(name.end() - 1);
|
|
}
|
|
}
|
|
|
|
bool CClaimTrieCacheBase::iterateTrie(CNodeCallback& callback) const
|
|
{
|
|
try
|
|
{
|
|
std::string name;
|
|
recursiveIterateTrie(name, getRoot(), callback);
|
|
assert(name.empty());
|
|
}
|
|
catch (const CNodeCallback::CRecursionInterruptionException& ex)
|
|
{
|
|
return ex.success;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
claimsForNameType CClaimTrieCacheBase::getClaimsForName(const std::string& name) const
|
|
{
|
|
int nLastTakeoverHeight = 0;
|
|
std::vector<CClaimValue> claims;
|
|
if (const CClaimTrieNode* node = getNodeForName(name))
|
|
{
|
|
claims = node->claims;
|
|
getLastTakeoverForName(name, nLastTakeoverHeight);
|
|
}
|
|
supportMapEntryType supports;
|
|
getSupportsForName(name, supports);
|
|
return claimsForNameType(claims, supports, nLastTakeoverHeight, name);
|
|
}
|
|
|
|
CAmount CClaimTrieCacheBase::getEffectiveAmountForClaim(const std::string& name, const uint160& claimId, std::vector<CSupportValue>* supports) const
|
|
{
|
|
return getEffectiveAmountForClaim(getClaimsForName(name), claimId, supports);
|
|
}
|
|
|
|
CAmount CClaimTrieCacheBase::getEffectiveAmountForClaim(const claimsForNameType& claims, const uint160& claimId, std::vector<CSupportValue>* supports) const
|
|
{
|
|
CAmount effectiveAmount = 0;
|
|
for (std::vector<CClaimValue>::const_iterator it = claims.claims.begin(); it != claims.claims.end(); ++it)
|
|
{
|
|
if (it->claimId == claimId && it->nValidAtHeight < nCurrentHeight)
|
|
{
|
|
effectiveAmount += it->nAmount;
|
|
for (std::vector<CSupportValue>::const_iterator it = claims.supports.begin(); it != claims.supports.end(); ++it)
|
|
{
|
|
if (it->supportedClaimId == claimId && it->nValidAtHeight < nCurrentHeight)
|
|
{
|
|
effectiveAmount += it->nAmount;
|
|
if (supports)
|
|
supports->push_back(*it);
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
return effectiveAmount;
|
|
}
|
|
|
|
bool CClaimTrieCacheBase::getInfoForName(const std::string& name, CClaimValue& claim) const
|
|
{
|
|
if (dirty())
|
|
getMerkleHash();
|
|
|
|
CClaimTrieNode* current = getRoot();
|
|
nodeCacheType::const_iterator cachedNode;
|
|
|
|
for (std::string::const_iterator itName = name.begin(); current; ++itName)
|
|
{
|
|
std::string currentPosition(name.begin(), itName);
|
|
cachedNode = cache.find(currentPosition);
|
|
if (cachedNode != cache.end())
|
|
current = cachedNode->second;
|
|
|
|
if (itName == name.end())
|
|
return current->getBestClaim(claim);
|
|
|
|
nodeMapType::const_iterator itChildren = current->children.find(*itName);
|
|
if (itChildren != current->children.end())
|
|
current = itChildren->second;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
bool CClaimTrieCacheBase::getProofForName(const std::string& name, CClaimTrieProof& proof) const
|
|
{
|
|
if (dirty())
|
|
getMerkleHash();
|
|
|
|
std::vector<CClaimTrieProofNode> nodes;
|
|
CClaimTrieNode* current = getRoot();
|
|
nodeCacheType::const_iterator cachedNode;
|
|
bool fNameHasValue = false;
|
|
COutPoint outPoint;
|
|
int nHeightOfLastTakeover = 0;
|
|
for (std::string::const_iterator itName = name.begin(); current; ++itName)
|
|
{
|
|
std::string currentPosition(name.begin(), itName);
|
|
cachedNode = cache.find(currentPosition);
|
|
if (cachedNode != cache.end())
|
|
current = cachedNode->second;
|
|
CClaimValue claim;
|
|
bool fNodeHasValue = current->getBestClaim(claim);
|
|
uint256 valueHash;
|
|
if (fNodeHasValue)
|
|
{
|
|
int nHeightOfLastTakeover;
|
|
if (!getLastTakeoverForName(currentPosition, nHeightOfLastTakeover))
|
|
return false;
|
|
valueHash = getValueHash(claim.outPoint, nHeightOfLastTakeover);
|
|
}
|
|
std::vector<std::pair<unsigned char, uint256> > children;
|
|
CClaimTrieNode* nextCurrent = nullptr;
|
|
for (nodeMapType::const_iterator itChildren = current->children.begin(); itChildren != current->children.end(); ++itChildren)
|
|
{
|
|
std::stringstream ss;
|
|
ss << itChildren->first;
|
|
const std::string& curStr = ss.str();
|
|
if (itName == name.end() || curStr[0] != *itName) // Leaf node
|
|
{
|
|
uint256 childHash = getLeafHashForProof(currentPosition + curStr, itChildren->second);
|
|
children.push_back(std::make_pair(curStr[0], childHash));
|
|
}
|
|
else // Full node
|
|
{
|
|
nextCurrent = itChildren->second;
|
|
uint256 childHash;
|
|
children.push_back(std::make_pair(curStr[0], childHash));
|
|
}
|
|
}
|
|
if (currentPosition == name)
|
|
{
|
|
fNameHasValue = fNodeHasValue;
|
|
if (fNameHasValue)
|
|
{
|
|
outPoint = claim.outPoint;
|
|
if (!getLastTakeoverForName(name, nHeightOfLastTakeover))
|
|
return false;
|
|
}
|
|
valueHash.SetNull();
|
|
}
|
|
nodes.emplace_back(std::move(children), fNodeHasValue, valueHash);
|
|
current = nextCurrent;
|
|
}
|
|
proof = CClaimTrieProof(std::move(nodes), fNameHasValue, outPoint, nHeightOfLastTakeover);
|
|
return true;
|
|
}
|