// 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 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 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())); } bool IsProTxCollateral(const CTransaction& tx, uint32_t n) { return GetProTxCollateralIndex(tx) == n; } uint32_t GetProTxCollateralIndex(const CTransaction& tx) { if (tx.nVersion < 3 || tx.nType != TRANSACTION_PROVIDER_REGISTER) return (uint32_t) - 1; CProRegTx proTx; if (!GetTxPayload(tx, proTx)) assert(false); return proTx.nCollateralIndex; }