123 lines
5.0 KiB
C++
123 lines
5.0 KiB
C++
|
// Copyright (c) 2018 The Dash Core developers
|
||
|
// Distributed under the MIT software license, see the accompanying
|
||
|
// file COPYING or http://www.opensource.org/licenses/mit-license.php.
|
||
|
|
||
|
#include "providertx.h"
|
||
|
#include "specialtx.h"
|
||
|
|
||
|
#include "hash.h"
|
||
|
#include "clientversion.h"
|
||
|
#include "streams.h"
|
||
|
#include "messagesigner.h"
|
||
|
#include "chainparams.h"
|
||
|
#include "validation.h"
|
||
|
#include "univalue.h"
|
||
|
#include "core_io.h"
|
||
|
#include "script/standard.h"
|
||
|
#include "base58.h"
|
||
|
|
||
|
template <typename ProTx>
|
||
|
static bool CheckService(const uint256& proTxHash, const ProTx& proTx, const CBlockIndex* pindexPrev, CValidationState& state)
|
||
|
{
|
||
|
if (proTx.nProtocolVersion < MIN_PROTX_PROTO_VERSION || proTx.nProtocolVersion > MAX_PROTX_PROTO_VERSION)
|
||
|
return state.DoS(10, false, REJECT_INVALID, "bad-protx-proto-version");
|
||
|
|
||
|
if (!proTx.addr.IsValid())
|
||
|
return state.DoS(10, false, REJECT_INVALID, "bad-protx-addr");
|
||
|
if (Params().NetworkIDString() != CBaseChainParams::REGTEST && !proTx.addr.IsRoutable())
|
||
|
return state.DoS(10, false, REJECT_INVALID, "bad-protx-addr");
|
||
|
|
||
|
if (!proTx.addr.IsIPv4())
|
||
|
return state.DoS(10, false, REJECT_INVALID, "bad-protx-addr");
|
||
|
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
template <typename ProTx>
|
||
|
static bool CheckInputsHashAndSig(const CTransaction &tx, const ProTx& proTx, const CKeyID &keyID, CValidationState& state)
|
||
|
{
|
||
|
uint256 inputsHash = CalcTxInputsHash(tx);
|
||
|
if (inputsHash != proTx.inputsHash)
|
||
|
return state.DoS(100, false, REJECT_INVALID, "bad-protx-inputs-hash");
|
||
|
|
||
|
std::string strError;
|
||
|
if (!CHashSigner::VerifyHash(::SerializeHash(proTx), keyID, proTx.vchSig, strError))
|
||
|
return state.DoS(100, false, REJECT_INVALID, "bad-protx-sig", false, strError);
|
||
|
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
bool CheckProRegTx(const CTransaction& tx, const CBlockIndex* pindexPrev, CValidationState& state)
|
||
|
{
|
||
|
AssertLockHeld(cs_main);
|
||
|
|
||
|
CProRegTx ptx;
|
||
|
if (!GetTxPayload(tx, ptx))
|
||
|
return state.DoS(100, false, REJECT_INVALID, "bad-tx-payload");
|
||
|
|
||
|
if (ptx.nVersion > CProRegTx::CURRENT_VERSION)
|
||
|
return state.DoS(100, false, REJECT_INVALID, "bad-protx-version");
|
||
|
|
||
|
if (ptx.nCollateralIndex >= tx.vout.size())
|
||
|
return state.DoS(10, false, REJECT_INVALID, "bad-protx-collateral-index");
|
||
|
if (tx.vout[ptx.nCollateralIndex].nValue != 1000 * COIN)
|
||
|
return state.DoS(10, false, REJECT_INVALID, "bad-protx-collateral");
|
||
|
if (ptx.keyIDOwner.IsNull() || ptx.keyIDOperator.IsNull() || ptx.keyIDVoting.IsNull())
|
||
|
return state.DoS(10, false, REJECT_INVALID, "bad-protx-key-null");
|
||
|
// we may support P2SH later, but restrict it for now (while in transitioning phase from old MN list to deterministic list)
|
||
|
if (!ptx.scriptPayout.IsPayToPublicKeyHash())
|
||
|
return state.DoS(10, false, REJECT_INVALID, "bad-protx-payee");
|
||
|
|
||
|
// This is a temporary restriction that will be lifted later
|
||
|
// It is required while we are transitioning from the old MN list to the deterministic list
|
||
|
if (tx.vout[ptx.nCollateralIndex].scriptPubKey != ptx.scriptPayout)
|
||
|
return state.DoS(10, false, REJECT_INVALID, "bad-protx-payee-collateral");
|
||
|
|
||
|
// It's allowed to set addr/protocolVersion to 0, which will put the MN into PoSe-banned state and require a ProUpServTx to be issues later
|
||
|
// If any of both is set, it must be valid however
|
||
|
if ((ptx.addr != CService() || ptx.nProtocolVersion != 0) && !CheckService(tx.GetHash(), ptx, pindexPrev, state))
|
||
|
return false;
|
||
|
|
||
|
if (ptx.nOperatorReward > 10000)
|
||
|
return state.DoS(10, false, REJECT_INVALID, "bad-protx-operator-reward");
|
||
|
|
||
|
if (!CheckInputsHashAndSig(tx, ptx, ptx.keyIDOwner, state))
|
||
|
return false;
|
||
|
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
std::string CProRegTx::ToString() const
|
||
|
{
|
||
|
CTxDestination dest;
|
||
|
std::string payee = "unknown";
|
||
|
if (ExtractDestination(scriptPayout, dest)) {
|
||
|
payee = CBitcoinAddress(dest).ToString();
|
||
|
}
|
||
|
|
||
|
return strprintf("CProRegTx(nVersion=%d, nProtocolVersion=%d, nCollateralIndex=%d, addr=%s, nOperatorReward=%f, keyIDOwner=%s, keyIDOperator=%s, keyIDVoting=%s, scriptPayout=%s)",
|
||
|
nVersion, nProtocolVersion, nCollateralIndex, addr.ToString(), (double)nOperatorReward / 100, keyIDOwner.ToString(), keyIDOperator.ToString(), keyIDVoting.ToString(), payee);
|
||
|
}
|
||
|
|
||
|
void CProRegTx::ToJson(UniValue& obj) const
|
||
|
{
|
||
|
obj.clear();
|
||
|
obj.setObject();
|
||
|
obj.push_back(Pair("version", nVersion));
|
||
|
obj.push_back(Pair("protocolVersion", nProtocolVersion));
|
||
|
obj.push_back(Pair("collateralIndex", (int)nCollateralIndex));
|
||
|
obj.push_back(Pair("service", addr.ToString(false)));
|
||
|
obj.push_back(Pair("keyIDOwner", keyIDOwner.ToString()));
|
||
|
obj.push_back(Pair("keyIDOperator", keyIDOperator.ToString()));
|
||
|
obj.push_back(Pair("keyIDVoting", keyIDVoting.ToString()));
|
||
|
|
||
|
CTxDestination dest;
|
||
|
if (ExtractDestination(scriptPayout, dest)) {
|
||
|
CBitcoinAddress bitcoinAddress(dest);
|
||
|
obj.push_back(Pair("payoutAddress", bitcoinAddress.ToString()));
|
||
|
}
|
||
|
obj.push_back(Pair("operatorReward", (double)nOperatorReward / 100));
|
||
|
|
||
|
obj.push_back(Pair("inputsHash", inputsHash.ToString()));
|
||
|
}
|