1678 lines
69 KiB
C++
1678 lines
69 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-client.h"
|
|
|
|
#include "consensus/validation.h"
|
|
#include "core_io.h"
|
|
#include "init.h"
|
|
#include "masternode-payments.h"
|
|
#include "masternode-sync.h"
|
|
#include "masternode-meta.h"
|
|
#include "netmessagemaker.h"
|
|
#include "script/sign.h"
|
|
#include "txmempool.h"
|
|
#include "util.h"
|
|
#include "utilmoneystr.h"
|
|
#include "validation.h"
|
|
#include "wallet/coincontrol.h"
|
|
|
|
#include <memory>
|
|
|
|
CPrivateSendClientManager privateSendClient;
|
|
|
|
void CPrivateSendClientManager::ProcessMessage(CNode* pfrom, const std::string& strCommand, CDataStream& vRecv, CConnman& connman)
|
|
{
|
|
if (fMasternodeMode) return;
|
|
if (fLiteMode) return; // ignore all Dash related functionality
|
|
if (!masternodeSync.IsBlockchainSynced()) return;
|
|
|
|
if (!CheckDiskSpace()) {
|
|
ResetPool();
|
|
fEnablePrivateSend = false;
|
|
LogPrintf("CPrivateSendClientManager::ProcessMessage -- Not enough disk space, disabling PrivateSend.\n");
|
|
return;
|
|
}
|
|
|
|
if (strCommand == NetMsgType::DSQUEUE) {
|
|
if (pfrom->nVersion < MIN_PRIVATESEND_PEER_PROTO_VERSION) {
|
|
LogPrint("privatesend", "DSQUEUE -- peer=%d using obsolete version %i\n", pfrom->id, pfrom->nVersion);
|
|
connman.PushMessage(pfrom, CNetMsgMaker(pfrom->GetSendVersion()).Make(NetMsgType::REJECT, strCommand, REJECT_OBSOLETE, strprintf("Version must be %d or greater", MIN_PRIVATESEND_PEER_PROTO_VERSION)));
|
|
return;
|
|
}
|
|
|
|
CPrivateSendQueue dsq;
|
|
vRecv >> dsq;
|
|
|
|
{
|
|
TRY_LOCK(cs_vecqueue, lockRecv);
|
|
if (!lockRecv) return;
|
|
|
|
// process every dsq only once
|
|
for (const auto& q : vecPrivateSendQueue) {
|
|
if (q == dsq) {
|
|
// LogPrint("privatesend", "DSQUEUE -- %s seen\n", dsq.ToString());
|
|
return;
|
|
}
|
|
}
|
|
} // cs_vecqueue
|
|
|
|
LogPrint("privatesend", "DSQUEUE -- %s new\n", dsq.ToString());
|
|
|
|
if (dsq.IsExpired()) return;
|
|
|
|
auto mnList = deterministicMNManager->GetListAtChainTip();
|
|
auto dmn = mnList.GetValidMNByCollateral(dsq.masternodeOutpoint);
|
|
if (!dmn) return;
|
|
|
|
if (!dsq.CheckSignature(dmn->pdmnState->pubKeyOperator)) {
|
|
LOCK(cs_main);
|
|
Misbehaving(pfrom->id, 10);
|
|
return;
|
|
}
|
|
|
|
// if the queue is ready, submit if we can
|
|
if (dsq.fReady) {
|
|
LOCK(cs_deqsessions);
|
|
for (auto& session : deqSessions) {
|
|
CDeterministicMNCPtr mnMixing;
|
|
if (session.GetMixingMasternodeInfo(mnMixing) && mnMixing->pdmnState->addr == dmn->pdmnState->addr && session.GetState() == POOL_STATE_QUEUE) {
|
|
LogPrint("privatesend", "DSQUEUE -- PrivateSend queue (%s) is ready on masternode %s\n", dsq.ToString(), dmn->pdmnState->addr.ToString());
|
|
session.SubmitDenominate(connman);
|
|
return;
|
|
}
|
|
}
|
|
} else {
|
|
LOCK(cs_deqsessions); // have to lock this first to avoid deadlocks with cs_vecqueue
|
|
TRY_LOCK(cs_vecqueue, lockRecv);
|
|
if (!lockRecv) return;
|
|
|
|
for (const auto& q : vecPrivateSendQueue) {
|
|
if (q.masternodeOutpoint == dsq.masternodeOutpoint) {
|
|
// no way same mn can send another "not yet ready" dsq this soon
|
|
LogPrint("privatesend", "DSQUEUE -- Masternode %s is sending WAY too many dsq messages\n", dmn->pdmnState->ToString());
|
|
return;
|
|
}
|
|
}
|
|
|
|
int64_t nLastDsq = mmetaman.GetMetaInfo(dmn->proTxHash)->GetLastDsq();
|
|
int nThreshold = nLastDsq + mnList.GetValidMNsCount() / 5;
|
|
LogPrint("privatesend", "DSQUEUE -- nLastDsq: %d threshold: %d nDsqCount: %d\n", nLastDsq, nThreshold, mmetaman.GetDsqCount());
|
|
//don't allow a few nodes to dominate the queuing process
|
|
if (nLastDsq != 0 && nThreshold > mmetaman.GetDsqCount()) {
|
|
LogPrint("privatesend", "DSQUEUE -- Masternode %s is sending too many dsq messages\n", dmn->proTxHash.ToString());
|
|
return;
|
|
}
|
|
|
|
mmetaman.AllowMixing(dmn->proTxHash);
|
|
|
|
LogPrint("privatesend", "DSQUEUE -- new PrivateSend queue (%s) from masternode %s\n", dsq.ToString(), dmn->pdmnState->addr.ToString());
|
|
for (auto& session : deqSessions) {
|
|
CDeterministicMNCPtr mnMixing;
|
|
if (session.GetMixingMasternodeInfo(mnMixing) && mnMixing->collateralOutpoint == dsq.masternodeOutpoint) {
|
|
dsq.fTried = true;
|
|
}
|
|
}
|
|
vecPrivateSendQueue.push_back(dsq);
|
|
dsq.Relay(connman);
|
|
}
|
|
|
|
} else if (
|
|
strCommand == NetMsgType::DSSTATUSUPDATE ||
|
|
strCommand == NetMsgType::DSFINALTX ||
|
|
strCommand == NetMsgType::DSCOMPLETE) {
|
|
LOCK(cs_deqsessions);
|
|
for (auto& session : deqSessions) {
|
|
session.ProcessMessage(pfrom, strCommand, vRecv, connman);
|
|
}
|
|
}
|
|
}
|
|
|
|
void CPrivateSendClientSession::ProcessMessage(CNode* pfrom, const std::string& strCommand, CDataStream& vRecv, CConnman& connman)
|
|
{
|
|
if (fMasternodeMode) return;
|
|
if (fLiteMode) return; // ignore all Dash related functionality
|
|
if (!masternodeSync.IsBlockchainSynced()) return;
|
|
|
|
if (strCommand == NetMsgType::DSSTATUSUPDATE) {
|
|
if (pfrom->nVersion < MIN_PRIVATESEND_PEER_PROTO_VERSION) {
|
|
LogPrint("privatesend", "DSSTATUSUPDATE -- peer=%d using obsolete version %i\n", pfrom->id, pfrom->nVersion);
|
|
connman.PushMessage(pfrom, CNetMsgMaker(pfrom->GetSendVersion()).Make(NetMsgType::REJECT, strCommand, REJECT_OBSOLETE, strprintf("Version must be %d or greater", MIN_PRIVATESEND_PEER_PROTO_VERSION)));
|
|
return;
|
|
}
|
|
|
|
if (!mixingMasternode) return;
|
|
if (mixingMasternode->pdmnState->addr != pfrom->addr) {
|
|
//LogPrintf("DSSTATUSUPDATE -- message doesn't match current Masternode: infoMixingMasternode %s addr %s\n", infoMixingMasternode.addr.ToString(), pfrom->addr.ToString());
|
|
return;
|
|
}
|
|
|
|
int nMsgSessionID;
|
|
int nMsgState;
|
|
int nMsgEntriesCount;
|
|
int nMsgStatusUpdate;
|
|
int nMsgMessageID;
|
|
vRecv >> nMsgSessionID >> nMsgState >> nMsgEntriesCount >> nMsgStatusUpdate >> nMsgMessageID;
|
|
|
|
if (nMsgState < POOL_STATE_MIN || nMsgState > POOL_STATE_MAX) {
|
|
LogPrint("privatesend", "DSSTATUSUPDATE -- nMsgState is out of bounds: %d\n", nMsgState);
|
|
return;
|
|
}
|
|
|
|
if (nMsgStatusUpdate < STATUS_REJECTED || nMsgStatusUpdate > STATUS_ACCEPTED) {
|
|
LogPrint("privatesend", "DSSTATUSUPDATE -- nMsgStatusUpdate is out of bounds: %d\n", nMsgStatusUpdate);
|
|
return;
|
|
}
|
|
|
|
if (nMsgMessageID < MSG_POOL_MIN || nMsgMessageID > MSG_POOL_MAX) {
|
|
LogPrint("privatesend", "DSSTATUSUPDATE -- nMsgMessageID is out of bounds: %d\n", nMsgMessageID);
|
|
return;
|
|
}
|
|
|
|
LogPrint("privatesend", "DSSTATUSUPDATE -- nMsgSessionID %d nMsgState: %d nEntriesCount: %d nMsgStatusUpdate: %d nMsgMessageID %d (%s)\n",
|
|
nMsgSessionID, nMsgState, nEntriesCount, nMsgStatusUpdate, nMsgMessageID, CPrivateSend::GetMessageByID(PoolMessage(nMsgMessageID)));
|
|
|
|
if (!CheckPoolStateUpdate(PoolState(nMsgState), nMsgEntriesCount, PoolStatusUpdate(nMsgStatusUpdate), PoolMessage(nMsgMessageID), nMsgSessionID)) {
|
|
LogPrint("privatesend", "DSSTATUSUPDATE -- CheckPoolStateUpdate failed\n");
|
|
}
|
|
|
|
} else if (strCommand == NetMsgType::DSFINALTX) {
|
|
if (pfrom->nVersion < MIN_PRIVATESEND_PEER_PROTO_VERSION) {
|
|
LogPrint("privatesend", "DSFINALTX -- peer=%d using obsolete version %i\n", pfrom->id, pfrom->nVersion);
|
|
connman.PushMessage(pfrom, CNetMsgMaker(pfrom->GetSendVersion()).Make(NetMsgType::REJECT, strCommand, REJECT_OBSOLETE, strprintf("Version must be %d or greater", MIN_PRIVATESEND_PEER_PROTO_VERSION)));
|
|
return;
|
|
}
|
|
|
|
if (!mixingMasternode) return;
|
|
if (mixingMasternode->pdmnState->addr != pfrom->addr) {
|
|
//LogPrintf("DSFINALTX -- message doesn't match current Masternode: infoMixingMasternode %s addr %s\n", infoMixingMasternode.addr.ToString(), pfrom->addr.ToString());
|
|
return;
|
|
}
|
|
|
|
int nMsgSessionID;
|
|
vRecv >> nMsgSessionID;
|
|
CTransaction txNew(deserialize, vRecv);
|
|
|
|
if (nSessionID != nMsgSessionID) {
|
|
LogPrint("privatesend", "DSFINALTX -- message doesn't match current PrivateSend session: nSessionID: %d nMsgSessionID: %d\n", nSessionID, nMsgSessionID);
|
|
return;
|
|
}
|
|
|
|
LogPrint("privatesend", "DSFINALTX -- txNew %s", txNew.ToString());
|
|
|
|
//check to see if input is spent already? (and probably not confirmed)
|
|
SignFinalTransaction(txNew, pfrom, connman);
|
|
|
|
} else if (strCommand == NetMsgType::DSCOMPLETE) {
|
|
if (pfrom->nVersion < MIN_PRIVATESEND_PEER_PROTO_VERSION) {
|
|
LogPrint("privatesend", "DSCOMPLETE -- peer=%d using obsolete version %i\n", pfrom->id, pfrom->nVersion);
|
|
connman.PushMessage(pfrom, CNetMsgMaker(pfrom->GetSendVersion()).Make(NetMsgType::REJECT, strCommand, REJECT_OBSOLETE, strprintf("Version must be %d or greater", MIN_PRIVATESEND_PEER_PROTO_VERSION)));
|
|
return;
|
|
}
|
|
|
|
if (!mixingMasternode) return;
|
|
if (mixingMasternode->pdmnState->addr != pfrom->addr) {
|
|
LogPrint("privatesend", "DSCOMPLETE -- message doesn't match current Masternode: infoMixingMasternode=%s addr=%s\n", mixingMasternode->pdmnState->addr.ToString(), pfrom->addr.ToString());
|
|
return;
|
|
}
|
|
|
|
int nMsgSessionID;
|
|
int nMsgMessageID;
|
|
vRecv >> nMsgSessionID >> nMsgMessageID;
|
|
|
|
if (nMsgMessageID < MSG_POOL_MIN || nMsgMessageID > MSG_POOL_MAX) {
|
|
LogPrint("privatesend", "DSCOMPLETE -- nMsgMessageID is out of bounds: %d\n", nMsgMessageID);
|
|
return;
|
|
}
|
|
|
|
if (nSessionID != nMsgSessionID) {
|
|
LogPrint("privatesend", "DSCOMPLETE -- message doesn't match current PrivateSend session: nSessionID: %d nMsgSessionID: %d\n", nSessionID, nMsgSessionID);
|
|
return;
|
|
}
|
|
|
|
LogPrint("privatesend", "DSCOMPLETE -- nMsgSessionID %d nMsgMessageID %d (%s)\n", nMsgSessionID, nMsgMessageID, CPrivateSend::GetMessageByID(PoolMessage(nMsgMessageID)));
|
|
|
|
CompletedTransaction(PoolMessage(nMsgMessageID));
|
|
}
|
|
}
|
|
|
|
void CPrivateSendClientSession::ResetPool()
|
|
{
|
|
txMyCollateral = CMutableTransaction();
|
|
UnlockCoins();
|
|
keyHolderStorage.ReturnAll();
|
|
SetNull();
|
|
}
|
|
|
|
void CPrivateSendClientManager::ResetPool()
|
|
{
|
|
LOCK(cs_deqsessions);
|
|
nCachedLastSuccessBlock = 0;
|
|
vecMasternodesUsed.clear();
|
|
for (auto& session : deqSessions) {
|
|
session.ResetPool();
|
|
}
|
|
deqSessions.clear();
|
|
}
|
|
|
|
void CPrivateSendClientSession::SetNull()
|
|
{
|
|
// Client side
|
|
nEntriesCount = 0;
|
|
fLastEntryAccepted = false;
|
|
mixingMasternode = nullptr;
|
|
pendingDsaRequest = CPendingDsaRequest();
|
|
|
|
CPrivateSendBaseSession::SetNull();
|
|
}
|
|
|
|
//
|
|
// Unlock coins after mixing fails or succeeds
|
|
//
|
|
void CPrivateSendClientSession::UnlockCoins()
|
|
{
|
|
if (!pwalletMain) return;
|
|
|
|
while (true) {
|
|
TRY_LOCK(pwalletMain->cs_wallet, lockWallet);
|
|
if (!lockWallet) {
|
|
MilliSleep(50);
|
|
continue;
|
|
}
|
|
for (const auto& outpoint : vecOutPointLocked)
|
|
pwalletMain->UnlockCoin(outpoint);
|
|
break;
|
|
}
|
|
|
|
vecOutPointLocked.clear();
|
|
}
|
|
|
|
std::string CPrivateSendClientSession::GetStatus(bool fWaitForBlock)
|
|
{
|
|
static int nStatusMessageProgress = 0;
|
|
nStatusMessageProgress += 10;
|
|
std::string strSuffix = "";
|
|
|
|
if (fWaitForBlock || !masternodeSync.IsBlockchainSynced())
|
|
return strAutoDenomResult;
|
|
|
|
switch (nState) {
|
|
case POOL_STATE_IDLE:
|
|
return _("PrivateSend is idle.");
|
|
case POOL_STATE_QUEUE:
|
|
if (nStatusMessageProgress % 70 <= 30)
|
|
strSuffix = ".";
|
|
else if (nStatusMessageProgress % 70 <= 50)
|
|
strSuffix = "..";
|
|
else if (nStatusMessageProgress % 70 <= 70)
|
|
strSuffix = "...";
|
|
return strprintf(_("Submitted to masternode, waiting in queue %s"), strSuffix);
|
|
case POOL_STATE_ACCEPTING_ENTRIES:
|
|
if (nEntriesCount == 0) {
|
|
nStatusMessageProgress = 0;
|
|
return strAutoDenomResult;
|
|
} else if (fLastEntryAccepted) {
|
|
if (nStatusMessageProgress % 10 > 8) {
|
|
fLastEntryAccepted = false;
|
|
nStatusMessageProgress = 0;
|
|
}
|
|
return _("PrivateSend request complete:") + " " + _("Your transaction was accepted into the pool!");
|
|
} else {
|
|
if (nStatusMessageProgress % 70 <= 40)
|
|
return strprintf(_("Submitted following entries to masternode: %u"), nEntriesCount);
|
|
else if (nStatusMessageProgress % 70 <= 50)
|
|
strSuffix = ".";
|
|
else if (nStatusMessageProgress % 70 <= 60)
|
|
strSuffix = "..";
|
|
else if (nStatusMessageProgress % 70 <= 70)
|
|
strSuffix = "...";
|
|
return strprintf(_("Submitted to masternode, waiting for more entries ( %u ) %s"), nEntriesCount, strSuffix);
|
|
}
|
|
case POOL_STATE_SIGNING:
|
|
if (nStatusMessageProgress % 70 <= 40)
|
|
return _("Found enough users, signing ...");
|
|
else if (nStatusMessageProgress % 70 <= 50)
|
|
strSuffix = ".";
|
|
else if (nStatusMessageProgress % 70 <= 60)
|
|
strSuffix = "..";
|
|
else if (nStatusMessageProgress % 70 <= 70)
|
|
strSuffix = "...";
|
|
return strprintf(_("Found enough users, signing ( waiting %s )"), strSuffix);
|
|
case POOL_STATE_ERROR:
|
|
return _("PrivateSend request incomplete:") + " " + strLastMessage + " " + _("Will retry...");
|
|
case POOL_STATE_SUCCESS:
|
|
return _("PrivateSend request complete:") + " " + strLastMessage;
|
|
default:
|
|
return strprintf(_("Unknown state: id = %u"), nState);
|
|
}
|
|
}
|
|
|
|
std::string CPrivateSendClientManager::GetStatuses()
|
|
{
|
|
LOCK(cs_deqsessions);
|
|
std::string strStatus;
|
|
bool fWaitForBlock = WaitForAnotherBlock();
|
|
|
|
for (auto& session : deqSessions) {
|
|
strStatus += session.GetStatus(fWaitForBlock) + "; ";
|
|
}
|
|
return strStatus;
|
|
}
|
|
|
|
std::string CPrivateSendClientManager::GetSessionDenoms()
|
|
{
|
|
LOCK(cs_deqsessions);
|
|
std::string strSessionDenoms;
|
|
|
|
for (auto& session : deqSessions) {
|
|
strSessionDenoms += (session.nSessionDenom ? CPrivateSend::GetDenominationsToString(session.nSessionDenom) : "N/A") + "; ";
|
|
}
|
|
return strSessionDenoms.empty() ? "N/A" : strSessionDenoms;
|
|
}
|
|
|
|
bool CPrivateSendClientSession::GetMixingMasternodeInfo(CDeterministicMNCPtr& ret) const
|
|
{
|
|
ret = mixingMasternode;
|
|
return ret != nullptr;
|
|
}
|
|
|
|
bool CPrivateSendClientManager::GetMixingMasternodesInfo(std::vector<CDeterministicMNCPtr>& vecDmnsRet) const
|
|
{
|
|
LOCK(cs_deqsessions);
|
|
for (const auto& session : deqSessions) {
|
|
CDeterministicMNCPtr dmn;
|
|
if (session.GetMixingMasternodeInfo(dmn)) {
|
|
vecDmnsRet.push_back(dmn);
|
|
}
|
|
}
|
|
return !vecDmnsRet.empty();
|
|
}
|
|
|
|
//
|
|
// Check the mixing progress and send client updates if a Masternode
|
|
//
|
|
void CPrivateSendClientSession::CheckPool()
|
|
{
|
|
// reset if we're here for 10 seconds
|
|
if ((nState == POOL_STATE_ERROR || nState == POOL_STATE_SUCCESS) && GetTime() - nTimeLastSuccessfulStep >= 10) {
|
|
LogPrint("privatesend", "CPrivateSendClientSession::CheckPool -- timeout, RESETTING\n");
|
|
UnlockCoins();
|
|
if (nState == POOL_STATE_ERROR) {
|
|
keyHolderStorage.ReturnAll();
|
|
} else {
|
|
keyHolderStorage.KeepAll();
|
|
}
|
|
SetNull();
|
|
}
|
|
}
|
|
|
|
//
|
|
// Check session timeouts
|
|
//
|
|
bool CPrivateSendClientSession::CheckTimeout()
|
|
{
|
|
if (fMasternodeMode) return false;
|
|
|
|
// catching hanging sessions
|
|
switch (nState) {
|
|
case POOL_STATE_ERROR:
|
|
LogPrint("privatesend", "CPrivateSendClientSession::CheckTimeout -- Pool error -- Running CheckPool\n");
|
|
CheckPool();
|
|
break;
|
|
case POOL_STATE_SUCCESS:
|
|
LogPrint("privatesend", "CPrivateSendClientSession::CheckTimeout -- Pool success -- Running CheckPool\n");
|
|
CheckPool();
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
int nLagTime = 10; // give the server a few extra seconds before resetting.
|
|
int nTimeout = (nState == POOL_STATE_SIGNING) ? PRIVATESEND_SIGNING_TIMEOUT : PRIVATESEND_QUEUE_TIMEOUT;
|
|
bool fTimeout = GetTime() - nTimeLastSuccessfulStep >= nTimeout + nLagTime;
|
|
|
|
if (nState == POOL_STATE_IDLE || !fTimeout)
|
|
return false;
|
|
|
|
LogPrint("privatesend", "CPrivateSendClientSession::CheckTimeout -- %s timed out (%ds) -- resetting\n",
|
|
(nState == POOL_STATE_SIGNING) ? "Signing" : "Session", nTimeout);
|
|
UnlockCoins();
|
|
keyHolderStorage.ReturnAll();
|
|
SetNull();
|
|
SetState(POOL_STATE_ERROR);
|
|
|
|
return true;
|
|
}
|
|
|
|
//
|
|
// Check all queues and sessions for timeouts
|
|
//
|
|
void CPrivateSendClientManager::CheckTimeout()
|
|
{
|
|
if (fMasternodeMode) return;
|
|
|
|
CheckQueue();
|
|
|
|
if (!fEnablePrivateSend) return;
|
|
|
|
LOCK(cs_deqsessions);
|
|
for (auto& session : deqSessions) {
|
|
if (session.CheckTimeout()) {
|
|
strAutoDenomResult = _("Session timed out.");
|
|
}
|
|
}
|
|
}
|
|
|
|
//
|
|
// Execute a mixing denomination via a Masternode.
|
|
// This is only ran from clients
|
|
//
|
|
bool CPrivateSendClientSession::SendDenominate(const std::vector<std::pair<CTxDSIn, CTxOut> >& vecPSInOutPairsIn, CConnman& connman)
|
|
{
|
|
if (fMasternodeMode) {
|
|
LogPrintf("CPrivateSendClientSession::SendDenominate -- PrivateSend from a Masternode is not supported currently.\n");
|
|
return false;
|
|
}
|
|
|
|
if (txMyCollateral == CMutableTransaction()) {
|
|
LogPrintf("CPrivateSendClient:SendDenominate -- PrivateSend collateral not set\n");
|
|
return false;
|
|
}
|
|
|
|
// lock the funds we're going to use
|
|
for (const auto& txin : txMyCollateral.vin)
|
|
vecOutPointLocked.push_back(txin.prevout);
|
|
|
|
for (const auto& pair : vecPSInOutPairsIn)
|
|
vecOutPointLocked.push_back(pair.first.prevout);
|
|
|
|
// we should already be connected to a Masternode
|
|
if (!nSessionID) {
|
|
LogPrintf("CPrivateSendClientSession::SendDenominate -- No Masternode has been selected yet.\n");
|
|
UnlockCoins();
|
|
keyHolderStorage.ReturnAll();
|
|
SetNull();
|
|
return false;
|
|
}
|
|
|
|
if (!CheckDiskSpace()) {
|
|
UnlockCoins();
|
|
keyHolderStorage.ReturnAll();
|
|
SetNull();
|
|
LogPrintf("CPrivateSendClientSession::SendDenominate -- Not enough disk space.\n");
|
|
return false;
|
|
}
|
|
|
|
SetState(POOL_STATE_ACCEPTING_ENTRIES);
|
|
strLastMessage = "";
|
|
|
|
LogPrintf("CPrivateSendClientSession::SendDenominate -- Added transaction to pool.\n");
|
|
|
|
CMutableTransaction tx; // for debug purposes only
|
|
std::vector<CTxDSIn> vecTxDSInTmp;
|
|
std::vector<CTxOut> vecTxOutTmp;
|
|
|
|
for (const auto& pair : vecPSInOutPairsIn) {
|
|
vecTxDSInTmp.emplace_back(pair.first);
|
|
vecTxOutTmp.emplace_back(pair.second);
|
|
tx.vin.emplace_back(pair.first);
|
|
tx.vout.emplace_back(pair.second);
|
|
}
|
|
|
|
LogPrintf("CPrivateSendClientSession::SendDenominate -- Submitting partial tx %s", tx.ToString());
|
|
|
|
// store our entry for later use
|
|
vecEntries.emplace_back(vecTxDSInTmp, vecTxOutTmp, txMyCollateral);
|
|
RelayIn(vecEntries.back(), connman);
|
|
nTimeLastSuccessfulStep = GetTime();
|
|
|
|
return true;
|
|
}
|
|
|
|
// Incoming message from Masternode updating the progress of mixing
|
|
bool CPrivateSendClientSession::CheckPoolStateUpdate(PoolState nStateNew, int nEntriesCountNew, PoolStatusUpdate nStatusUpdate, PoolMessage nMessageID, int nSessionIDNew)
|
|
{
|
|
if (fMasternodeMode) return false;
|
|
|
|
// do not update state when mixing client state is one of these
|
|
if (nState == POOL_STATE_IDLE || nState == POOL_STATE_ERROR || nState == POOL_STATE_SUCCESS) return false;
|
|
|
|
strAutoDenomResult = _("Masternode:") + " " + CPrivateSend::GetMessageByID(nMessageID);
|
|
|
|
// if rejected at any state
|
|
if (nStatusUpdate == STATUS_REJECTED) {
|
|
LogPrintf("CPrivateSendClientSession::CheckPoolStateUpdate -- entry is rejected by Masternode\n");
|
|
UnlockCoins();
|
|
keyHolderStorage.ReturnAll();
|
|
SetNull();
|
|
SetState(POOL_STATE_ERROR);
|
|
strLastMessage = CPrivateSend::GetMessageByID(nMessageID);
|
|
return true;
|
|
}
|
|
|
|
if (nStatusUpdate == STATUS_ACCEPTED && nState == nStateNew) {
|
|
if (nStateNew == POOL_STATE_QUEUE && nSessionID == 0 && nSessionIDNew != 0) {
|
|
// new session id should be set only in POOL_STATE_QUEUE state
|
|
nSessionID = nSessionIDNew;
|
|
nTimeLastSuccessfulStep = GetTime();
|
|
LogPrintf("CPrivateSendClientSession::CheckPoolStateUpdate -- set nSessionID to %d\n", nSessionID);
|
|
return true;
|
|
} else if (nStateNew == POOL_STATE_ACCEPTING_ENTRIES && nEntriesCount != nEntriesCountNew) {
|
|
nEntriesCount = nEntriesCountNew;
|
|
nTimeLastSuccessfulStep = GetTime();
|
|
fLastEntryAccepted = true;
|
|
LogPrintf("CPrivateSendClientSession::CheckPoolStateUpdate -- new entry accepted!\n");
|
|
return true;
|
|
}
|
|
}
|
|
|
|
// only situations above are allowed, fail in any other case
|
|
return false;
|
|
}
|
|
|
|
//
|
|
// After we receive the finalized transaction from the Masternode, we must
|
|
// check it to make sure it's what we want, then sign it if we agree.
|
|
// If we refuse to sign, it's possible we'll be charged collateral
|
|
//
|
|
bool CPrivateSendClientSession::SignFinalTransaction(const CTransaction& finalTransactionNew, CNode* pnode, CConnman& connman)
|
|
{
|
|
if (!pwalletMain) return false;
|
|
|
|
if (fMasternodeMode || pnode == nullptr) return false;
|
|
if (!mixingMasternode) return false;
|
|
|
|
finalMutableTransaction = finalTransactionNew;
|
|
LogPrintf("CPrivateSendClientSession::SignFinalTransaction -- finalMutableTransaction=%s", finalMutableTransaction.ToString());
|
|
|
|
// Make sure it's BIP69 compliant
|
|
sort(finalMutableTransaction.vin.begin(), finalMutableTransaction.vin.end(), CompareInputBIP69());
|
|
sort(finalMutableTransaction.vout.begin(), finalMutableTransaction.vout.end(), CompareOutputBIP69());
|
|
|
|
if (finalMutableTransaction.GetHash() != finalTransactionNew.GetHash()) {
|
|
LogPrintf("CPrivateSendClientSession::SignFinalTransaction -- WARNING! Masternode %s is not BIP69 compliant!\n", mixingMasternode->proTxHash.ToString());
|
|
UnlockCoins();
|
|
keyHolderStorage.ReturnAll();
|
|
SetNull();
|
|
return false;
|
|
}
|
|
|
|
std::vector<CTxIn> sigs;
|
|
|
|
//make sure my inputs/outputs are present, otherwise refuse to sign
|
|
for (const auto& entry : vecEntries) {
|
|
for (const auto& txdsin : entry.vecTxDSIn) {
|
|
/* Sign my transaction and all outputs */
|
|
int nMyInputIndex = -1;
|
|
CScript prevPubKey = CScript();
|
|
CTxIn txin = CTxIn();
|
|
|
|
for (unsigned int i = 0; i < finalMutableTransaction.vin.size(); i++) {
|
|
if (finalMutableTransaction.vin[i] == txdsin) {
|
|
nMyInputIndex = i;
|
|
prevPubKey = txdsin.prevPubKey;
|
|
txin = txdsin;
|
|
}
|
|
}
|
|
|
|
if (nMyInputIndex >= 0) { //might have to do this one input at a time?
|
|
int nFoundOutputsCount = 0;
|
|
CAmount nValue1 = 0;
|
|
CAmount nValue2 = 0;
|
|
|
|
for (const auto& txoutFinal : finalMutableTransaction.vout) {
|
|
for (const auto& txout : entry.vecTxOut) {
|
|
if (txoutFinal == txout) {
|
|
nFoundOutputsCount++;
|
|
nValue1 += txoutFinal.nValue;
|
|
}
|
|
}
|
|
}
|
|
|
|
for (const auto& txout : entry.vecTxOut)
|
|
nValue2 += txout.nValue;
|
|
|
|
int nTargetOuputsCount = entry.vecTxOut.size();
|
|
if (nFoundOutputsCount < nTargetOuputsCount || nValue1 != nValue2) {
|
|
// in this case, something went wrong and we'll refuse to sign. It's possible we'll be charged collateral. But that's
|
|
// better then signing if the transaction doesn't look like what we wanted.
|
|
LogPrintf("CPrivateSendClientSession::SignFinalTransaction -- My entries are not correct! Refusing to sign: nFoundOutputsCount: %d, nTargetOuputsCount: %d\n", nFoundOutputsCount, nTargetOuputsCount);
|
|
UnlockCoins();
|
|
keyHolderStorage.ReturnAll();
|
|
SetNull();
|
|
|
|
return false;
|
|
}
|
|
|
|
const CKeyStore& keystore = *pwalletMain;
|
|
|
|
LogPrint("privatesend", "CPrivateSendClientSession::SignFinalTransaction -- Signing my input %i\n", nMyInputIndex);
|
|
if (!SignSignature(keystore, prevPubKey, finalMutableTransaction, nMyInputIndex, int(SIGHASH_ALL | SIGHASH_ANYONECANPAY))) { // changes scriptSig
|
|
LogPrint("privatesend", "CPrivateSendClientSession::SignFinalTransaction -- Unable to sign my own transaction!\n");
|
|
// not sure what to do here, it will timeout...?
|
|
}
|
|
|
|
sigs.push_back(finalMutableTransaction.vin[nMyInputIndex]);
|
|
LogPrint("privatesend", "CPrivateSendClientSession::SignFinalTransaction -- nMyInputIndex: %d, sigs.size(): %d, scriptSig=%s\n", nMyInputIndex, (int)sigs.size(), ScriptToAsmStr(finalMutableTransaction.vin[nMyInputIndex].scriptSig));
|
|
}
|
|
}
|
|
}
|
|
|
|
if (sigs.empty()) {
|
|
LogPrintf("CPrivateSendClientSession::SignFinalTransaction -- can't sign anything!\n");
|
|
UnlockCoins();
|
|
keyHolderStorage.ReturnAll();
|
|
SetNull();
|
|
|
|
return false;
|
|
}
|
|
|
|
// push all of our signatures to the Masternode
|
|
LogPrintf("CPrivateSendClientSession::SignFinalTransaction -- pushing sigs to the masternode, finalMutableTransaction=%s", finalMutableTransaction.ToString());
|
|
CNetMsgMaker msgMaker(pnode->GetSendVersion());
|
|
connman.PushMessage(pnode, msgMaker.Make(NetMsgType::DSSIGNFINALTX, sigs));
|
|
SetState(POOL_STATE_SIGNING);
|
|
nTimeLastSuccessfulStep = GetTime();
|
|
|
|
return true;
|
|
}
|
|
|
|
// mixing transaction was completed (failed or successful)
|
|
void CPrivateSendClientSession::CompletedTransaction(PoolMessage nMessageID)
|
|
{
|
|
if (fMasternodeMode) return;
|
|
|
|
if (nMessageID == MSG_SUCCESS) {
|
|
LogPrintf("CompletedTransaction -- success\n");
|
|
privateSendClient.UpdatedSuccessBlock();
|
|
keyHolderStorage.KeepAll();
|
|
} else {
|
|
LogPrintf("CompletedTransaction -- error\n");
|
|
keyHolderStorage.ReturnAll();
|
|
}
|
|
UnlockCoins();
|
|
SetNull();
|
|
strLastMessage = CPrivateSend::GetMessageByID(nMessageID);
|
|
}
|
|
|
|
void CPrivateSendClientManager::UpdatedSuccessBlock()
|
|
{
|
|
if (fMasternodeMode) return;
|
|
nCachedLastSuccessBlock = nCachedBlockHeight;
|
|
}
|
|
|
|
bool CPrivateSendClientManager::IsDenomSkipped(const CAmount& nDenomValue)
|
|
{
|
|
return std::find(vecDenominationsSkipped.begin(), vecDenominationsSkipped.end(), nDenomValue) != vecDenominationsSkipped.end();
|
|
}
|
|
|
|
void CPrivateSendClientManager::AddSkippedDenom(const CAmount& nDenomValue)
|
|
{
|
|
vecDenominationsSkipped.push_back(nDenomValue);
|
|
}
|
|
|
|
bool CPrivateSendClientManager::WaitForAnotherBlock()
|
|
{
|
|
if (!masternodeSync.IsBlockchainSynced())
|
|
return true;
|
|
|
|
if (fPrivateSendMultiSession)
|
|
return false;
|
|
|
|
return nCachedBlockHeight - nCachedLastSuccessBlock < nMinBlocksToWait;
|
|
}
|
|
|
|
bool CPrivateSendClientManager::CheckAutomaticBackup()
|
|
{
|
|
if (!pwalletMain) {
|
|
LogPrint("privatesend", "CPrivateSendClientManager::CheckAutomaticBackup -- Wallet is not initialized, no mixing available.\n");
|
|
strAutoDenomResult = _("Wallet is not initialized") + ", " + _("no mixing available.");
|
|
fEnablePrivateSend = false; // no mixing
|
|
return false;
|
|
}
|
|
|
|
switch (nWalletBackups) {
|
|
case 0:
|
|
LogPrint("privatesend", "CPrivateSendClientManager::CheckAutomaticBackup -- Automatic backups disabled, no mixing available.\n");
|
|
strAutoDenomResult = _("Automatic backups disabled") + ", " + _("no mixing available.");
|
|
fEnablePrivateSend = false; // stop mixing
|
|
pwalletMain->nKeysLeftSinceAutoBackup = 0; // no backup, no "keys since last backup"
|
|
return false;
|
|
case -1:
|
|
// Automatic backup failed, nothing else we can do until user fixes the issue manually.
|
|
// There is no way to bring user attention in daemon mode so we just update status and
|
|
// keep spamming if debug is on.
|
|
LogPrint("privatesend", "CPrivateSendClientManager::CheckAutomaticBackup -- ERROR! Failed to create automatic backup.\n");
|
|
strAutoDenomResult = _("ERROR! Failed to create automatic backup") + ", " + _("see debug.log for details.");
|
|
return false;
|
|
case -2:
|
|
// We were able to create automatic backup but keypool was not replenished because wallet is locked.
|
|
// There is no way to bring user attention in daemon mode so we just update status and
|
|
// keep spamming if debug is on.
|
|
LogPrint("privatesend", "CPrivateSendClientManager::CheckAutomaticBackup -- WARNING! Failed to create replenish keypool, please unlock your wallet to do so.\n");
|
|
strAutoDenomResult = _("WARNING! Failed to replenish keypool, please unlock your wallet to do so.") + ", " + _("see debug.log for details.");
|
|
return false;
|
|
}
|
|
|
|
if (pwalletMain->nKeysLeftSinceAutoBackup < PRIVATESEND_KEYS_THRESHOLD_STOP) {
|
|
// We should never get here via mixing itself but probably smth else is still actively using keypool
|
|
LogPrint("privatesend", "CPrivateSendClientManager::CheckAutomaticBackup -- Very low number of keys left: %d, no mixing available.\n", pwalletMain->nKeysLeftSinceAutoBackup);
|
|
strAutoDenomResult = strprintf(_("Very low number of keys left: %d") + ", " + _("no mixing available."), pwalletMain->nKeysLeftSinceAutoBackup);
|
|
// It's getting really dangerous, stop mixing
|
|
fEnablePrivateSend = false;
|
|
return false;
|
|
} else if (pwalletMain->nKeysLeftSinceAutoBackup < PRIVATESEND_KEYS_THRESHOLD_WARNING) {
|
|
// Low number of keys left but it's still more or less safe to continue
|
|
LogPrint("privatesend", "CPrivateSendClientManager::CheckAutomaticBackup -- Very low number of keys left: %d\n", pwalletMain->nKeysLeftSinceAutoBackup);
|
|
strAutoDenomResult = strprintf(_("Very low number of keys left: %d"), pwalletMain->nKeysLeftSinceAutoBackup);
|
|
|
|
if (fCreateAutoBackups) {
|
|
LogPrint("privatesend", "CPrivateSendClientManager::CheckAutomaticBackup -- Trying to create new backup.\n");
|
|
std::string warningString;
|
|
std::string errorString;
|
|
|
|
if (!AutoBackupWallet(pwalletMain, "", warningString, errorString)) {
|
|
if (!warningString.empty()) {
|
|
// There were some issues saving backup but yet more or less safe to continue
|
|
LogPrintf("CPrivateSendClientManager::CheckAutomaticBackup -- WARNING! Something went wrong on automatic backup: %s\n", warningString);
|
|
}
|
|
if (!errorString.empty()) {
|
|
// Things are really broken
|
|
LogPrintf("CPrivateSendClientManager::CheckAutomaticBackup -- ERROR! Failed to create automatic backup: %s\n", errorString);
|
|
strAutoDenomResult = strprintf(_("ERROR! Failed to create automatic backup") + ": %s", errorString);
|
|
return false;
|
|
}
|
|
}
|
|
} else {
|
|
// Wait for smth else (e.g. GUI action) to create automatic backup for us
|
|
return false;
|
|
}
|
|
}
|
|
|
|
LogPrint("privatesend", "CPrivateSendClientManager::CheckAutomaticBackup -- Keys left since latest backup: %d\n", pwalletMain->nKeysLeftSinceAutoBackup);
|
|
|
|
return true;
|
|
}
|
|
|
|
//
|
|
// Passively run mixing in the background to anonymize funds based on the given configuration.
|
|
//
|
|
bool CPrivateSendClientSession::DoAutomaticDenominating(CConnman& connman, bool fDryRun)
|
|
{
|
|
if (fMasternodeMode) return false; // no client-side mixing on masternodes
|
|
if (nState != POOL_STATE_IDLE) return false;
|
|
|
|
if (!masternodeSync.IsBlockchainSynced()) {
|
|
strAutoDenomResult = _("Can't mix while sync in progress.");
|
|
return false;
|
|
}
|
|
|
|
if (!pwalletMain) {
|
|
strAutoDenomResult = _("Wallet is not initialized");
|
|
return false;
|
|
}
|
|
|
|
CAmount nBalanceNeedsAnonymized;
|
|
CAmount nValueMin = CPrivateSend::GetSmallestDenomination();
|
|
|
|
{
|
|
LOCK2(cs_main, pwalletMain->cs_wallet);
|
|
|
|
if (!fDryRun && pwalletMain->IsLocked(true)) {
|
|
strAutoDenomResult = _("Wallet is locked.");
|
|
return false;
|
|
}
|
|
|
|
if (GetEntriesCount() > 0) {
|
|
strAutoDenomResult = _("Mixing in progress...");
|
|
return false;
|
|
}
|
|
|
|
TRY_LOCK(cs_privatesend, lockDS);
|
|
if (!lockDS) {
|
|
strAutoDenomResult = _("Lock is already in place.");
|
|
return false;
|
|
}
|
|
|
|
if (deterministicMNManager->GetListAtChainTip().GetValidMNsCount() == 0) {
|
|
LogPrint("privatesend", "CPrivateSendClientSession::DoAutomaticDenominating -- No Masternodes detected\n");
|
|
strAutoDenomResult = _("No Masternodes detected.");
|
|
return false;
|
|
}
|
|
|
|
// if there are no confirmed DS collateral inputs yet
|
|
if (!pwalletMain->HasCollateralInputs()) {
|
|
// should have some additional amount for them
|
|
nValueMin += CPrivateSend::GetMaxCollateralAmount();
|
|
}
|
|
|
|
// including denoms but applying some restrictions
|
|
nBalanceNeedsAnonymized = pwalletMain->GetNeedsToBeAnonymizedBalance(nValueMin);
|
|
|
|
// anonymizable balance is way too small
|
|
if (nBalanceNeedsAnonymized < nValueMin) {
|
|
LogPrintf("CPrivateSendClientSession::DoAutomaticDenominating -- Not enough funds to anonymize\n");
|
|
strAutoDenomResult = _("Not enough funds to anonymize.");
|
|
return false;
|
|
}
|
|
|
|
// excluding denoms
|
|
CAmount nBalanceAnonimizableNonDenom = pwalletMain->GetAnonymizableBalance(true);
|
|
// denoms
|
|
CAmount nBalanceDenominatedConf = pwalletMain->GetDenominatedBalance();
|
|
CAmount nBalanceDenominatedUnconf = pwalletMain->GetDenominatedBalance(true);
|
|
CAmount nBalanceDenominated = nBalanceDenominatedConf + nBalanceDenominatedUnconf;
|
|
|
|
LogPrint("privatesend", "CPrivateSendClientSession::DoAutomaticDenominating -- nValueMin: %f, nBalanceNeedsAnonymized: %f, nBalanceAnonimizableNonDenom: %f, nBalanceDenominatedConf: %f, nBalanceDenominatedUnconf: %f, nBalanceDenominated: %f\n",
|
|
(float)nValueMin / COIN,
|
|
(float)nBalanceNeedsAnonymized / COIN,
|
|
(float)nBalanceAnonimizableNonDenom / COIN,
|
|
(float)nBalanceDenominatedConf / COIN,
|
|
(float)nBalanceDenominatedUnconf / COIN,
|
|
(float)nBalanceDenominated / COIN);
|
|
|
|
if (fDryRun) return true;
|
|
|
|
// Check if we have should create more denominated inputs i.e.
|
|
// there are funds to denominate and denominated balance does not exceed
|
|
// max amount to mix yet.
|
|
if (nBalanceAnonimizableNonDenom >= nValueMin + CPrivateSend::GetCollateralAmount() && nBalanceDenominated < privateSendClient.nPrivateSendAmount * COIN)
|
|
return CreateDenominated(connman);
|
|
|
|
//check if we have the collateral sized inputs
|
|
if (!pwalletMain->HasCollateralInputs())
|
|
return !pwalletMain->HasCollateralInputs(false) && MakeCollateralAmounts(connman);
|
|
|
|
if (nSessionID) {
|
|
strAutoDenomResult = _("Mixing in progress...");
|
|
return false;
|
|
}
|
|
|
|
// Initial phase, find a Masternode
|
|
// Clean if there is anything left from previous session
|
|
UnlockCoins();
|
|
keyHolderStorage.ReturnAll();
|
|
SetNull();
|
|
|
|
// should be no unconfirmed denoms in non-multi-session mode
|
|
if (!privateSendClient.fPrivateSendMultiSession && nBalanceDenominatedUnconf > 0) {
|
|
LogPrintf("CPrivateSendClientSession::DoAutomaticDenominating -- Found unconfirmed denominated outputs, will wait till they confirm to continue.\n");
|
|
strAutoDenomResult = _("Found unconfirmed denominated outputs, will wait till they confirm to continue.");
|
|
return false;
|
|
}
|
|
|
|
//check our collateral and create new if needed
|
|
std::string strReason;
|
|
if (txMyCollateral == CMutableTransaction()) {
|
|
if (!pwalletMain->CreateCollateralTransaction(txMyCollateral, strReason)) {
|
|
LogPrintf("CPrivateSendClientSession::DoAutomaticDenominating -- create collateral error:%s\n", strReason);
|
|
return false;
|
|
}
|
|
} else {
|
|
if (!CPrivateSend::IsCollateralValid(txMyCollateral)) {
|
|
LogPrintf("CPrivateSendClientSession::DoAutomaticDenominating -- invalid collateral, recreating...\n");
|
|
if (!pwalletMain->CreateCollateralTransaction(txMyCollateral, strReason)) {
|
|
LogPrintf("CPrivateSendClientSession::DoAutomaticDenominating -- create collateral error: %s\n", strReason);
|
|
return false;
|
|
}
|
|
}
|
|
}
|
|
} // LOCK2(cs_main, pwalletMain->cs_wallet);
|
|
|
|
bool fUseQueue = GetRandInt(100) > 33;
|
|
// don't use the queues all of the time for mixing unless we are a liquidity provider
|
|
if ((privateSendClient.nLiquidityProvider || fUseQueue) && JoinExistingQueue(nBalanceNeedsAnonymized, connman))
|
|
return true;
|
|
|
|
// do not initiate queue if we are a liquidity provider to avoid useless inter-mixing
|
|
if (privateSendClient.nLiquidityProvider) return false;
|
|
|
|
if (StartNewQueue(nValueMin, nBalanceNeedsAnonymized, connman))
|
|
return true;
|
|
|
|
strAutoDenomResult = _("No compatible Masternode found.");
|
|
return false;
|
|
}
|
|
|
|
bool CPrivateSendClientManager::DoAutomaticDenominating(CConnman& connman, bool fDryRun)
|
|
{
|
|
if (fMasternodeMode) return false; // no client-side mixing on masternodes
|
|
if (!fEnablePrivateSend) return false;
|
|
|
|
if (!masternodeSync.IsBlockchainSynced()) {
|
|
strAutoDenomResult = _("Can't mix while sync in progress.");
|
|
return false;
|
|
}
|
|
|
|
if (!pwalletMain) {
|
|
strAutoDenomResult = _("Wallet is not initialized");
|
|
return false;
|
|
}
|
|
|
|
if (!fDryRun && pwalletMain->IsLocked(true)) {
|
|
strAutoDenomResult = _("Wallet is locked.");
|
|
return false;
|
|
}
|
|
|
|
int nMnCountEnabled = deterministicMNManager->GetListAtChainTip().GetValidMNsCount();
|
|
|
|
// If we've used 90% of the Masternode list then drop the oldest first ~30%
|
|
int nThreshold_high = nMnCountEnabled * 0.9;
|
|
int nThreshold_low = nThreshold_high * 0.7;
|
|
LogPrint("privatesend", "Checking vecMasternodesUsed: size: %d, threshold: %d\n", (int)vecMasternodesUsed.size(), nThreshold_high);
|
|
|
|
if ((int)vecMasternodesUsed.size() > nThreshold_high) {
|
|
vecMasternodesUsed.erase(vecMasternodesUsed.begin(), vecMasternodesUsed.begin() + vecMasternodesUsed.size() - nThreshold_low);
|
|
LogPrint("privatesend", " vecMasternodesUsed: new size: %d, threshold: %d\n", (int)vecMasternodesUsed.size(), nThreshold_high);
|
|
}
|
|
|
|
LOCK(cs_deqsessions);
|
|
bool fResult = true;
|
|
if ((int)deqSessions.size() < nPrivateSendSessions) {
|
|
deqSessions.emplace_back();
|
|
}
|
|
for (auto& session : deqSessions) {
|
|
if (!CheckAutomaticBackup())
|
|
return false;
|
|
|
|
if (WaitForAnotherBlock()) {
|
|
LogPrintf("CPrivateSendClientManager::DoAutomaticDenominating -- Last successful PrivateSend action was too recent\n");
|
|
strAutoDenomResult = _("Last successful PrivateSend action was too recent.");
|
|
return false;
|
|
}
|
|
|
|
fResult &= session.DoAutomaticDenominating(connman, fDryRun);
|
|
}
|
|
|
|
return fResult;
|
|
}
|
|
|
|
void CPrivateSendClientManager::AddUsedMasternode(const COutPoint& outpointMn)
|
|
{
|
|
vecMasternodesUsed.push_back(outpointMn);
|
|
}
|
|
|
|
CDeterministicMNCPtr CPrivateSendClientManager::GetRandomNotUsedMasternode()
|
|
{
|
|
auto mnList = deterministicMNManager->GetListAtChainTip();
|
|
|
|
int nCountEnabled = mnList.GetValidMNsCount();
|
|
int nCountNotExcluded = nCountEnabled - vecMasternodesUsed.size();
|
|
|
|
LogPrintf("CPrivateSendClientManager::%s -- %d enabled masternodes, %d masternodes to choose from\n", __func__, nCountEnabled, nCountNotExcluded);
|
|
if(nCountNotExcluded < 1) {
|
|
return nullptr;
|
|
}
|
|
|
|
// fill a vector
|
|
std::vector<CDeterministicMNCPtr> vpMasternodesShuffled;
|
|
vpMasternodesShuffled.reserve((size_t)nCountEnabled);
|
|
mnList.ForEachMN(true, [&](const CDeterministicMNCPtr& dmn) {
|
|
vpMasternodesShuffled.emplace_back(dmn);
|
|
});
|
|
|
|
FastRandomContext insecure_rand;
|
|
// shuffle pointers
|
|
std::random_shuffle(vpMasternodesShuffled.begin(), vpMasternodesShuffled.end(), insecure_rand);
|
|
|
|
std::set<COutPoint> excludeSet(vecMasternodesUsed.begin(), vecMasternodesUsed.end());
|
|
|
|
// loop through
|
|
for (const auto& dmn : vpMasternodesShuffled) {
|
|
if (excludeSet.count(dmn->collateralOutpoint)) {
|
|
continue;
|
|
}
|
|
|
|
LogPrint("masternode", "CPrivateSendClientManager::%s -- found, masternode=%s\n", __func__, dmn->collateralOutpoint.ToStringShort());
|
|
return dmn;
|
|
}
|
|
|
|
LogPrint("masternode", "CPrivateSendClientManager::%s -- failed\n", __func__);
|
|
return nullptr;
|
|
}
|
|
|
|
bool CPrivateSendClientSession::JoinExistingQueue(CAmount nBalanceNeedsAnonymized, CConnman& connman)
|
|
{
|
|
if (!pwalletMain) return false;
|
|
|
|
auto mnList = deterministicMNManager->GetListAtChainTip();
|
|
|
|
std::vector<CAmount> vecStandardDenoms = CPrivateSend::GetStandardDenominations();
|
|
// Look through the queues and see if anything matches
|
|
CPrivateSendQueue dsq;
|
|
while (privateSendClient.GetQueueItemAndTry(dsq)) {
|
|
auto dmn = mnList.GetValidMNByCollateral(dsq.masternodeOutpoint);
|
|
|
|
if (!dmn) {
|
|
LogPrintf("CPrivateSendClientSession::JoinExistingQueue -- dsq masternode is not in masternode list, masternode=%s\n", dsq.masternodeOutpoint.ToStringShort());
|
|
continue;
|
|
}
|
|
|
|
// skip next mn payments winners
|
|
if (mnpayments.IsScheduled(dmn, 0)) {
|
|
LogPrintf("CPrivateSendClientSession::JoinExistingQueue -- skipping winner, masternode=%s\n", dmn->proTxHash.ToString());
|
|
continue;
|
|
}
|
|
|
|
std::vector<int> vecBits;
|
|
if (!CPrivateSend::GetDenominationsBits(dsq.nDenom, vecBits)) {
|
|
// incompatible denom
|
|
continue;
|
|
}
|
|
|
|
// mixing rate limit i.e. nLastDsq check should already pass in DSQUEUE ProcessMessage
|
|
// in order for dsq to get into vecPrivateSendQueue, so we should be safe to mix already,
|
|
// no need for additional verification here
|
|
|
|
LogPrint("privatesend", "CPrivateSendClientSession::JoinExistingQueue -- found valid queue: %s\n", dsq.ToString());
|
|
|
|
std::vector<std::pair<CTxDSIn, CTxOut> > vecPSInOutPairsTmp;
|
|
CAmount nMinAmount = vecStandardDenoms[vecBits.front()];
|
|
CAmount nMaxAmount = nBalanceNeedsAnonymized;
|
|
|
|
// Try to match their denominations if possible, select exact number of denominations
|
|
if (!pwalletMain->SelectPSInOutPairsByDenominations(dsq.nDenom, nMinAmount, nMaxAmount, vecPSInOutPairsTmp)) {
|
|
LogPrintf("CPrivateSendClientSession::JoinExistingQueue -- Couldn't match %d denominations %d (%s)\n", vecBits.front(), dsq.nDenom, CPrivateSend::GetDenominationsToString(dsq.nDenom));
|
|
continue;
|
|
}
|
|
|
|
privateSendClient.AddUsedMasternode(dsq.masternodeOutpoint);
|
|
|
|
if (connman.IsMasternodeOrDisconnectRequested(dmn->pdmnState->addr)) {
|
|
LogPrintf("CPrivateSendClientSession::JoinExistingQueue -- skipping masternode connection, addr=%s\n", dmn->pdmnState->addr.ToString());
|
|
continue;
|
|
}
|
|
|
|
nSessionDenom = dsq.nDenom;
|
|
mixingMasternode = dmn;
|
|
pendingDsaRequest = CPendingDsaRequest(dmn->pdmnState->addr, CPrivateSendAccept(nSessionDenom, txMyCollateral));
|
|
connman.AddPendingMasternode(dmn->pdmnState->addr);
|
|
// TODO: add new state POOL_STATE_CONNECTING and bump MIN_PRIVATESEND_PEER_PROTO_VERSION
|
|
SetState(POOL_STATE_QUEUE);
|
|
nTimeLastSuccessfulStep = GetTime();
|
|
LogPrintf("CPrivateSendClientSession::JoinExistingQueue -- pending connection (from queue): nSessionDenom: %d (%s), addr=%s\n",
|
|
nSessionDenom, CPrivateSend::GetDenominationsToString(nSessionDenom), dmn->pdmnState->addr.ToString());
|
|
strAutoDenomResult = _("Trying to connect...");
|
|
return true;
|
|
}
|
|
strAutoDenomResult = _("Failed to find mixing queue to join");
|
|
return false;
|
|
}
|
|
|
|
bool CPrivateSendClientSession::StartNewQueue(CAmount nValueMin, CAmount nBalanceNeedsAnonymized, CConnman& connman)
|
|
{
|
|
if (!pwalletMain) return false;
|
|
|
|
int nTries = 0;
|
|
int nMnCount = deterministicMNManager->GetListAtChainTip().GetValidMNsCount();
|
|
|
|
// ** find the coins we'll use
|
|
std::vector<CTxIn> vecTxIn;
|
|
CAmount nValueInTmp = 0;
|
|
if (!pwalletMain->SelectPrivateCoins(nValueMin, nBalanceNeedsAnonymized, vecTxIn, nValueInTmp, 0, privateSendClient.nPrivateSendRounds - 1)) {
|
|
// this should never happen
|
|
LogPrintf("CPrivateSendClientSession::StartNewQueue -- Can't mix: no compatible inputs found!\n");
|
|
strAutoDenomResult = _("Can't mix: no compatible inputs found!");
|
|
return false;
|
|
}
|
|
|
|
// otherwise, try one randomly
|
|
while (nTries < 10) {
|
|
auto dmn = privateSendClient.GetRandomNotUsedMasternode();
|
|
|
|
if (!dmn) {
|
|
LogPrintf("CPrivateSendClientSession::StartNewQueue -- Can't find random masternode!\n");
|
|
strAutoDenomResult = _("Can't find random Masternode.");
|
|
return false;
|
|
}
|
|
|
|
privateSendClient.AddUsedMasternode(dmn->collateralOutpoint);
|
|
|
|
// skip next mn payments winners
|
|
if (mnpayments.IsScheduled(dmn, 0)) {
|
|
LogPrintf("CPrivateSendClientSession::StartNewQueue -- skipping winner, masternode=%s\n", dmn->proTxHash.ToString());
|
|
nTries++;
|
|
continue;
|
|
}
|
|
|
|
int64_t nLastDsq = mmetaman.GetMetaInfo(dmn->proTxHash)->GetLastDsq();
|
|
if (nLastDsq != 0 && nLastDsq + nMnCount / 5 > mmetaman.GetDsqCount()) {
|
|
LogPrintf("CPrivateSendClientSession::StartNewQueue -- Too early to mix on this masternode!"
|
|
" masternode=%s addr=%s nLastDsq=%d CountEnabled/5=%d nDsqCount=%d\n",
|
|
dmn->proTxHash.ToString(), dmn->pdmnState->addr.ToString(), nLastDsq,
|
|
nMnCount / 5, mmetaman.GetDsqCount());
|
|
nTries++;
|
|
continue;
|
|
}
|
|
|
|
if (connman.IsMasternodeOrDisconnectRequested(dmn->pdmnState->addr)) {
|
|
LogPrintf("CPrivateSendClientSession::StartNewQueue -- skipping masternode connection, addr=%s\n", dmn->pdmnState->addr.ToString());
|
|
nTries++;
|
|
continue;
|
|
}
|
|
|
|
LogPrintf("CPrivateSendClientSession::StartNewQueue -- attempt %d connection to Masternode %s\n", nTries, dmn->pdmnState->addr.ToString());
|
|
|
|
std::vector<CAmount> vecAmounts;
|
|
pwalletMain->ConvertList(vecTxIn, vecAmounts);
|
|
// try to get a single random denom out of vecAmounts
|
|
while (nSessionDenom == 0) {
|
|
nSessionDenom = CPrivateSend::GetDenominationsByAmounts(vecAmounts);
|
|
}
|
|
|
|
mixingMasternode = dmn;
|
|
connman.AddPendingMasternode(dmn->pdmnState->addr);
|
|
pendingDsaRequest = CPendingDsaRequest(dmn->pdmnState->addr, CPrivateSendAccept(nSessionDenom, txMyCollateral));
|
|
// TODO: add new state POOL_STATE_CONNECTING and bump MIN_PRIVATESEND_PEER_PROTO_VERSION
|
|
SetState(POOL_STATE_QUEUE);
|
|
nTimeLastSuccessfulStep = GetTime();
|
|
LogPrintf("CPrivateSendClientSession::StartNewQueue -- pending connection, nSessionDenom: %d (%s), addr=%s\n",
|
|
nSessionDenom, CPrivateSend::GetDenominationsToString(nSessionDenom), dmn->pdmnState->addr.ToString());
|
|
strAutoDenomResult = _("Trying to connect...");
|
|
return true;
|
|
}
|
|
strAutoDenomResult = _("Failed to start a new mixing queue");
|
|
return false;
|
|
}
|
|
|
|
bool CPrivateSendClientSession::ProcessPendingDsaRequest(CConnman& connman)
|
|
{
|
|
if (!pendingDsaRequest) return false;
|
|
|
|
bool fDone = connman.ForNode(pendingDsaRequest.GetAddr(), [&](CNode* pnode) {
|
|
LogPrint("privatesend", "-- processing dsa queue for addr=%s\n", pnode->addr.ToString());
|
|
nTimeLastSuccessfulStep = GetTime();
|
|
// TODO: this vvvv should be here after new state POOL_STATE_CONNECTING is added and MIN_PRIVATESEND_PEER_PROTO_VERSION is bumped
|
|
// SetState(POOL_STATE_QUEUE);
|
|
CNetMsgMaker msgMaker(pnode->GetSendVersion());
|
|
connman.PushMessage(pnode, msgMaker.Make(NetMsgType::DSACCEPT, pendingDsaRequest.GetDSA()));
|
|
return true;
|
|
});
|
|
|
|
if (fDone) {
|
|
pendingDsaRequest = CPendingDsaRequest();
|
|
} else if (pendingDsaRequest.IsExpired()) {
|
|
LogPrint("privatesend", "CPrivateSendClientSession::%s -- failed to connect to %s\n", __func__, pendingDsaRequest.GetAddr().ToString());
|
|
SetNull();
|
|
}
|
|
|
|
return fDone;
|
|
}
|
|
|
|
void CPrivateSendClientManager::ProcessPendingDsaRequest(CConnman& connman)
|
|
{
|
|
LOCK(cs_deqsessions);
|
|
for (auto& session : deqSessions) {
|
|
if (session.ProcessPendingDsaRequest(connman)) {
|
|
strAutoDenomResult = _("Mixing in progress...");
|
|
}
|
|
}
|
|
}
|
|
|
|
bool CPrivateSendClientSession::SubmitDenominate(CConnman& connman)
|
|
{
|
|
LOCK2(cs_main, pwalletMain->cs_wallet);
|
|
|
|
std::string strError;
|
|
std::vector<std::pair<CTxDSIn, CTxOut> > vecPSInOutPairs, vecPSInOutPairsTmp;
|
|
|
|
if (!SelectDenominate(strError, vecPSInOutPairs)) {
|
|
LogPrintf("CPrivateSendClientSession::SubmitDenominate -- SelectDenominate failed, error: %s\n", strError);
|
|
return false;
|
|
}
|
|
|
|
std::vector<std::pair<int, size_t> > vecInputsByRounds;
|
|
// Note: liquidity providers are fine with whatever number of inputs they've got
|
|
bool fDryRun = privateSendClient.nLiquidityProvider == 0;
|
|
|
|
for (int i = 0; i < privateSendClient.nPrivateSendRounds; i++) {
|
|
if (PrepareDenominate(i, i, strError, vecPSInOutPairs, vecPSInOutPairsTmp, fDryRun)) {
|
|
LogPrintf("CPrivateSendClientSession::SubmitDenominate -- Running PrivateSend denominate for %d rounds, success\n", i);
|
|
if (!fDryRun) {
|
|
return SendDenominate(vecPSInOutPairsTmp, connman);
|
|
}
|
|
vecInputsByRounds.emplace_back(i, vecPSInOutPairsTmp.size());
|
|
} else {
|
|
LogPrint("privatesend", "CPrivateSendClientSession::SubmitDenominate -- Running PrivateSend denominate for %d rounds, error: %s\n", i, strError);
|
|
}
|
|
}
|
|
|
|
// more inputs first, for equal input count prefer the one with less rounds
|
|
std::sort(vecInputsByRounds.begin(), vecInputsByRounds.end(), [](const auto& a, const auto& b) {
|
|
return a.second > b.second || (a.second == b.second && a.first < b.first);
|
|
});
|
|
|
|
LogPrint("privatesend", "vecInputsByRounds for denom %d\n", nSessionDenom);
|
|
for (const auto& pair : vecInputsByRounds) {
|
|
LogPrint("privatesend", "vecInputsByRounds: rounds: %d, inputs: %d\n", pair.first, pair.second);
|
|
}
|
|
|
|
int nRounds = vecInputsByRounds.begin()->first;
|
|
if (PrepareDenominate(nRounds, nRounds, strError, vecPSInOutPairs, vecPSInOutPairsTmp)) {
|
|
LogPrintf("CPrivateSendClientSession::SubmitDenominate -- Running PrivateSend denominate for %d rounds, success\n", nRounds);
|
|
return SendDenominate(vecPSInOutPairsTmp, connman);
|
|
}
|
|
|
|
// We failed? That's strange but let's just make final attempt and try to mix everything
|
|
if (PrepareDenominate(0, privateSendClient.nPrivateSendRounds - 1, strError, vecPSInOutPairs, vecPSInOutPairsTmp)) {
|
|
LogPrintf("CPrivateSendClientSession::SubmitDenominate -- Running PrivateSend denominate for all rounds, success\n");
|
|
return SendDenominate(vecPSInOutPairsTmp, connman);
|
|
}
|
|
|
|
// Should never actually get here but just in case
|
|
LogPrintf("CPrivateSendClientSession::SubmitDenominate -- Running PrivateSend denominate for all rounds, error: %s\n", strError);
|
|
strAutoDenomResult = strError;
|
|
return false;
|
|
}
|
|
|
|
bool CPrivateSendClientSession::SelectDenominate(std::string& strErrorRet, std::vector<std::pair<CTxDSIn, CTxOut> >& vecPSInOutPairsRet)
|
|
{
|
|
if (!pwalletMain) {
|
|
strErrorRet = "Wallet is not initialized";
|
|
return false;
|
|
}
|
|
|
|
if (pwalletMain->IsLocked(true)) {
|
|
strErrorRet = "Wallet locked, unable to create transaction!";
|
|
return false;
|
|
}
|
|
|
|
if (GetEntriesCount() > 0) {
|
|
strErrorRet = "Already have pending entries in the PrivateSend pool";
|
|
return false;
|
|
}
|
|
|
|
vecPSInOutPairsRet.clear();
|
|
|
|
std::vector<int> vecBits;
|
|
if (!CPrivateSend::GetDenominationsBits(nSessionDenom, vecBits)) {
|
|
strErrorRet = "Incorrect session denom";
|
|
return false;
|
|
}
|
|
std::vector<CAmount> vecStandardDenoms = CPrivateSend::GetStandardDenominations();
|
|
|
|
bool fSelected = pwalletMain->SelectPSInOutPairsByDenominations(nSessionDenom, vecStandardDenoms[vecBits.front()], CPrivateSend::GetMaxPoolAmount(), vecPSInOutPairsRet);
|
|
if (!fSelected) {
|
|
strErrorRet = "Can't select current denominated inputs";
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
bool CPrivateSendClientSession::PrepareDenominate(int nMinRounds, int nMaxRounds, std::string& strErrorRet, const std::vector<std::pair<CTxDSIn, CTxOut> >& vecPSInOutPairsIn, std::vector<std::pair<CTxDSIn, CTxOut> >& vecPSInOutPairsRet, bool fDryRun)
|
|
{
|
|
std::vector<int> vecBits;
|
|
if (!CPrivateSend::GetDenominationsBits(nSessionDenom, vecBits)) {
|
|
strErrorRet = "Incorrect session denom";
|
|
return false;
|
|
}
|
|
|
|
for (const auto& pair : vecPSInOutPairsIn) {
|
|
pwalletMain->LockCoin(pair.first.prevout);
|
|
}
|
|
|
|
// NOTE: No need to randomize order of inputs because they were
|
|
// initially shuffled in CWallet::SelectPSInOutPairsByDenominations already.
|
|
int nDenomResult{0};
|
|
|
|
std::vector<CAmount> vecStandardDenoms = CPrivateSend::GetStandardDenominations();
|
|
std::vector<int> vecSteps(vecStandardDenoms.size(), 0);
|
|
vecPSInOutPairsRet.clear();
|
|
|
|
// Try to add up to PRIVATESEND_ENTRY_MAX_SIZE of every needed denomination
|
|
for (const auto& pair : vecPSInOutPairsIn) {
|
|
if (pair.second.nRounds < nMinRounds || pair.second.nRounds > nMaxRounds) {
|
|
// unlock unused coins
|
|
pwalletMain->UnlockCoin(pair.first.prevout);
|
|
continue;
|
|
}
|
|
bool fFound = false;
|
|
for (const auto& nBit : vecBits) {
|
|
if (vecSteps[nBit] >= PRIVATESEND_ENTRY_MAX_SIZE) break;
|
|
CAmount nValueDenom = vecStandardDenoms[nBit];
|
|
if (pair.second.nValue == nValueDenom) {
|
|
CScript scriptDenom;
|
|
if (fDryRun) {
|
|
scriptDenom = CScript();
|
|
} else {
|
|
// randomly skip some inputs when we have at least one of the same denom already
|
|
// TODO: make it adjustable via options/cmd-line params
|
|
if (vecSteps[nBit] >= 1 && GetRandInt(5) == 0) {
|
|
// still count it as a step to randomize number of inputs
|
|
// if we have more than (or exactly) PRIVATESEND_ENTRY_MAX_SIZE of them
|
|
++vecSteps[nBit];
|
|
break;
|
|
}
|
|
scriptDenom = keyHolderStorage.AddKey(pwalletMain);
|
|
}
|
|
vecPSInOutPairsRet.emplace_back(pair.first, CTxOut(nValueDenom, scriptDenom));
|
|
fFound = true;
|
|
nDenomResult |= 1 << nBit;
|
|
// step is complete
|
|
++vecSteps[nBit];
|
|
break;
|
|
}
|
|
}
|
|
if (!fFound || fDryRun) {
|
|
// unlock unused coins and if we are not going to mix right away
|
|
pwalletMain->UnlockCoin(pair.first.prevout);
|
|
}
|
|
}
|
|
|
|
if (nDenomResult != nSessionDenom) {
|
|
// unlock used coins on failure
|
|
for (const auto& pair : vecPSInOutPairsRet) {
|
|
pwalletMain->UnlockCoin(pair.first.prevout);
|
|
}
|
|
keyHolderStorage.ReturnAll();
|
|
strErrorRet = "Can't prepare current denominated outputs";
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
// Create collaterals by looping through inputs grouped by addresses
|
|
bool CPrivateSendClientSession::MakeCollateralAmounts(CConnman& connman)
|
|
{
|
|
if (!pwalletMain) return false;
|
|
|
|
std::vector<CompactTallyItem> vecTally;
|
|
if (!pwalletMain->SelectCoinsGroupedByAddresses(vecTally, false, false)) {
|
|
LogPrint("privatesend", "CPrivateSendClientSession::MakeCollateralAmounts -- SelectCoinsGroupedByAddresses can't find any inputs!\n");
|
|
return false;
|
|
}
|
|
|
|
// Start from smallest balances first to consume tiny amounts and cleanup UTXO a bit
|
|
std::sort(vecTally.begin(), vecTally.end(), [](const CompactTallyItem& a, const CompactTallyItem& b) {
|
|
return a.nAmount < b.nAmount;
|
|
});
|
|
|
|
// First try to use only non-denominated funds
|
|
for (const auto& item : vecTally) {
|
|
if (!MakeCollateralAmounts(item, false, connman)) continue;
|
|
return true;
|
|
}
|
|
|
|
// There should be at least some denominated funds we should be able to break in pieces to continue mixing
|
|
for (const auto& item : vecTally) {
|
|
if (!MakeCollateralAmounts(item, true, connman)) continue;
|
|
return true;
|
|
}
|
|
|
|
// If we got here then smth is terribly broken actually
|
|
LogPrintf("CPrivateSendClientSession::MakeCollateralAmounts -- ERROR: Can't make collaterals!\n");
|
|
return false;
|
|
}
|
|
|
|
// Split up large inputs or create fee sized inputs
|
|
bool CPrivateSendClientSession::MakeCollateralAmounts(const CompactTallyItem& tallyItem, bool fTryDenominated, CConnman& connman)
|
|
{
|
|
if (!pwalletMain) return false;
|
|
|
|
LOCK2(cs_main, pwalletMain->cs_wallet);
|
|
|
|
// denominated input is always a single one, so we can check its amount directly and return early
|
|
if (!fTryDenominated && tallyItem.vecOutPoints.size() == 1 && CPrivateSend::IsDenominatedAmount(tallyItem.nAmount))
|
|
return false;
|
|
|
|
CWalletTx wtx;
|
|
CAmount nFeeRet = 0;
|
|
int nChangePosRet = -1;
|
|
std::string strFail = "";
|
|
std::vector<CRecipient> vecSend;
|
|
|
|
// make our collateral address
|
|
CReserveKey reservekeyCollateral(pwalletMain);
|
|
// make our change address
|
|
CReserveKey reservekeyChange(pwalletMain);
|
|
|
|
CScript scriptCollateral;
|
|
CPubKey vchPubKey;
|
|
assert(reservekeyCollateral.GetReservedKey(vchPubKey, false)); // should never fail, as we just unlocked
|
|
scriptCollateral = GetScriptForDestination(vchPubKey.GetID());
|
|
|
|
vecSend.push_back((CRecipient){scriptCollateral, CPrivateSend::GetMaxCollateralAmount(), false});
|
|
|
|
// try to use non-denominated and not mn-like funds first, select them explicitly
|
|
CCoinControl coinControl;
|
|
coinControl.fAllowOtherInputs = false;
|
|
coinControl.fAllowWatchOnly = false;
|
|
// send change to the same address so that we were able create more denoms out of it later
|
|
coinControl.destChange = tallyItem.txdest;
|
|
for (const auto& outpoint : tallyItem.vecOutPoints)
|
|
coinControl.Select(outpoint);
|
|
|
|
bool fSuccess = pwalletMain->CreateTransaction(vecSend, wtx, reservekeyChange,
|
|
nFeeRet, nChangePosRet, strFail, &coinControl, true, ONLY_NONDENOMINATED);
|
|
if (!fSuccess) {
|
|
LogPrintf("CPrivateSendClientSession::MakeCollateralAmounts -- ONLY_NONDENOMINATED: %s\n", strFail);
|
|
// If we failed then most likely there are not enough funds on this address.
|
|
if (fTryDenominated) {
|
|
// Try to also use denominated coins (we can't mix denominated without collaterals anyway).
|
|
if (!pwalletMain->CreateTransaction(vecSend, wtx, reservekeyChange,
|
|
nFeeRet, nChangePosRet, strFail, &coinControl, true, ALL_COINS)) {
|
|
LogPrintf("CPrivateSendClientSession::MakeCollateralAmounts -- ALL_COINS Error: %s\n", strFail);
|
|
reservekeyCollateral.ReturnKey();
|
|
return false;
|
|
}
|
|
} else {
|
|
// Nothing else we can do.
|
|
reservekeyCollateral.ReturnKey();
|
|
return false;
|
|
}
|
|
}
|
|
|
|
reservekeyCollateral.KeepKey();
|
|
|
|
LogPrintf("CPrivateSendClientSession::MakeCollateralAmounts -- txid=%s\n", wtx.GetHash().GetHex());
|
|
|
|
// use the same nCachedLastSuccessBlock as for DS mixing to prevent race
|
|
CValidationState state;
|
|
if (!pwalletMain->CommitTransaction(wtx, reservekeyChange, &connman, state)) {
|
|
LogPrintf("CPrivateSendClientSession::MakeCollateralAmounts -- CommitTransaction failed! Reason given: %s\n", state.GetRejectReason());
|
|
return false;
|
|
}
|
|
|
|
privateSendClient.UpdatedSuccessBlock();
|
|
|
|
return true;
|
|
}
|
|
|
|
// Create denominations by looping through inputs grouped by addresses
|
|
bool CPrivateSendClientSession::CreateDenominated(CConnman& connman)
|
|
{
|
|
if (!pwalletMain) return false;
|
|
|
|
LOCK2(cs_main, pwalletMain->cs_wallet);
|
|
|
|
// NOTE: We do not allow txes larger than 100kB, so we have to limit number of inputs here.
|
|
// We still want to consume a lot of inputs to avoid creating only smaller denoms though.
|
|
// Knowing that each CTxIn is at least 148b big, 400 inputs should take 400 x ~148b = ~60kB.
|
|
// This still leaves more than enough room for another data of typical CreateDenominated tx.
|
|
std::vector<CompactTallyItem> vecTally;
|
|
if (!pwalletMain->SelectCoinsGroupedByAddresses(vecTally, true, true, true, 400)) {
|
|
LogPrint("privatesend", "CPrivateSendClientSession::CreateDenominated -- SelectCoinsGroupedByAddresses can't find any inputs!\n");
|
|
return false;
|
|
}
|
|
|
|
// Start from largest balances first to speed things up by creating txes with larger/largest denoms included
|
|
std::sort(vecTally.begin(), vecTally.end(), [](const CompactTallyItem& a, const CompactTallyItem& b) {
|
|
return a.nAmount > b.nAmount;
|
|
});
|
|
|
|
bool fCreateMixingCollaterals = !pwalletMain->HasCollateralInputs();
|
|
|
|
for (const auto& item : vecTally) {
|
|
if (!CreateDenominated(item, fCreateMixingCollaterals, connman)) continue;
|
|
return true;
|
|
}
|
|
|
|
LogPrintf("CPrivateSendClientSession::CreateDenominated -- failed!\n");
|
|
return false;
|
|
}
|
|
|
|
// Create denominations
|
|
bool CPrivateSendClientSession::CreateDenominated(const CompactTallyItem& tallyItem, bool fCreateMixingCollaterals, CConnman& connman)
|
|
{
|
|
if (!pwalletMain) return false;
|
|
|
|
std::vector<CRecipient> vecSend;
|
|
CKeyHolderStorage keyHolderStorageDenom;
|
|
|
|
CAmount nValueLeft = tallyItem.nAmount;
|
|
nValueLeft -= CPrivateSend::GetCollateralAmount(); // leave some room for fees
|
|
|
|
LogPrintf("CPrivateSendClientSession::CreateDenominated -- 0 - %s nValueLeft: %f\n", CBitcoinAddress(tallyItem.txdest).ToString(), (float)nValueLeft / COIN);
|
|
|
|
// ****** Add an output for mixing collaterals ************ /
|
|
|
|
if (fCreateMixingCollaterals) {
|
|
CScript scriptCollateral = keyHolderStorageDenom.AddKey(pwalletMain);
|
|
vecSend.push_back((CRecipient){scriptCollateral, CPrivateSend::GetMaxCollateralAmount(), false});
|
|
nValueLeft -= CPrivateSend::GetMaxCollateralAmount();
|
|
}
|
|
|
|
// ****** Add outputs for denoms ************ /
|
|
|
|
// try few times - skipping smallest denoms first if there are too many of them already, if failed - use them too
|
|
int nOutputsTotal = 0;
|
|
bool fSkip = true;
|
|
do {
|
|
std::vector<CAmount> vecStandardDenoms = CPrivateSend::GetStandardDenominations();
|
|
|
|
for (auto it = vecStandardDenoms.rbegin(); it != vecStandardDenoms.rend(); ++it) {
|
|
CAmount nDenomValue = *it;
|
|
|
|
if (fSkip) {
|
|
// Note: denoms are skipped if there are already DENOMS_COUNT_MAX of them
|
|
// and there are still larger denoms which can be used for mixing
|
|
|
|
// check skipped denoms
|
|
if (privateSendClient.IsDenomSkipped(nDenomValue)) {
|
|
strAutoDenomResult = strprintf(_("Too many %f denominations, skipping."), (float)nDenomValue / COIN);
|
|
LogPrintf("CPrivateSendClientSession::CreateDenominated -- %s\n", strAutoDenomResult);
|
|
continue;
|
|
}
|
|
|
|
// find new denoms to skip if any (ignore the largest one)
|
|
if (nDenomValue != vecStandardDenoms.front() && pwalletMain->CountInputsWithAmount(nDenomValue) > DENOMS_COUNT_MAX) {
|
|
strAutoDenomResult = strprintf(_("Too many %f denominations, removing."), (float)nDenomValue / COIN);
|
|
LogPrintf("CPrivateSendClientSession::CreateDenominated -- %s\n", strAutoDenomResult);
|
|
privateSendClient.AddSkippedDenom(nDenomValue);
|
|
continue;
|
|
}
|
|
}
|
|
|
|
int nOutputs = 0;
|
|
|
|
// add each output up to 11 times until it can't be added again
|
|
while (nValueLeft - nDenomValue >= 0 && nOutputs <= 10) {
|
|
CScript scriptDenom = keyHolderStorageDenom.AddKey(pwalletMain);
|
|
|
|
vecSend.push_back((CRecipient){scriptDenom, nDenomValue, false});
|
|
|
|
//increment outputs and subtract denomination amount
|
|
nOutputs++;
|
|
nValueLeft -= nDenomValue;
|
|
LogPrintf("CPrivateSendClientSession::CreateDenominated -- 1 - totalOutputs: %d, nOutputsTotal: %d, nOutputs: %d, nValueLeft: %f\n", nOutputsTotal + nOutputs, nOutputsTotal, nOutputs, (float)nValueLeft / COIN);
|
|
}
|
|
|
|
nOutputsTotal += nOutputs;
|
|
if (nValueLeft == 0) break;
|
|
}
|
|
LogPrintf("CPrivateSendClientSession::CreateDenominated -- 2 - nOutputsTotal: %d, nValueLeft: %f\n", nOutputsTotal, (float)nValueLeft / COIN);
|
|
// if there were no outputs added, start over without skipping
|
|
fSkip = !fSkip;
|
|
} while (nOutputsTotal == 0 && !fSkip);
|
|
LogPrintf("CPrivateSendClientSession::CreateDenominated -- 3 - nOutputsTotal: %d, nValueLeft: %f\n", nOutputsTotal, (float)nValueLeft / COIN);
|
|
|
|
// No reasons to create mixing collaterals if we can't create denoms to mix
|
|
if (nOutputsTotal == 0) return false;
|
|
|
|
// if we have anything left over, it will be automatically send back as change - there is no need to send it manually
|
|
|
|
CCoinControl coinControl;
|
|
coinControl.fAllowOtherInputs = false;
|
|
coinControl.fAllowWatchOnly = false;
|
|
// send change to the same address so that we were able create more denoms out of it later
|
|
coinControl.destChange = tallyItem.txdest;
|
|
for (const auto& outpoint : tallyItem.vecOutPoints)
|
|
coinControl.Select(outpoint);
|
|
|
|
CWalletTx wtx;
|
|
CAmount nFeeRet = 0;
|
|
int nChangePosRet = -1;
|
|
std::string strFail = "";
|
|
// make our change address
|
|
CReserveKey reservekeyChange(pwalletMain);
|
|
|
|
bool fSuccess = pwalletMain->CreateTransaction(vecSend, wtx, reservekeyChange,
|
|
nFeeRet, nChangePosRet, strFail, &coinControl, true, ONLY_NONDENOMINATED);
|
|
if (!fSuccess) {
|
|
LogPrintf("CPrivateSendClientSession::CreateDenominated -- Error: %s\n", strFail);
|
|
keyHolderStorageDenom.ReturnAll();
|
|
return false;
|
|
}
|
|
|
|
keyHolderStorageDenom.KeepAll();
|
|
|
|
CValidationState state;
|
|
if (!pwalletMain->CommitTransaction(wtx, reservekeyChange, &connman, state)) {
|
|
LogPrintf("CPrivateSendClientSession::CreateDenominated -- CommitTransaction failed! Reason given: %s\n", state.GetRejectReason());
|
|
return false;
|
|
}
|
|
|
|
// use the same nCachedLastSuccessBlock as for DS mixing to prevent race
|
|
privateSendClient.UpdatedSuccessBlock();
|
|
LogPrintf("CPrivateSendClientSession::CreateDenominated -- txid=%s\n", wtx.GetHash().GetHex());
|
|
|
|
return true;
|
|
}
|
|
|
|
void CPrivateSendClientSession::RelayIn(const CPrivateSendEntry& entry, CConnman& connman)
|
|
{
|
|
if (!mixingMasternode) return;
|
|
|
|
connman.ForNode(mixingMasternode->pdmnState->addr, [&entry, &connman](CNode* pnode) {
|
|
LogPrintf("CPrivateSendClientSession::RelayIn -- found master, relaying message to %s\n", pnode->addr.ToString());
|
|
CNetMsgMaker msgMaker(pnode->GetSendVersion());
|
|
connman.PushMessage(pnode, msgMaker.Make(NetMsgType::DSVIN, entry));
|
|
return true;
|
|
});
|
|
}
|
|
|
|
void CPrivateSendClientSession::SetState(PoolState nStateNew)
|
|
{
|
|
LogPrintf("CPrivateSendClientSession::SetState -- nState: %d, nStateNew: %d\n", nState, nStateNew);
|
|
nState = nStateNew;
|
|
}
|
|
|
|
void CPrivateSendClientManager::UpdatedBlockTip(const CBlockIndex* pindex)
|
|
{
|
|
nCachedBlockHeight = pindex->nHeight;
|
|
LogPrint("privatesend", "CPrivateSendClientManager::UpdatedBlockTip -- nCachedBlockHeight: %d\n", nCachedBlockHeight);
|
|
}
|
|
|
|
void CPrivateSendClientManager::DoMaintenance(CConnman& connman)
|
|
{
|
|
if (fLiteMode) return; // disable all Dash specific functionality
|
|
if (fMasternodeMode) return; // no client-side mixing on masternodes
|
|
|
|
if (!masternodeSync.IsBlockchainSynced() || ShutdownRequested())
|
|
return;
|
|
|
|
static unsigned int nTick = 0;
|
|
static unsigned int nDoAutoNextRun = nTick + PRIVATESEND_AUTO_TIMEOUT_MIN;
|
|
|
|
nTick++;
|
|
CheckTimeout();
|
|
ProcessPendingDsaRequest(connman);
|
|
if (nDoAutoNextRun == nTick) {
|
|
DoAutomaticDenominating(connman);
|
|
nDoAutoNextRun = nTick + PRIVATESEND_AUTO_TIMEOUT_MIN + GetRandInt(PRIVATESEND_AUTO_TIMEOUT_MAX - PRIVATESEND_AUTO_TIMEOUT_MIN);
|
|
}
|
|
}
|