2016-12-20 14:26:45 +01:00
|
|
|
// Copyright (c) 2014-2017 The Dash Core developers
|
2016-02-02 16:28:56 +01:00
|
|
|
// Distributed under the MIT software license, see the accompanying
|
|
|
|
// file COPYING or http://www.opensource.org/licenses/mit-license.php.
|
2015-02-09 21:54:51 +01:00
|
|
|
|
2018-03-02 14:15:04 +01:00
|
|
|
#include "spork.h"
|
|
|
|
|
|
|
|
#include "base58.h"
|
2017-04-12 09:04:06 +02:00
|
|
|
#include "chainparams.h"
|
2017-08-09 02:19:06 +02:00
|
|
|
#include "validation.h"
|
2017-04-12 09:04:06 +02:00
|
|
|
#include "messagesigner.h"
|
2017-08-09 02:19:06 +02:00
|
|
|
#include "net_processing.h"
|
2016-11-25 20:01:56 +01:00
|
|
|
#include "netmessagemaker.h"
|
2015-02-09 21:54:51 +01:00
|
|
|
|
2018-07-12 11:02:20 +02:00
|
|
|
#include <string>
|
2015-02-09 21:54:51 +01:00
|
|
|
|
2015-02-26 15:02:39 +01:00
|
|
|
CSporkManager sporkManager;
|
|
|
|
|
2015-02-09 21:54:51 +01:00
|
|
|
std::map<uint256, CSporkMessage> mapSporks;
|
2018-04-18 13:50:26 +02:00
|
|
|
std::map<int, int64_t> mapSporkDefaults = {
|
|
|
|
{SPORK_2_INSTANTSEND_ENABLED, 0}, // ON
|
|
|
|
{SPORK_3_INSTANTSEND_BLOCK_FILTERING, 0}, // ON
|
|
|
|
{SPORK_5_INSTANTSEND_MAX_VALUE, 1000}, // 1000 Dash
|
|
|
|
{SPORK_6_NEW_SIGS, 4070908800ULL}, // OFF
|
|
|
|
{SPORK_8_MASTERNODE_PAYMENT_ENFORCEMENT, 4070908800ULL}, // OFF
|
|
|
|
{SPORK_9_SUPERBLOCKS_ENABLED, 4070908800ULL}, // OFF
|
|
|
|
{SPORK_10_MASTERNODE_PAY_UPDATED_NODES, 4070908800ULL}, // OFF
|
|
|
|
{SPORK_12_RECONSIDER_BLOCKS, 0}, // 0 BLOCKS
|
|
|
|
{SPORK_14_REQUIRE_SENTINEL_FLAG, 4070908800ULL}, // OFF
|
|
|
|
};
|
2015-02-09 21:54:51 +01:00
|
|
|
|
2017-02-06 14:31:37 +01:00
|
|
|
void CSporkManager::ProcessSpork(CNode* pfrom, const std::string& strCommand, CDataStream& vRecv, CConnman& connman)
|
2015-02-09 21:54:51 +01:00
|
|
|
{
|
2016-07-30 13:04:27 +02:00
|
|
|
if(fLiteMode) return; // disable all Dash specific functionality
|
2015-02-09 21:54:51 +01:00
|
|
|
|
2016-08-29 21:16:02 +02:00
|
|
|
if (strCommand == NetMsgType::SPORK) {
|
|
|
|
|
2015-02-09 21:54:51 +01:00
|
|
|
CSporkMessage spork;
|
|
|
|
vRecv >> spork;
|
|
|
|
|
|
|
|
uint256 hash = spork.GetHash();
|
2016-09-15 08:50:16 +02:00
|
|
|
|
|
|
|
std::string strLogMsg;
|
|
|
|
{
|
|
|
|
LOCK(cs_main);
|
2017-01-17 21:02:59 +01:00
|
|
|
pfrom->setAskFor.erase(hash);
|
2016-09-15 08:50:16 +02:00
|
|
|
if(!chainActive.Tip()) return;
|
|
|
|
strLogMsg = strprintf("SPORK -- hash: %s id: %d value: %10d bestHeight: %d peer=%d", hash.ToString(), spork.nSporkID, spork.nValue, chainActive.Height(), pfrom->id);
|
|
|
|
}
|
|
|
|
|
2015-04-07 17:17:50 +02:00
|
|
|
if(mapSporksActive.count(spork.nSporkID)) {
|
2016-08-29 21:16:02 +02:00
|
|
|
if (mapSporksActive[spork.nSporkID].nTimeSigned >= spork.nTimeSigned) {
|
2016-09-15 08:50:16 +02:00
|
|
|
LogPrint("spork", "%s seen\n", strLogMsg);
|
2015-02-09 21:54:51 +01:00
|
|
|
return;
|
|
|
|
} else {
|
2016-09-15 08:50:16 +02:00
|
|
|
LogPrintf("%s updated\n", strLogMsg);
|
2015-02-09 21:54:51 +01:00
|
|
|
}
|
2016-09-15 08:50:16 +02:00
|
|
|
} else {
|
|
|
|
LogPrintf("%s new\n", strLogMsg);
|
2015-02-09 21:54:51 +01:00
|
|
|
}
|
|
|
|
|
2018-03-02 14:15:04 +01:00
|
|
|
if(!spork.CheckSignature(sporkPubKeyID)) {
|
2018-03-19 14:09:29 +01:00
|
|
|
LOCK(cs_main);
|
2018-02-26 12:10:20 +01:00
|
|
|
LogPrintf("CSporkManager::ProcessSpork -- ERROR: invalid signature\n");
|
2015-02-09 21:54:51 +01:00
|
|
|
Misbehaving(pfrom->GetId(), 100);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2015-02-11 21:39:33 +01:00
|
|
|
mapSporks[hash] = spork;
|
2015-02-09 21:54:51 +01:00
|
|
|
mapSporksActive[spork.nSporkID] = spork;
|
2017-09-07 17:58:38 +02:00
|
|
|
spork.Relay(connman);
|
2015-02-09 21:54:51 +01:00
|
|
|
|
2015-02-12 05:05:09 +01:00
|
|
|
//does a task if needed
|
|
|
|
ExecuteSpork(spork.nSporkID, spork.nValue);
|
2016-08-29 21:16:02 +02:00
|
|
|
|
|
|
|
} else if (strCommand == NetMsgType::GETSPORKS) {
|
2018-07-12 11:07:51 +02:00
|
|
|
for (const auto& pair : mapSporksActive) {
|
|
|
|
connman.PushMessage(pfrom, CNetMsgMaker(pfrom->GetSendVersion()).Make(NetMsgType::SPORK, pair.second));
|
2015-02-09 21:54:51 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2016-07-30 13:04:27 +02:00
|
|
|
void CSporkManager::ExecuteSpork(int nSporkID, int nValue)
|
|
|
|
{
|
|
|
|
//correct fork via spork technology
|
|
|
|
if(nSporkID == SPORK_12_RECONSIDER_BLOCKS && nValue > 0) {
|
2017-02-04 01:42:04 +01:00
|
|
|
// allow to reprocess 24h of blocks max, which should be enough to resolve any issues
|
|
|
|
int64_t nMaxBlocks = 576;
|
|
|
|
// this potentially can be a heavy operation, so only allow this to be executed once per 10 minutes
|
|
|
|
int64_t nTimeout = 10 * 60;
|
|
|
|
|
|
|
|
static int64_t nTimeExecuted = 0; // i.e. it was never executed before
|
|
|
|
|
|
|
|
if(GetTime() - nTimeExecuted < nTimeout) {
|
|
|
|
LogPrint("spork", "CSporkManager::ExecuteSpork -- ERROR: Trying to reconsider blocks, too soon - %d/%d\n", GetTime() - nTimeExecuted, nTimeout);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(nValue > nMaxBlocks) {
|
|
|
|
LogPrintf("CSporkManager::ExecuteSpork -- ERROR: Trying to reconsider too many blocks %d/%d\n", nValue, nMaxBlocks);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-07-30 13:04:27 +02:00
|
|
|
LogPrintf("CSporkManager::ExecuteSpork -- Reconsider Last %d Blocks\n", nValue);
|
|
|
|
|
|
|
|
ReprocessBlocks(nValue);
|
2017-02-04 01:42:04 +01:00
|
|
|
nTimeExecuted = GetTime();
|
2016-07-30 13:04:27 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-09-07 17:58:38 +02:00
|
|
|
bool CSporkManager::UpdateSpork(int nSporkID, int64_t nValue, CConnman& connman)
|
2016-07-30 13:04:27 +02:00
|
|
|
{
|
|
|
|
|
2017-08-29 01:51:44 +02:00
|
|
|
CSporkMessage spork = CSporkMessage(nSporkID, nValue, GetAdjustedTime());
|
2016-07-30 13:04:27 +02:00
|
|
|
|
2018-03-02 14:15:04 +01:00
|
|
|
if(spork.Sign(sporkPrivKey)) {
|
2017-09-07 17:58:38 +02:00
|
|
|
spork.Relay(connman);
|
2016-07-30 13:04:27 +02:00
|
|
|
mapSporks[spork.GetHash()] = spork;
|
|
|
|
mapSporksActive[nSporkID] = spork;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2015-02-09 21:54:51 +01:00
|
|
|
// grab the spork, otherwise say it's off
|
2016-07-30 13:04:27 +02:00
|
|
|
bool CSporkManager::IsSporkActive(int nSporkID)
|
2015-02-09 21:54:51 +01:00
|
|
|
{
|
2015-06-25 21:59:11 +02:00
|
|
|
int64_t r = -1;
|
2015-02-09 21:54:51 +01:00
|
|
|
|
|
|
|
if(mapSporksActive.count(nSporkID)){
|
2015-02-11 15:47:21 +01:00
|
|
|
r = mapSporksActive[nSporkID].nValue;
|
2018-04-18 13:50:26 +02:00
|
|
|
} else if (mapSporkDefaults.count(nSporkID)) {
|
|
|
|
r = mapSporkDefaults[nSporkID];
|
2015-02-09 21:54:51 +01:00
|
|
|
} else {
|
2018-04-18 13:50:26 +02:00
|
|
|
LogPrint("spork", "CSporkManager::IsSporkActive -- Unknown Spork ID %d\n", nSporkID);
|
|
|
|
r = 4070908800ULL; // 2099-1-1 i.e. off by default
|
2015-02-09 21:54:51 +01:00
|
|
|
}
|
|
|
|
|
2017-08-29 01:51:44 +02:00
|
|
|
return r < GetAdjustedTime();
|
2015-02-09 21:54:51 +01:00
|
|
|
}
|
|
|
|
|
2015-02-11 15:47:21 +01:00
|
|
|
// grab the value of the spork on the network, or the default
|
2016-07-30 13:04:27 +02:00
|
|
|
int64_t CSporkManager::GetSporkValue(int nSporkID)
|
2015-02-11 15:47:21 +01:00
|
|
|
{
|
2016-08-29 21:16:02 +02:00
|
|
|
if (mapSporksActive.count(nSporkID))
|
|
|
|
return mapSporksActive[nSporkID].nValue;
|
|
|
|
|
2018-04-18 13:50:26 +02:00
|
|
|
if (mapSporkDefaults.count(nSporkID)) {
|
|
|
|
return mapSporkDefaults[nSporkID];
|
2015-02-11 15:47:21 +01:00
|
|
|
}
|
|
|
|
|
2018-04-18 13:50:26 +02:00
|
|
|
LogPrint("spork", "CSporkManager::GetSporkValue -- Unknown Spork ID %d\n", nSporkID);
|
|
|
|
return -1;
|
2015-02-11 15:47:21 +01:00
|
|
|
}
|
|
|
|
|
2018-02-12 13:49:00 +01:00
|
|
|
int CSporkManager::GetSporkIDByName(const std::string& strName)
|
2015-02-12 05:05:09 +01:00
|
|
|
{
|
2016-09-01 16:55:25 +02:00
|
|
|
if (strName == "SPORK_2_INSTANTSEND_ENABLED") return SPORK_2_INSTANTSEND_ENABLED;
|
|
|
|
if (strName == "SPORK_3_INSTANTSEND_BLOCK_FILTERING") return SPORK_3_INSTANTSEND_BLOCK_FILTERING;
|
|
|
|
if (strName == "SPORK_5_INSTANTSEND_MAX_VALUE") return SPORK_5_INSTANTSEND_MAX_VALUE;
|
2018-02-15 15:44:22 +01:00
|
|
|
if (strName == "SPORK_6_NEW_SIGS") return SPORK_6_NEW_SIGS;
|
2016-08-29 21:16:02 +02:00
|
|
|
if (strName == "SPORK_8_MASTERNODE_PAYMENT_ENFORCEMENT") return SPORK_8_MASTERNODE_PAYMENT_ENFORCEMENT;
|
|
|
|
if (strName == "SPORK_9_SUPERBLOCKS_ENABLED") return SPORK_9_SUPERBLOCKS_ENABLED;
|
|
|
|
if (strName == "SPORK_10_MASTERNODE_PAY_UPDATED_NODES") return SPORK_10_MASTERNODE_PAY_UPDATED_NODES;
|
|
|
|
if (strName == "SPORK_12_RECONSIDER_BLOCKS") return SPORK_12_RECONSIDER_BLOCKS;
|
2016-11-23 16:30:36 +01:00
|
|
|
if (strName == "SPORK_14_REQUIRE_SENTINEL_FLAG") return SPORK_14_REQUIRE_SENTINEL_FLAG;
|
2016-08-29 21:16:02 +02:00
|
|
|
|
2016-09-15 08:50:16 +02:00
|
|
|
LogPrint("spork", "CSporkManager::GetSporkIDByName -- Unknown Spork name '%s'\n", strName);
|
2016-07-30 13:04:27 +02:00
|
|
|
return -1;
|
2015-08-03 01:08:37 +02:00
|
|
|
}
|
2015-08-02 23:59:28 +02:00
|
|
|
|
2016-07-30 13:04:27 +02:00
|
|
|
std::string CSporkManager::GetSporkNameByID(int nSporkID)
|
|
|
|
{
|
2016-08-29 21:16:02 +02:00
|
|
|
switch (nSporkID) {
|
2016-09-01 16:55:25 +02:00
|
|
|
case SPORK_2_INSTANTSEND_ENABLED: return "SPORK_2_INSTANTSEND_ENABLED";
|
|
|
|
case SPORK_3_INSTANTSEND_BLOCK_FILTERING: return "SPORK_3_INSTANTSEND_BLOCK_FILTERING";
|
|
|
|
case SPORK_5_INSTANTSEND_MAX_VALUE: return "SPORK_5_INSTANTSEND_MAX_VALUE";
|
2018-02-15 15:44:22 +01:00
|
|
|
case SPORK_6_NEW_SIGS: return "SPORK_6_NEW_SIGS";
|
2016-09-01 16:55:25 +02:00
|
|
|
case SPORK_8_MASTERNODE_PAYMENT_ENFORCEMENT: return "SPORK_8_MASTERNODE_PAYMENT_ENFORCEMENT";
|
2016-08-29 21:16:02 +02:00
|
|
|
case SPORK_9_SUPERBLOCKS_ENABLED: return "SPORK_9_SUPERBLOCKS_ENABLED";
|
2016-09-01 16:55:25 +02:00
|
|
|
case SPORK_10_MASTERNODE_PAY_UPDATED_NODES: return "SPORK_10_MASTERNODE_PAY_UPDATED_NODES";
|
|
|
|
case SPORK_12_RECONSIDER_BLOCKS: return "SPORK_12_RECONSIDER_BLOCKS";
|
2016-11-23 16:30:36 +01:00
|
|
|
case SPORK_14_REQUIRE_SENTINEL_FLAG: return "SPORK_14_REQUIRE_SENTINEL_FLAG";
|
2016-08-29 21:16:02 +02:00
|
|
|
default:
|
2016-09-15 08:50:16 +02:00
|
|
|
LogPrint("spork", "CSporkManager::GetSporkNameByID -- Unknown Spork ID %d\n", nSporkID);
|
2016-08-29 21:16:02 +02:00
|
|
|
return "Unknown";
|
|
|
|
}
|
2015-02-12 05:05:09 +01:00
|
|
|
}
|
|
|
|
|
2018-03-02 14:15:04 +01:00
|
|
|
bool CSporkManager::SetSporkAddress(const std::string& strAddress) {
|
|
|
|
CBitcoinAddress address(strAddress);
|
|
|
|
if (!address.IsValid() || !address.GetKeyID(sporkPubKeyID)) {
|
|
|
|
LogPrintf("CSporkManager::SetSporkAddress -- Failed to parse spork address\n");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2018-02-12 13:49:00 +01:00
|
|
|
bool CSporkManager::SetPrivKey(const std::string& strPrivKey)
|
2015-02-09 21:54:51 +01:00
|
|
|
{
|
2018-03-02 14:15:04 +01:00
|
|
|
CKey key;
|
|
|
|
CPubKey pubKey;
|
|
|
|
if(!CMessageSigner::GetKeysFromSecret(strPrivKey, key, pubKey)) {
|
|
|
|
LogPrintf("CSporkManager::SetPrivKey -- Failed to parse private key\n");
|
|
|
|
return false;
|
|
|
|
}
|
2015-02-09 21:54:51 +01:00
|
|
|
|
2018-03-02 14:15:04 +01:00
|
|
|
if (pubKey.GetID() != sporkPubKeyID) {
|
|
|
|
LogPrintf("CSporkManager::SetPrivKey -- New private key does not belong to spork address\n");
|
|
|
|
return false;
|
|
|
|
}
|
2016-07-30 13:04:27 +02:00
|
|
|
|
2018-03-02 14:15:04 +01:00
|
|
|
CSporkMessage spork;
|
|
|
|
if (spork.Sign(key)) {
|
2016-07-30 13:04:27 +02:00
|
|
|
// Test signing successful, proceed
|
|
|
|
LogPrintf("CSporkManager::SetPrivKey -- Successfully initialized as spork signer\n");
|
2018-03-02 14:15:04 +01:00
|
|
|
|
|
|
|
sporkPrivKey = key;
|
2016-07-30 13:04:27 +02:00
|
|
|
return true;
|
|
|
|
} else {
|
2018-03-02 14:15:04 +01:00
|
|
|
LogPrintf("CSporkManager::SetPrivKey -- Test signing failed\n");
|
2015-02-09 21:54:51 +01:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-02-16 15:54:53 +01:00
|
|
|
uint256 CSporkMessage::GetHash() const
|
|
|
|
{
|
|
|
|
return SerializeHash(*this);
|
|
|
|
}
|
|
|
|
|
|
|
|
uint256 CSporkMessage::GetSignatureHash() const
|
|
|
|
{
|
|
|
|
return GetHash();
|
|
|
|
}
|
|
|
|
|
2018-03-02 14:15:04 +01:00
|
|
|
bool CSporkMessage::Sign(const CKey& key)
|
2015-02-09 21:54:51 +01:00
|
|
|
{
|
2018-06-19 01:40:55 +02:00
|
|
|
if (!key.IsValid()) {
|
|
|
|
LogPrintf("CSporkMessage::Sign -- signing key is not valid\n");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2018-03-02 14:15:04 +01:00
|
|
|
CKeyID pubKeyId = key.GetPubKey().GetID();
|
2016-08-19 13:50:04 +02:00
|
|
|
std::string strError = "";
|
2015-02-09 21:54:51 +01:00
|
|
|
|
2018-02-16 15:54:53 +01:00
|
|
|
if (sporkManager.IsSporkActive(SPORK_6_NEW_SIGS)) {
|
|
|
|
uint256 hash = GetSignatureHash();
|
2015-02-09 21:54:51 +01:00
|
|
|
|
2018-02-16 15:54:53 +01:00
|
|
|
if(!CHashSigner::SignHash(hash, key, vchSig)) {
|
|
|
|
LogPrintf("CSporkMessage::Sign -- SignHash() failed\n");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2018-03-02 14:15:04 +01:00
|
|
|
if (!CHashSigner::VerifyHash(hash, pubKeyId, vchSig, strError)) {
|
2018-02-16 15:54:53 +01:00
|
|
|
LogPrintf("CSporkMessage::Sign -- VerifyHash() failed, error: %s\n", strError);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
} else {
|
2018-07-12 11:02:20 +02:00
|
|
|
std::string strMessage = std::to_string(nSporkID) + std::to_string(nValue) + std::to_string(nTimeSigned);
|
2018-02-16 15:54:53 +01:00
|
|
|
|
|
|
|
if(!CMessageSigner::SignMessage(strMessage, vchSig, key)) {
|
|
|
|
LogPrintf("CSporkMessage::Sign -- SignMessage() failed\n");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2018-03-02 14:15:04 +01:00
|
|
|
if(!CMessageSigner::VerifyMessage(pubKeyId, vchSig, strMessage, strError)) {
|
2018-02-16 15:54:53 +01:00
|
|
|
LogPrintf("CSporkMessage::Sign -- VerifyMessage() failed, error: %s\n", strError);
|
|
|
|
return false;
|
|
|
|
}
|
2015-02-09 21:54:51 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2018-03-02 14:15:04 +01:00
|
|
|
bool CSporkMessage::CheckSignature(const CKeyID& pubKeyId) const
|
2015-02-09 21:54:51 +01:00
|
|
|
{
|
2016-08-19 13:50:04 +02:00
|
|
|
std::string strError = "";
|
2015-02-09 21:54:51 +01:00
|
|
|
|
2018-02-16 15:54:53 +01:00
|
|
|
if (sporkManager.IsSporkActive(SPORK_6_NEW_SIGS)) {
|
|
|
|
uint256 hash = GetSignatureHash();
|
|
|
|
|
2018-03-02 14:15:04 +01:00
|
|
|
if (!CHashSigner::VerifyHash(hash, pubKeyId, vchSig, strError)) {
|
2018-02-16 15:54:53 +01:00
|
|
|
// Note: unlike for many other messages when SPORK_6_NEW_SIGS is ON sporks with sigs in old format
|
|
|
|
// and newer timestamps should not be accepted, so if we failed here - that's it
|
|
|
|
LogPrintf("CSporkMessage::CheckSignature -- VerifyHash() failed, error: %s\n", strError);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
} else {
|
2018-07-12 11:02:20 +02:00
|
|
|
std::string strMessage = std::to_string(nSporkID) + std::to_string(nValue) + std::to_string(nTimeSigned);
|
2018-02-16 15:54:53 +01:00
|
|
|
|
2018-03-02 14:15:04 +01:00
|
|
|
if (!CMessageSigner::VerifyMessage(pubKeyId, vchSig, strMessage, strError)){
|
2018-05-13 22:57:32 +02:00
|
|
|
// Note: unlike for other messages we have to check for new format even with SPORK_6_NEW_SIGS
|
|
|
|
// inactive because SPORK_6_NEW_SIGS default is OFF and it is not the first spork to sync
|
|
|
|
// (and even if it would, spork order can't be guaranteed anyway).
|
|
|
|
uint256 hash = GetSignatureHash();
|
|
|
|
if (!CHashSigner::VerifyHash(hash, pubKeyId, vchSig, strError)) {
|
|
|
|
LogPrintf("CSporkMessage::CheckSignature -- VerifyHash() failed, error: %s\n", strError);
|
|
|
|
return false;
|
|
|
|
}
|
2018-02-16 15:54:53 +01:00
|
|
|
}
|
2015-02-09 21:54:51 +01:00
|
|
|
}
|
|
|
|
|
2016-07-30 13:04:27 +02:00
|
|
|
return true;
|
2015-02-09 21:54:51 +01:00
|
|
|
}
|
|
|
|
|
2017-09-07 17:58:38 +02:00
|
|
|
void CSporkMessage::Relay(CConnman& connman)
|
2015-02-09 21:54:51 +01:00
|
|
|
{
|
2016-07-30 13:04:27 +02:00
|
|
|
CInv inv(MSG_SPORK, GetHash());
|
2017-09-07 17:58:38 +02:00
|
|
|
connman.RelayInv(inv);
|
2015-04-03 00:51:08 +02:00
|
|
|
}
|