d1910eaff4
* Split up remaining logic from CMasternodeMan into CMasternodeMetaMan and CMasternodeUtils Also get rid of CMastermode and store remaining meta info in CMasternodeMetaInfo * Also allow non-const T in Serialize/Unserialize for shared_ptr * Rename CActiveDeterministicMasternodeManager to CActiveMasternodeManager * Fix nowallet compile in masternode-utils.cpp
490 lines
14 KiB
C++
490 lines
14 KiB
C++
// Copyright (c) 2014-2018 The Dash Core developers
|
|
// Distributed under the MIT/X11 software license, see the accompanying
|
|
// file COPYING or http://www.opensource.org/licenses/mit-license.php.
|
|
|
|
#include "privatesend.h"
|
|
|
|
#include "activemasternode.h"
|
|
#include "consensus/validation.h"
|
|
#include "masternode-payments.h"
|
|
#include "masternode-sync.h"
|
|
#include "messagesigner.h"
|
|
#include "netmessagemaker.h"
|
|
#include "script/sign.h"
|
|
#include "txmempool.h"
|
|
#include "util.h"
|
|
#include "utilmoneystr.h"
|
|
#include "validation.h"
|
|
|
|
#include <string>
|
|
|
|
bool CPrivateSendEntry::AddScriptSig(const CTxIn& txin)
|
|
{
|
|
for (auto& txdsin : vecTxDSIn) {
|
|
if (txdsin.prevout == txin.prevout && txdsin.nSequence == txin.nSequence) {
|
|
if (txdsin.fHasSig) return false;
|
|
|
|
txdsin.scriptSig = txin.scriptSig;
|
|
txdsin.fHasSig = true;
|
|
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
uint256 CPrivateSendQueue::GetSignatureHash() const
|
|
{
|
|
return SerializeHash(*this);
|
|
}
|
|
|
|
bool CPrivateSendQueue::Sign()
|
|
{
|
|
if (!fMasternodeMode) return false;
|
|
|
|
|
|
uint256 hash = GetSignatureHash();
|
|
CBLSSignature sig = activeMasternodeInfo.blsKeyOperator->Sign(hash);
|
|
if (!sig.IsValid()) {
|
|
return false;
|
|
}
|
|
sig.GetBuf(vchSig);
|
|
|
|
return true;
|
|
}
|
|
|
|
bool CPrivateSendQueue::CheckSignature(const CBLSPublicKey& blsPubKey) const
|
|
{
|
|
uint256 hash = GetSignatureHash();
|
|
|
|
CBLSSignature sig;
|
|
sig.SetBuf(vchSig);
|
|
if (!sig.IsValid() || !sig.VerifyInsecure(blsPubKey, hash)) {
|
|
LogPrintf("CTxLockVote::CheckSignature -- VerifyInsecure() failed\n");
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
bool CPrivateSendQueue::Relay(CConnman& connman)
|
|
{
|
|
connman.ForEachNode([&connman, this](CNode* pnode) {
|
|
CNetMsgMaker msgMaker(pnode->GetSendVersion());
|
|
if (pnode->nVersion >= MIN_PRIVATESEND_PEER_PROTO_VERSION)
|
|
connman.PushMessage(pnode, msgMaker.Make(NetMsgType::DSQUEUE, (*this)));
|
|
});
|
|
return true;
|
|
}
|
|
|
|
uint256 CPrivateSendBroadcastTx::GetSignatureHash() const
|
|
{
|
|
return SerializeHash(*this);
|
|
}
|
|
|
|
bool CPrivateSendBroadcastTx::Sign()
|
|
{
|
|
if (!fMasternodeMode) return false;
|
|
|
|
|
|
uint256 hash = GetSignatureHash();
|
|
|
|
CBLSSignature sig = activeMasternodeInfo.blsKeyOperator->Sign(hash);
|
|
if (!sig.IsValid()) {
|
|
return false;
|
|
}
|
|
sig.GetBuf(vchSig);
|
|
|
|
return true;
|
|
}
|
|
|
|
bool CPrivateSendBroadcastTx::CheckSignature(const CBLSPublicKey& blsPubKey) const
|
|
{
|
|
|
|
uint256 hash = GetSignatureHash();
|
|
|
|
CBLSSignature sig;
|
|
sig.SetBuf(vchSig);
|
|
if (!sig.IsValid() || !sig.VerifyInsecure(blsPubKey, hash)) {
|
|
LogPrintf("CTxLockVote::CheckSignature -- VerifyInsecure() failed\n");
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
bool CPrivateSendBroadcastTx::IsExpired(int nHeight)
|
|
{
|
|
// expire confirmed DSTXes after ~1h since confirmation
|
|
return (nConfirmedHeight != -1) && (nHeight - nConfirmedHeight > 24);
|
|
}
|
|
|
|
void CPrivateSendBaseSession::SetNull()
|
|
{
|
|
// Both sides
|
|
LOCK(cs_privatesend);
|
|
nState = POOL_STATE_IDLE;
|
|
nSessionID = 0;
|
|
nSessionDenom = 0;
|
|
vecEntries.clear();
|
|
finalMutableTransaction.vin.clear();
|
|
finalMutableTransaction.vout.clear();
|
|
nTimeLastSuccessfulStep = GetTime();
|
|
}
|
|
|
|
void CPrivateSendBaseManager::SetNull()
|
|
{
|
|
LOCK(cs_vecqueue);
|
|
vecPrivateSendQueue.clear();
|
|
}
|
|
|
|
void CPrivateSendBaseManager::CheckQueue()
|
|
{
|
|
TRY_LOCK(cs_vecqueue, lockDS);
|
|
if (!lockDS) return; // it's ok to fail here, we run this quite frequently
|
|
|
|
// check mixing queue objects for timeouts
|
|
std::vector<CPrivateSendQueue>::iterator it = vecPrivateSendQueue.begin();
|
|
while (it != vecPrivateSendQueue.end()) {
|
|
if ((*it).IsExpired()) {
|
|
LogPrint("privatesend", "CPrivateSendBaseManager::%s -- Removing expired queue (%s)\n", __func__, (*it).ToString());
|
|
it = vecPrivateSendQueue.erase(it);
|
|
} else
|
|
++it;
|
|
}
|
|
}
|
|
|
|
bool CPrivateSendBaseManager::GetQueueItemAndTry(CPrivateSendQueue& dsqRet)
|
|
{
|
|
TRY_LOCK(cs_vecqueue, lockDS);
|
|
if (!lockDS) return false; // it's ok to fail here, we run this quite frequently
|
|
|
|
for (auto& dsq : vecPrivateSendQueue) {
|
|
// only try each queue once
|
|
if (dsq.fTried || dsq.IsExpired()) continue;
|
|
dsq.fTried = true;
|
|
dsqRet = dsq;
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
std::string CPrivateSendBaseSession::GetStateString() const
|
|
{
|
|
switch (nState) {
|
|
case POOL_STATE_IDLE:
|
|
return "IDLE";
|
|
case POOL_STATE_QUEUE:
|
|
return "QUEUE";
|
|
case POOL_STATE_ACCEPTING_ENTRIES:
|
|
return "ACCEPTING_ENTRIES";
|
|
case POOL_STATE_SIGNING:
|
|
return "SIGNING";
|
|
case POOL_STATE_ERROR:
|
|
return "ERROR";
|
|
case POOL_STATE_SUCCESS:
|
|
return "SUCCESS";
|
|
default:
|
|
return "UNKNOWN";
|
|
}
|
|
}
|
|
|
|
// Definitions for static data members
|
|
std::vector<CAmount> CPrivateSend::vecStandardDenominations;
|
|
std::map<uint256, CPrivateSendBroadcastTx> CPrivateSend::mapDSTX;
|
|
CCriticalSection CPrivateSend::cs_mapdstx;
|
|
|
|
void CPrivateSend::InitStandardDenominations()
|
|
{
|
|
vecStandardDenominations.clear();
|
|
/* Denominations
|
|
|
|
A note about convertibility. Within mixing pools, each denomination
|
|
is convertible to another.
|
|
|
|
For example:
|
|
1DRK+1000 == (.1DRK+100)*10
|
|
10DRK+10000 == (1DRK+1000)*10
|
|
*/
|
|
/* Disabled
|
|
vecStandardDenominations.push_back( (100 * COIN)+100000 );
|
|
*/
|
|
vecStandardDenominations.push_back((10 * COIN) + 10000);
|
|
vecStandardDenominations.push_back((1 * COIN) + 1000);
|
|
vecStandardDenominations.push_back((.1 * COIN) + 100);
|
|
vecStandardDenominations.push_back((.01 * COIN) + 10);
|
|
vecStandardDenominations.push_back((.001 * COIN) + 1);
|
|
}
|
|
|
|
// check to make sure the collateral provided by the client is valid
|
|
bool CPrivateSend::IsCollateralValid(const CTransaction& txCollateral)
|
|
{
|
|
if (txCollateral.vout.empty()) return false;
|
|
if (txCollateral.nLockTime != 0) return false;
|
|
|
|
CAmount nValueIn = 0;
|
|
CAmount nValueOut = 0;
|
|
|
|
for (const auto& txout : txCollateral.vout) {
|
|
nValueOut += txout.nValue;
|
|
|
|
if (!txout.scriptPubKey.IsPayToPublicKeyHash() && !txout.scriptPubKey.IsUnspendable()) {
|
|
LogPrintf("CPrivateSend::IsCollateralValid -- Invalid Script, txCollateral=%s", txCollateral.ToString());
|
|
return false;
|
|
}
|
|
}
|
|
|
|
for (const auto& txin : txCollateral.vin) {
|
|
Coin coin;
|
|
if (!GetUTXOCoin(txin.prevout, coin)) {
|
|
LogPrint("privatesend", "CPrivateSend::IsCollateralValid -- Unknown inputs in collateral transaction, txCollateral=%s", txCollateral.ToString());
|
|
return false;
|
|
}
|
|
nValueIn += coin.out.nValue;
|
|
}
|
|
|
|
//collateral transactions are required to pay out a small fee to the miners
|
|
if (nValueIn - nValueOut < GetCollateralAmount()) {
|
|
LogPrint("privatesend", "CPrivateSend::IsCollateralValid -- did not include enough fees in transaction: fees: %d, txCollateral=%s", nValueOut - nValueIn, txCollateral.ToString());
|
|
return false;
|
|
}
|
|
|
|
LogPrint("privatesend", "CPrivateSend::IsCollateralValid -- %s", txCollateral.ToString());
|
|
|
|
{
|
|
LOCK(cs_main);
|
|
CValidationState validationState;
|
|
if (!AcceptToMemoryPool(mempool, validationState, MakeTransactionRef(txCollateral), false, NULL, false, maxTxFee, true)) {
|
|
LogPrint("privatesend", "CPrivateSend::IsCollateralValid -- didn't pass AcceptToMemoryPool()\n");
|
|
return false;
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
bool CPrivateSend::IsCollateralAmount(CAmount nInputAmount)
|
|
{
|
|
// collateral input can be anything between 1x and "max" (including both)
|
|
return (nInputAmount >= GetCollateralAmount() && nInputAmount <= GetMaxCollateralAmount());
|
|
}
|
|
|
|
/* Create a nice string to show the denominations
|
|
Function returns as follows (for 4 denominations):
|
|
( bit on if present )
|
|
bit 0 - 10
|
|
bit 1 - 1
|
|
bit 2 - .1
|
|
bit 3 - .01
|
|
bit 4 and so on - out-of-bounds
|
|
none of above - non-denom
|
|
*/
|
|
std::string CPrivateSend::GetDenominationsToString(int nDenom)
|
|
{
|
|
std::string strDenom = "";
|
|
int nMaxDenoms = vecStandardDenominations.size();
|
|
|
|
if (nDenom >= (1 << nMaxDenoms)) {
|
|
return "out-of-bounds";
|
|
}
|
|
|
|
for (int i = 0; i < nMaxDenoms; ++i) {
|
|
if (nDenom & (1 << i)) {
|
|
strDenom += (strDenom.empty() ? "" : "+") + FormatMoney(vecStandardDenominations[i]);
|
|
}
|
|
}
|
|
|
|
if (strDenom.empty()) {
|
|
return "non-denom";
|
|
}
|
|
|
|
return strDenom;
|
|
}
|
|
|
|
/* Return a bitshifted integer representing the denominations in this list
|
|
Function returns as follows (for 4 denominations):
|
|
( bit on if present )
|
|
10 - bit 0
|
|
1 - bit 1
|
|
.1 - bit 2
|
|
.01 - bit 3
|
|
non-denom - 0, all bits off
|
|
*/
|
|
int CPrivateSend::GetDenominations(const std::vector<CTxOut>& vecTxOut, bool fSingleRandomDenom)
|
|
{
|
|
std::vector<std::pair<CAmount, int> > vecDenomUsed;
|
|
|
|
// make a list of denominations, with zero uses
|
|
for (const auto& nDenomValue : vecStandardDenominations)
|
|
vecDenomUsed.push_back(std::make_pair(nDenomValue, 0));
|
|
|
|
// look for denominations and update uses to 1
|
|
for (const auto& txout : vecTxOut) {
|
|
bool found = false;
|
|
for (auto& s : vecDenomUsed) {
|
|
if (txout.nValue == s.first) {
|
|
s.second = 1;
|
|
found = true;
|
|
}
|
|
}
|
|
if (!found) return 0;
|
|
}
|
|
|
|
int nDenom = 0;
|
|
int c = 0;
|
|
// if the denomination is used, shift the bit on
|
|
for (const auto& s : vecDenomUsed) {
|
|
int bit = (fSingleRandomDenom ? GetRandInt(2) : 1) & s.second;
|
|
nDenom |= bit << c++;
|
|
if (fSingleRandomDenom && bit) break; // use just one random denomination
|
|
}
|
|
|
|
return nDenom;
|
|
}
|
|
|
|
bool CPrivateSend::GetDenominationsBits(int nDenom, std::vector<int>& vecBitsRet)
|
|
{
|
|
// ( bit on if present, 4 denominations example )
|
|
// bit 0 - 100DASH+1
|
|
// bit 1 - 10DASH+1
|
|
// bit 2 - 1DASH+1
|
|
// bit 3 - .1DASH+1
|
|
|
|
int nMaxDenoms = vecStandardDenominations.size();
|
|
|
|
if (nDenom >= (1 << nMaxDenoms)) return false;
|
|
|
|
vecBitsRet.clear();
|
|
|
|
for (int i = 0; i < nMaxDenoms; ++i) {
|
|
if (nDenom & (1 << i)) {
|
|
vecBitsRet.push_back(i);
|
|
}
|
|
}
|
|
|
|
return !vecBitsRet.empty();
|
|
}
|
|
|
|
int CPrivateSend::GetDenominationsByAmounts(const std::vector<CAmount>& vecAmount)
|
|
{
|
|
CScript scriptTmp = CScript();
|
|
std::vector<CTxOut> vecTxOut;
|
|
|
|
for (auto it = vecAmount.rbegin(); it != vecAmount.rend(); ++it) {
|
|
CTxOut txout((*it), scriptTmp);
|
|
vecTxOut.push_back(txout);
|
|
}
|
|
|
|
return GetDenominations(vecTxOut, true);
|
|
}
|
|
|
|
bool CPrivateSend::IsDenominatedAmount(CAmount nInputAmount)
|
|
{
|
|
for (const auto& nDenomValue : vecStandardDenominations)
|
|
if (nInputAmount == nDenomValue)
|
|
return true;
|
|
return false;
|
|
}
|
|
|
|
std::string CPrivateSend::GetMessageByID(PoolMessage nMessageID)
|
|
{
|
|
switch (nMessageID) {
|
|
case ERR_ALREADY_HAVE:
|
|
return _("Already have that input.");
|
|
case ERR_DENOM:
|
|
return _("No matching denominations found for mixing.");
|
|
case ERR_ENTRIES_FULL:
|
|
return _("Entries are full.");
|
|
case ERR_EXISTING_TX:
|
|
return _("Not compatible with existing transactions.");
|
|
case ERR_FEES:
|
|
return _("Transaction fees are too high.");
|
|
case ERR_INVALID_COLLATERAL:
|
|
return _("Collateral not valid.");
|
|
case ERR_INVALID_INPUT:
|
|
return _("Input is not valid.");
|
|
case ERR_INVALID_SCRIPT:
|
|
return _("Invalid script detected.");
|
|
case ERR_INVALID_TX:
|
|
return _("Transaction not valid.");
|
|
case ERR_MAXIMUM:
|
|
return _("Entry exceeds maximum size.");
|
|
case ERR_MN_LIST:
|
|
return _("Not in the Masternode list.");
|
|
case ERR_MODE:
|
|
return _("Incompatible mode.");
|
|
case ERR_NON_STANDARD_PUBKEY:
|
|
return _("Non-standard public key detected.");
|
|
case ERR_NOT_A_MN:
|
|
return _("This is not a Masternode."); // not used
|
|
case ERR_QUEUE_FULL:
|
|
return _("Masternode queue is full.");
|
|
case ERR_RECENT:
|
|
return _("Last PrivateSend was too recent.");
|
|
case ERR_SESSION:
|
|
return _("Session not complete!");
|
|
case ERR_MISSING_TX:
|
|
return _("Missing input transaction information.");
|
|
case ERR_VERSION:
|
|
return _("Incompatible version.");
|
|
case MSG_NOERR:
|
|
return _("No errors detected.");
|
|
case MSG_SUCCESS:
|
|
return _("Transaction created successfully.");
|
|
case MSG_ENTRIES_ADDED:
|
|
return _("Your entries added successfully.");
|
|
default:
|
|
return _("Unknown response.");
|
|
}
|
|
}
|
|
|
|
void CPrivateSend::AddDSTX(const CPrivateSendBroadcastTx& dstx)
|
|
{
|
|
LOCK(cs_mapdstx);
|
|
mapDSTX.insert(std::make_pair(dstx.tx->GetHash(), dstx));
|
|
}
|
|
|
|
CPrivateSendBroadcastTx CPrivateSend::GetDSTX(const uint256& hash)
|
|
{
|
|
LOCK(cs_mapdstx);
|
|
auto it = mapDSTX.find(hash);
|
|
return (it == mapDSTX.end()) ? CPrivateSendBroadcastTx() : it->second;
|
|
}
|
|
|
|
void CPrivateSend::CheckDSTXes(int nHeight)
|
|
{
|
|
LOCK(cs_mapdstx);
|
|
std::map<uint256, CPrivateSendBroadcastTx>::iterator it = mapDSTX.begin();
|
|
while (it != mapDSTX.end()) {
|
|
if (it->second.IsExpired(nHeight)) {
|
|
mapDSTX.erase(it++);
|
|
} else {
|
|
++it;
|
|
}
|
|
}
|
|
LogPrint("privatesend", "CPrivateSend::CheckDSTXes -- mapDSTX.size()=%llu\n", mapDSTX.size());
|
|
}
|
|
|
|
void CPrivateSend::UpdatedBlockTip(const CBlockIndex* pindex)
|
|
{
|
|
if (pindex && !fLiteMode && masternodeSync.IsBlockchainSynced()) {
|
|
CheckDSTXes(pindex->nHeight);
|
|
}
|
|
}
|
|
|
|
void CPrivateSend::SyncTransaction(const CTransaction& tx, const CBlockIndex* pindex, int posInBlock)
|
|
{
|
|
if (tx.IsCoinBase()) return;
|
|
|
|
LOCK2(cs_main, cs_mapdstx);
|
|
|
|
uint256 txHash = tx.GetHash();
|
|
if (!mapDSTX.count(txHash)) return;
|
|
|
|
// When tx is 0-confirmed or conflicted, posInBlock is SYNC_TRANSACTION_NOT_IN_BLOCK and nConfirmedHeight should be set to -1
|
|
mapDSTX[txHash].SetConfirmedHeight(posInBlock == CMainSignals::SYNC_TRANSACTION_NOT_IN_BLOCK ? -1 : pindex->nHeight);
|
|
LogPrint("privatesend", "CPrivateSend::SyncTransaction -- txid=%s\n", txHash.ToString());
|
|
}
|