2015-03-18 00:06:58 +01:00
|
|
|
// Copyright (c) 2014-2015 The Dash developers
|
2015-02-24 15:02:22 +01:00
|
|
|
// Distributed under the MIT/X11 software license, see the accompanying
|
|
|
|
// file COPYING or http://www.opensource.org/licenses/mit-license.php.
|
|
|
|
|
2015-02-23 21:01:21 +01:00
|
|
|
#include "masternodeman.h"
|
2015-03-23 13:59:22 +01:00
|
|
|
#include "masternode.h"
|
2015-02-23 21:01:21 +01:00
|
|
|
#include "activemasternode.h"
|
|
|
|
#include "darksend.h"
|
|
|
|
#include "util.h"
|
|
|
|
#include "addrman.h"
|
2015-06-25 21:59:11 +02:00
|
|
|
#include "spork.h"
|
2015-02-23 21:01:21 +01:00
|
|
|
#include <boost/lexical_cast.hpp>
|
|
|
|
#include <boost/filesystem.hpp>
|
|
|
|
|
|
|
|
/** Masternode manager */
|
|
|
|
CMasternodeMan mnodeman;
|
|
|
|
|
2015-04-22 16:33:44 +02:00
|
|
|
// Keep track of all broadcasts I've seen
|
|
|
|
map<uint256, CMasternodeBroadcast> mapSeenMasternodeBroadcast;
|
|
|
|
|
|
|
|
// Keep track of all pings I've seen
|
|
|
|
map<uint256, CMasternodePing> mapSeenMasternodePing;
|
|
|
|
|
2015-02-23 21:01:21 +01:00
|
|
|
struct CompareValueOnly
|
|
|
|
{
|
|
|
|
bool operator()(const pair<int64_t, CTxIn>& t1,
|
|
|
|
const pair<int64_t, CTxIn>& t2) const
|
|
|
|
{
|
|
|
|
return t1.first < t2.first;
|
|
|
|
}
|
|
|
|
};
|
2015-03-14 19:34:51 +01:00
|
|
|
struct CompareValueOnlyMN
|
|
|
|
{
|
|
|
|
bool operator()(const pair<int64_t, CMasternode>& t1,
|
|
|
|
const pair<int64_t, CMasternode>& t2) const
|
|
|
|
{
|
|
|
|
return t1.first < t2.first;
|
|
|
|
}
|
|
|
|
};
|
2015-02-23 21:01:21 +01:00
|
|
|
|
|
|
|
//
|
|
|
|
// CMasternodeDB
|
|
|
|
//
|
|
|
|
|
|
|
|
CMasternodeDB::CMasternodeDB()
|
|
|
|
{
|
2015-03-05 18:39:47 +01:00
|
|
|
pathMN = GetDataDir() / "mncache.dat";
|
2015-03-24 01:21:07 +01:00
|
|
|
strMagicMessage = "MasternodeCache";
|
2015-02-23 21:01:21 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
bool CMasternodeDB::Write(const CMasternodeMan& mnodemanToSave)
|
|
|
|
{
|
2015-03-05 00:46:50 +01:00
|
|
|
int64_t nStart = GetTimeMillis();
|
|
|
|
|
2015-03-24 01:21:07 +01:00
|
|
|
// serialize, checksum data up to that point, then append checksum
|
2015-02-23 21:01:21 +01:00
|
|
|
CDataStream ssMasternodes(SER_DISK, CLIENT_VERSION);
|
2015-03-24 01:21:07 +01:00
|
|
|
ssMasternodes << strMagicMessage; // masternode cache file specific magic message
|
|
|
|
ssMasternodes << FLATDATA(Params().MessageStart()); // network specific magic number
|
2015-02-23 21:01:21 +01:00
|
|
|
ssMasternodes << mnodemanToSave;
|
|
|
|
uint256 hash = Hash(ssMasternodes.begin(), ssMasternodes.end());
|
|
|
|
ssMasternodes << hash;
|
|
|
|
|
|
|
|
// open output file, and associate with CAutoFile
|
|
|
|
FILE *file = fopen(pathMN.string().c_str(), "wb");
|
2015-04-03 00:51:08 +02:00
|
|
|
CAutoFile fileout(file, SER_DISK, CLIENT_VERSION);
|
|
|
|
if (fileout.IsNull())
|
2015-02-23 21:01:21 +01:00
|
|
|
return error("%s : Failed to open file %s", __func__, pathMN.string());
|
|
|
|
|
|
|
|
// Write and commit header, data
|
|
|
|
try {
|
|
|
|
fileout << ssMasternodes;
|
|
|
|
}
|
|
|
|
catch (std::exception &e) {
|
|
|
|
return error("%s : Serialize or I/O error - %s", __func__, e.what());
|
|
|
|
}
|
2015-04-03 00:51:08 +02:00
|
|
|
// FileCommit(fileout);
|
2015-02-23 21:01:21 +01:00
|
|
|
fileout.fclose();
|
|
|
|
|
2015-03-05 18:39:47 +01:00
|
|
|
LogPrintf("Written info to mncache.dat %dms\n", GetTimeMillis() - nStart);
|
2015-03-05 00:46:50 +01:00
|
|
|
LogPrintf(" %s\n", mnodemanToSave.ToString());
|
|
|
|
|
2015-02-23 21:01:21 +01:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2015-07-14 07:25:07 +02:00
|
|
|
CMasternodeDB::ReadResult CMasternodeDB::Read(CMasternodeMan& mnodemanToLoad, bool fDryRun)
|
2015-02-23 21:01:21 +01:00
|
|
|
{
|
2015-03-05 00:46:50 +01:00
|
|
|
int64_t nStart = GetTimeMillis();
|
2015-02-23 21:01:21 +01:00
|
|
|
// open input file, and associate with CAutoFile
|
|
|
|
FILE *file = fopen(pathMN.string().c_str(), "rb");
|
2015-04-03 00:51:08 +02:00
|
|
|
CAutoFile filein(file, SER_DISK, CLIENT_VERSION);
|
|
|
|
if (filein.IsNull())
|
2015-03-05 00:46:50 +01:00
|
|
|
{
|
|
|
|
error("%s : Failed to open file %s", __func__, pathMN.string());
|
|
|
|
return FileError;
|
|
|
|
}
|
2015-02-23 21:01:21 +01:00
|
|
|
|
|
|
|
// use file size to size memory buffer
|
|
|
|
int fileSize = boost::filesystem::file_size(pathMN);
|
|
|
|
int dataSize = fileSize - sizeof(uint256);
|
|
|
|
// Don't try to resize to a negative number if file is small
|
|
|
|
if (dataSize < 0)
|
|
|
|
dataSize = 0;
|
|
|
|
vector<unsigned char> vchData;
|
|
|
|
vchData.resize(dataSize);
|
|
|
|
uint256 hashIn;
|
|
|
|
|
|
|
|
// read data and checksum from file
|
|
|
|
try {
|
|
|
|
filein.read((char *)&vchData[0], dataSize);
|
|
|
|
filein >> hashIn;
|
|
|
|
}
|
|
|
|
catch (std::exception &e) {
|
2015-03-05 00:46:50 +01:00
|
|
|
error("%s : Deserialize or I/O error - %s", __func__, e.what());
|
|
|
|
return HashReadError;
|
2015-02-23 21:01:21 +01:00
|
|
|
}
|
|
|
|
filein.fclose();
|
|
|
|
|
|
|
|
CDataStream ssMasternodes(vchData, SER_DISK, CLIENT_VERSION);
|
|
|
|
|
|
|
|
// verify stored checksum matches input data
|
|
|
|
uint256 hashTmp = Hash(ssMasternodes.begin(), ssMasternodes.end());
|
|
|
|
if (hashIn != hashTmp)
|
2015-03-05 00:46:50 +01:00
|
|
|
{
|
|
|
|
error("%s : Checksum mismatch, data corrupted", __func__);
|
|
|
|
return IncorrectHash;
|
|
|
|
}
|
2015-02-23 21:01:21 +01:00
|
|
|
|
|
|
|
unsigned char pchMsgTmp[4];
|
2015-03-24 01:21:07 +01:00
|
|
|
std::string strMagicMessageTmp;
|
2015-02-23 21:01:21 +01:00
|
|
|
try {
|
2015-03-24 01:21:07 +01:00
|
|
|
// de-serialize file header (masternode cache file specific magic message) and ..
|
|
|
|
|
|
|
|
ssMasternodes >> strMagicMessageTmp;
|
|
|
|
|
|
|
|
// ... verify the message matches predefined one
|
|
|
|
if (strMagicMessage != strMagicMessageTmp)
|
|
|
|
{
|
|
|
|
error("%s : Invalid masternode cache magic message", __func__);
|
|
|
|
return IncorrectMagicMessage;
|
|
|
|
}
|
|
|
|
|
2015-02-23 21:01:21 +01:00
|
|
|
// de-serialize file header (network specific magic number) and ..
|
|
|
|
ssMasternodes >> FLATDATA(pchMsgTmp);
|
|
|
|
|
|
|
|
// ... verify the network matches ours
|
|
|
|
if (memcmp(pchMsgTmp, Params().MessageStart(), sizeof(pchMsgTmp)))
|
2015-03-05 00:46:50 +01:00
|
|
|
{
|
|
|
|
error("%s : Invalid network magic number", __func__);
|
2015-03-24 01:21:07 +01:00
|
|
|
return IncorrectMagicNumber;
|
2015-03-05 00:46:50 +01:00
|
|
|
}
|
2015-03-24 01:21:07 +01:00
|
|
|
// de-serialize data into CMasternodeMan object
|
2015-02-23 21:01:21 +01:00
|
|
|
ssMasternodes >> mnodemanToLoad;
|
|
|
|
}
|
|
|
|
catch (std::exception &e) {
|
|
|
|
mnodemanToLoad.Clear();
|
2015-03-05 00:46:50 +01:00
|
|
|
error("%s : Deserialize or I/O error - %s", __func__, e.what());
|
|
|
|
return IncorrectFormat;
|
2015-02-23 21:01:21 +01:00
|
|
|
}
|
|
|
|
|
2015-03-05 18:39:47 +01:00
|
|
|
LogPrintf("Loaded info from mncache.dat %dms\n", GetTimeMillis() - nStart);
|
2015-03-05 00:46:50 +01:00
|
|
|
LogPrintf(" %s\n", mnodemanToLoad.ToString());
|
2015-07-14 07:25:07 +02:00
|
|
|
LogPrintf("Masternode manager - cleaning....\n");
|
|
|
|
if(!fDryRun) mnodemanToLoad.CheckAndRemove(true);
|
|
|
|
LogPrintf("Masternode manager - result:\n");
|
|
|
|
LogPrintf(" %s\n", mnodemanToLoad.ToString());
|
2015-03-05 00:46:50 +01:00
|
|
|
|
|
|
|
return Ok;
|
2015-02-23 21:01:21 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void DumpMasternodes()
|
|
|
|
{
|
|
|
|
int64_t nStart = GetTimeMillis();
|
|
|
|
|
|
|
|
CMasternodeDB mndb;
|
2015-03-05 00:46:50 +01:00
|
|
|
CMasternodeMan tempMnodeman;
|
|
|
|
|
2015-03-05 18:39:47 +01:00
|
|
|
LogPrintf("Verifying mncache.dat format...\n");
|
2015-07-14 07:25:07 +02:00
|
|
|
CMasternodeDB::ReadResult readResult = mndb.Read(tempMnodeman, true);
|
2015-03-05 00:46:50 +01:00
|
|
|
// there was an error and it was not an error on file openning => do not proceed
|
|
|
|
if (readResult == CMasternodeDB::FileError)
|
2015-03-05 18:39:47 +01:00
|
|
|
LogPrintf("Missing masternode cache file - mncache.dat, will try to recreate\n");
|
2015-03-05 00:46:50 +01:00
|
|
|
else if (readResult != CMasternodeDB::Ok)
|
|
|
|
{
|
2015-03-24 01:21:07 +01:00
|
|
|
LogPrintf("Error reading mncache.dat: ");
|
|
|
|
if(readResult == CMasternodeDB::IncorrectFormat)
|
|
|
|
LogPrintf("magic is ok but data has invalid format, will try to recreate\n");
|
|
|
|
else
|
|
|
|
{
|
|
|
|
LogPrintf("file format is unknown or invalid, please fix it manually\n");
|
|
|
|
return;
|
|
|
|
}
|
2015-03-05 00:46:50 +01:00
|
|
|
}
|
2015-03-05 18:39:47 +01:00
|
|
|
LogPrintf("Writting info to mncache.dat...\n");
|
2015-02-23 21:01:21 +01:00
|
|
|
mndb.Write(mnodeman);
|
|
|
|
|
2015-03-05 00:46:50 +01:00
|
|
|
LogPrintf("Masternode dump finished %dms\n", GetTimeMillis() - nStart);
|
2015-02-23 21:01:21 +01:00
|
|
|
}
|
|
|
|
|
2015-03-06 18:25:48 +01:00
|
|
|
CMasternodeMan::CMasternodeMan() {
|
|
|
|
nDsqCount = 0;
|
|
|
|
}
|
2015-02-23 21:01:21 +01:00
|
|
|
|
|
|
|
bool CMasternodeMan::Add(CMasternode &mn)
|
|
|
|
{
|
|
|
|
LOCK(cs);
|
|
|
|
|
|
|
|
if (!mn.IsEnabled())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
CMasternode *pmn = Find(mn.vin);
|
2015-02-25 12:54:03 +01:00
|
|
|
if (pmn == NULL)
|
2015-02-23 21:01:21 +01:00
|
|
|
{
|
2015-03-05 09:10:15 +01:00
|
|
|
if(fDebug) LogPrintf("CMasternodeMan: Adding new Masternode %s - %i now\n", mn.addr.ToString().c_str(), size() + 1);
|
2015-02-23 21:01:21 +01:00
|
|
|
vMasternodes.push_back(mn);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
void CMasternodeMan::Check()
|
|
|
|
{
|
|
|
|
LOCK(cs);
|
|
|
|
|
2015-06-24 18:41:03 +02:00
|
|
|
BOOST_FOREACH(CMasternode& mn, vMasternodes) {
|
2015-02-23 21:01:21 +01:00
|
|
|
mn.Check();
|
2015-06-24 18:41:03 +02:00
|
|
|
}
|
2015-02-23 21:01:21 +01:00
|
|
|
}
|
|
|
|
|
2015-07-14 07:25:07 +02:00
|
|
|
void CMasternodeMan::CheckAndRemove(bool forceExpiredRemoval)
|
2015-02-23 21:01:21 +01:00
|
|
|
{
|
|
|
|
LOCK(cs);
|
|
|
|
|
2015-02-24 11:39:29 +01:00
|
|
|
Check();
|
2015-02-23 21:01:21 +01:00
|
|
|
|
2015-07-14 07:25:07 +02:00
|
|
|
//remove inactive and outdated
|
2015-02-24 11:39:29 +01:00
|
|
|
vector<CMasternode>::iterator it = vMasternodes.begin();
|
2015-02-23 21:01:21 +01:00
|
|
|
while(it != vMasternodes.end()){
|
2015-07-14 07:25:07 +02:00
|
|
|
if((*it).activeState == CMasternode::MASTERNODE_REMOVE ||
|
|
|
|
(*it).activeState == CMasternode::MASTERNODE_VIN_SPENT ||
|
|
|
|
(forceExpiredRemoval && (*it).activeState == CMasternode::MASTERNODE_EXPIRED) ||
|
2015-07-20 07:03:36 +02:00
|
|
|
(*it).protocolVersion < masternodePayments.GetMinMasternodePaymentsProto()) {
|
2015-03-05 09:10:15 +01:00
|
|
|
if(fDebug) LogPrintf("CMasternodeMan: Removing inactive Masternode %s - %i now\n", (*it).addr.ToString().c_str(), size() - 1);
|
2015-07-21 01:48:57 +02:00
|
|
|
|
|
|
|
//erase all of the broadcasts we've seen from this vin
|
|
|
|
// -- if we missed a few pings and the node was removed, this will allow is to get it back without them
|
|
|
|
// sending a brand new mnb
|
|
|
|
map<uint256, CMasternodeBroadcast>::iterator it3 = mapSeenMasternodeBroadcast.begin();
|
|
|
|
while(it3 != mapSeenMasternodeBroadcast.end()){
|
|
|
|
if((*it3).second.vin == (*it).vin){
|
|
|
|
mapSeenMasternodeBroadcast.erase(it3++);
|
|
|
|
} else {
|
|
|
|
++it3;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// allow us to ask for this masternode again if we see another ping
|
|
|
|
map<COutPoint, int64_t>::iterator it2 = mWeAskedForMasternodeListEntry.begin();
|
|
|
|
while(it2 != mWeAskedForMasternodeListEntry.end()){
|
|
|
|
if((*it2).first == (*it).vin.prevout){
|
|
|
|
mWeAskedForMasternodeListEntry.erase(it2++);
|
|
|
|
} else {
|
|
|
|
++it2;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-02-27 00:12:43 +01:00
|
|
|
it = vMasternodes.erase(it);
|
2015-02-23 21:01:21 +01:00
|
|
|
} else {
|
|
|
|
++it;
|
|
|
|
}
|
|
|
|
}
|
2015-02-26 00:21:28 +01:00
|
|
|
|
2015-03-05 09:10:15 +01:00
|
|
|
// check who's asked for the Masternode list
|
2015-02-26 00:21:28 +01:00
|
|
|
map<CNetAddr, int64_t>::iterator it1 = mAskedUsForMasternodeList.begin();
|
|
|
|
while(it1 != mAskedUsForMasternodeList.end()){
|
2015-02-26 15:02:39 +01:00
|
|
|
if((*it1).second < GetTime()) {
|
|
|
|
mAskedUsForMasternodeList.erase(it1++);
|
|
|
|
} else {
|
2015-02-27 00:12:43 +01:00
|
|
|
++it1;
|
2015-02-26 15:02:39 +01:00
|
|
|
}
|
2015-02-26 00:21:28 +01:00
|
|
|
}
|
|
|
|
|
2015-03-05 09:10:15 +01:00
|
|
|
// check who we asked for the Masternode list
|
2015-02-26 00:21:28 +01:00
|
|
|
it1 = mWeAskedForMasternodeList.begin();
|
|
|
|
while(it1 != mWeAskedForMasternodeList.end()){
|
2015-02-26 15:02:39 +01:00
|
|
|
if((*it1).second < GetTime()){
|
|
|
|
mWeAskedForMasternodeList.erase(it1++);
|
|
|
|
} else {
|
2015-02-27 00:12:43 +01:00
|
|
|
++it1;
|
2015-02-26 15:02:39 +01:00
|
|
|
}
|
2015-02-26 00:21:28 +01:00
|
|
|
}
|
|
|
|
|
2015-03-05 09:10:15 +01:00
|
|
|
// check which Masternodes we've asked for
|
2015-02-26 00:21:28 +01:00
|
|
|
map<COutPoint, int64_t>::iterator it2 = mWeAskedForMasternodeListEntry.begin();
|
|
|
|
while(it2 != mWeAskedForMasternodeListEntry.end()){
|
2015-02-26 15:02:39 +01:00
|
|
|
if((*it2).second < GetTime()){
|
2015-07-20 21:20:15 +02:00
|
|
|
//erase all of the broadcasts we've seen from this vin
|
|
|
|
// -- if we missed a few pings and the node was removed, this will allow is to get it back without them
|
|
|
|
// sending a brand new mnb
|
|
|
|
map<uint256, CMasternodeBroadcast>::iterator it3 = mapSeenMasternodeBroadcast.begin();
|
|
|
|
while(it3 != mapSeenMasternodeBroadcast.end()){
|
|
|
|
if((*it3).second.vin.prevout == (*it2).first){
|
|
|
|
mapSeenMasternodeBroadcast.erase(it3++);
|
|
|
|
} else {
|
|
|
|
++it3;
|
|
|
|
}
|
|
|
|
}
|
2015-02-26 15:02:39 +01:00
|
|
|
mWeAskedForMasternodeListEntry.erase(it2++);
|
|
|
|
} else {
|
2015-02-27 00:12:43 +01:00
|
|
|
++it2;
|
2015-02-26 15:02:39 +01:00
|
|
|
}
|
2015-02-26 00:21:28 +01:00
|
|
|
}
|
|
|
|
|
2015-02-23 21:01:21 +01:00
|
|
|
}
|
|
|
|
|
2015-03-01 01:04:17 +01:00
|
|
|
void CMasternodeMan::Clear()
|
|
|
|
{
|
|
|
|
LOCK(cs);
|
|
|
|
vMasternodes.clear();
|
|
|
|
mAskedUsForMasternodeList.clear();
|
|
|
|
mWeAskedForMasternodeList.clear();
|
|
|
|
mWeAskedForMasternodeListEntry.clear();
|
2015-03-06 18:25:48 +01:00
|
|
|
nDsqCount = 0;
|
2015-03-01 01:04:17 +01:00
|
|
|
}
|
|
|
|
|
2015-07-19 01:20:48 +02:00
|
|
|
int CMasternodeMan::CountEnabled(int protocolVersion)
|
2015-02-23 21:01:21 +01:00
|
|
|
{
|
|
|
|
int i = 0;
|
2015-07-20 07:03:36 +02:00
|
|
|
protocolVersion = protocolVersion == -1 ? masternodePayments.GetMinMasternodePaymentsProto() : protocolVersion;
|
2015-02-23 21:01:21 +01:00
|
|
|
|
|
|
|
BOOST_FOREACH(CMasternode& mn, vMasternodes) {
|
2015-02-24 11:39:29 +01:00
|
|
|
mn.Check();
|
2015-02-26 00:21:28 +01:00
|
|
|
if(mn.protocolVersion < protocolVersion || !mn.IsEnabled()) continue;
|
|
|
|
i++;
|
2015-02-23 21:01:21 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
return i;
|
|
|
|
}
|
|
|
|
|
2015-02-26 00:21:28 +01:00
|
|
|
void CMasternodeMan::DsegUpdate(CNode* pnode)
|
|
|
|
{
|
2015-02-27 00:12:43 +01:00
|
|
|
LOCK(cs);
|
|
|
|
|
2015-07-02 17:07:30 +02:00
|
|
|
if(!(pnode->addr.IsRFC1918() || pnode->addr.IsLocal())){
|
|
|
|
std::map<CNetAddr, int64_t>::iterator it = mWeAskedForMasternodeList.find(pnode->addr);
|
|
|
|
if (it != mWeAskedForMasternodeList.end())
|
|
|
|
{
|
|
|
|
if (GetTime() < (*it).second) {
|
|
|
|
LogPrintf("dseg - we already asked %s for the list; skipping...\n", pnode->addr.ToString());
|
|
|
|
return;
|
|
|
|
}
|
2015-02-26 00:21:28 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
pnode->PushMessage("dseg", CTxIn());
|
|
|
|
int64_t askAgain = GetTime() + MASTERNODES_DSEG_SECONDS;
|
|
|
|
mWeAskedForMasternodeList[pnode->addr] = askAgain;
|
|
|
|
}
|
|
|
|
|
2015-05-28 19:45:31 +02:00
|
|
|
CMasternode *CMasternodeMan::Find(const CScript &payee)
|
|
|
|
{
|
|
|
|
LOCK(cs);
|
|
|
|
CScript payee2;
|
|
|
|
|
|
|
|
BOOST_FOREACH(CMasternode& mn, vMasternodes)
|
|
|
|
{
|
|
|
|
payee2 = GetScriptForDestination(mn.pubkey.GetID());
|
|
|
|
if(payee2 == payee)
|
|
|
|
return &mn;
|
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2015-02-26 00:21:28 +01:00
|
|
|
CMasternode *CMasternodeMan::Find(const CTxIn &vin)
|
|
|
|
{
|
|
|
|
LOCK(cs);
|
|
|
|
|
|
|
|
BOOST_FOREACH(CMasternode& mn, vMasternodes)
|
|
|
|
{
|
2015-04-09 21:17:32 +02:00
|
|
|
if(mn.vin.prevout == vin.prevout)
|
2015-02-26 00:21:28 +01:00
|
|
|
return &mn;
|
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2015-04-07 21:59:30 +02:00
|
|
|
|
|
|
|
CMasternode *CMasternodeMan::Find(const CPubKey &pubKeyMasternode)
|
|
|
|
{
|
|
|
|
LOCK(cs);
|
|
|
|
|
|
|
|
BOOST_FOREACH(CMasternode& mn, vMasternodes)
|
|
|
|
{
|
|
|
|
if(mn.pubkey2 == pubKeyMasternode)
|
|
|
|
return &mn;
|
|
|
|
}
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2015-06-15 02:05:51 +02:00
|
|
|
CMasternode* CMasternodeMan::GetNextMasternodeInQueueForPayment(int nBlockHeight)
|
2015-02-26 00:21:28 +01:00
|
|
|
{
|
|
|
|
LOCK(cs);
|
|
|
|
|
2015-03-01 16:38:53 +01:00
|
|
|
CMasternode *pOldestMasternode = NULL;
|
|
|
|
|
2015-02-26 00:21:28 +01:00
|
|
|
BOOST_FOREACH(CMasternode &mn, vMasternodes)
|
|
|
|
{
|
|
|
|
mn.Check();
|
|
|
|
if(!mn.IsEnabled()) continue;
|
|
|
|
|
2015-06-25 21:59:11 +02:00
|
|
|
// //check protocol version
|
2015-07-08 03:57:32 +02:00
|
|
|
if(mn.protocolVersion < masternodePayments.GetMinMasternodePaymentsProto()) continue;
|
2015-06-25 21:59:11 +02:00
|
|
|
|
2015-05-27 21:47:01 +02:00
|
|
|
//it's in the list -- so let's skip it
|
2015-06-15 02:05:51 +02:00
|
|
|
if(masternodePayments.IsScheduled(mn, nBlockHeight)) continue;
|
2015-05-27 21:47:01 +02:00
|
|
|
|
2015-05-30 19:27:51 +02:00
|
|
|
//make sure it has as many confirmations as there are masternodes
|
2015-07-20 07:03:36 +02:00
|
|
|
if(mn.GetMasternodeInputAge() < CountEnabled()) continue;
|
2015-05-30 19:27:51 +02:00
|
|
|
|
2015-04-16 16:08:06 +02:00
|
|
|
if(pOldestMasternode == NULL || pOldestMasternode->SecondsSincePayment() < mn.SecondsSincePayment()){
|
2015-03-01 16:38:53 +01:00
|
|
|
pOldestMasternode = &mn;
|
2015-03-16 21:57:07 +01:00
|
|
|
}
|
2015-02-26 00:21:28 +01:00
|
|
|
}
|
|
|
|
|
2015-03-01 16:38:53 +01:00
|
|
|
return pOldestMasternode;
|
2015-02-26 00:21:28 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
CMasternode *CMasternodeMan::FindRandom()
|
|
|
|
{
|
|
|
|
LOCK(cs);
|
|
|
|
|
|
|
|
if(size() == 0) return NULL;
|
|
|
|
|
|
|
|
return &vMasternodes[GetRandInt(vMasternodes.size())];
|
|
|
|
}
|
|
|
|
|
2015-04-22 16:33:44 +02:00
|
|
|
void CMasternodeMan::DecrementVotedTimes()
|
|
|
|
{
|
|
|
|
BOOST_FOREACH(CMasternode& mn, vMasternodes)
|
|
|
|
if(--mn.nVotedTimes < 0) mn.nVotedTimes = 0;
|
|
|
|
}
|
|
|
|
|
2015-02-23 21:01:21 +01:00
|
|
|
CMasternode* CMasternodeMan::GetCurrentMasterNode(int mod, int64_t nBlockHeight, int minProtocol)
|
|
|
|
{
|
|
|
|
unsigned int score = 0;
|
|
|
|
CMasternode* winner = NULL;
|
|
|
|
|
|
|
|
// scan for winner
|
|
|
|
BOOST_FOREACH(CMasternode& mn, vMasternodes) {
|
|
|
|
mn.Check();
|
|
|
|
if(mn.protocolVersion < minProtocol || !mn.IsEnabled()) continue;
|
|
|
|
|
2015-03-05 09:10:15 +01:00
|
|
|
// calculate the score for each Masternode
|
2015-02-23 21:01:21 +01:00
|
|
|
uint256 n = mn.CalculateScore(mod, nBlockHeight);
|
|
|
|
unsigned int n2 = 0;
|
|
|
|
memcpy(&n2, &n, sizeof(n2));
|
|
|
|
|
|
|
|
// determine the winner
|
|
|
|
if(n2 > score){
|
|
|
|
score = n2;
|
|
|
|
winner = &mn;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return winner;
|
|
|
|
}
|
|
|
|
|
2015-03-13 10:28:20 +01:00
|
|
|
int CMasternodeMan::GetMasternodeRank(const CTxIn& vin, int64_t nBlockHeight, int minProtocol, bool fOnlyActive)
|
2015-02-23 21:01:21 +01:00
|
|
|
{
|
|
|
|
std::vector<pair<unsigned int, CTxIn> > vecMasternodeScores;
|
|
|
|
|
2015-03-23 13:59:22 +01:00
|
|
|
//make sure we know about this block
|
|
|
|
uint256 hash = 0;
|
|
|
|
if(!GetBlockHash(hash, nBlockHeight)) return -1;
|
|
|
|
|
2015-02-23 21:01:21 +01:00
|
|
|
// scan for winner
|
|
|
|
BOOST_FOREACH(CMasternode& mn, vMasternodes) {
|
|
|
|
if(mn.protocolVersion < minProtocol) continue;
|
2015-03-24 03:02:22 +01:00
|
|
|
if(fOnlyActive) {
|
|
|
|
mn.Check();
|
|
|
|
if(!mn.IsEnabled()) continue;
|
2015-02-23 21:01:21 +01:00
|
|
|
}
|
|
|
|
uint256 n = mn.CalculateScore(1, nBlockHeight);
|
|
|
|
unsigned int n2 = 0;
|
|
|
|
memcpy(&n2, &n, sizeof(n2));
|
|
|
|
|
|
|
|
vecMasternodeScores.push_back(make_pair(n2, mn.vin));
|
|
|
|
}
|
|
|
|
|
|
|
|
sort(vecMasternodeScores.rbegin(), vecMasternodeScores.rend(), CompareValueOnly());
|
|
|
|
|
2015-03-02 00:09:33 +01:00
|
|
|
int rank = 0;
|
2015-02-23 21:01:21 +01:00
|
|
|
BOOST_FOREACH (PAIRTYPE(unsigned int, CTxIn)& s, vecMasternodeScores){
|
|
|
|
rank++;
|
2015-06-15 02:05:51 +02:00
|
|
|
if(s.second.prevout == vin.prevout) {
|
2015-02-23 21:01:21 +01:00
|
|
|
return rank;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2015-03-14 19:34:51 +01:00
|
|
|
std::vector<pair<int, CMasternode> > CMasternodeMan::GetMasternodeRanks(int64_t nBlockHeight, int minProtocol)
|
|
|
|
{
|
|
|
|
std::vector<pair<unsigned int, CMasternode> > vecMasternodeScores;
|
|
|
|
std::vector<pair<int, CMasternode> > vecMasternodeRanks;
|
|
|
|
|
2015-03-23 13:59:22 +01:00
|
|
|
//make sure we know about this block
|
|
|
|
uint256 hash = 0;
|
|
|
|
if(!GetBlockHash(hash, nBlockHeight)) return vecMasternodeRanks;
|
|
|
|
|
2015-03-14 19:34:51 +01:00
|
|
|
// scan for winner
|
|
|
|
BOOST_FOREACH(CMasternode& mn, vMasternodes) {
|
|
|
|
|
|
|
|
mn.Check();
|
|
|
|
|
|
|
|
if(mn.protocolVersion < minProtocol) continue;
|
|
|
|
if(!mn.IsEnabled()) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint256 n = mn.CalculateScore(1, nBlockHeight);
|
|
|
|
unsigned int n2 = 0;
|
|
|
|
memcpy(&n2, &n, sizeof(n2));
|
|
|
|
|
|
|
|
vecMasternodeScores.push_back(make_pair(n2, mn));
|
|
|
|
}
|
|
|
|
|
|
|
|
sort(vecMasternodeScores.rbegin(), vecMasternodeScores.rend(), CompareValueOnlyMN());
|
|
|
|
|
|
|
|
int rank = 0;
|
|
|
|
BOOST_FOREACH (PAIRTYPE(unsigned int, CMasternode)& s, vecMasternodeScores){
|
|
|
|
rank++;
|
|
|
|
vecMasternodeRanks.push_back(make_pair(rank, s.second));
|
|
|
|
}
|
|
|
|
|
|
|
|
return vecMasternodeRanks;
|
|
|
|
}
|
|
|
|
|
2015-03-13 10:28:20 +01:00
|
|
|
CMasternode* CMasternodeMan::GetMasternodeByRank(int nRank, int64_t nBlockHeight, int minProtocol, bool fOnlyActive)
|
2015-03-02 00:09:33 +01:00
|
|
|
{
|
|
|
|
std::vector<pair<unsigned int, CTxIn> > vecMasternodeScores;
|
|
|
|
|
|
|
|
// scan for winner
|
|
|
|
BOOST_FOREACH(CMasternode& mn, vMasternodes) {
|
|
|
|
|
|
|
|
if(mn.protocolVersion < minProtocol) continue;
|
2015-03-24 03:02:22 +01:00
|
|
|
if(fOnlyActive) {
|
|
|
|
mn.Check();
|
|
|
|
if(!mn.IsEnabled()) continue;
|
2015-03-02 00:09:33 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
uint256 n = mn.CalculateScore(1, nBlockHeight);
|
|
|
|
unsigned int n2 = 0;
|
|
|
|
memcpy(&n2, &n, sizeof(n2));
|
|
|
|
|
|
|
|
vecMasternodeScores.push_back(make_pair(n2, mn.vin));
|
|
|
|
}
|
|
|
|
|
|
|
|
sort(vecMasternodeScores.rbegin(), vecMasternodeScores.rend(), CompareValueOnly());
|
|
|
|
|
|
|
|
int rank = 0;
|
|
|
|
BOOST_FOREACH (PAIRTYPE(unsigned int, CTxIn)& s, vecMasternodeScores){
|
|
|
|
rank++;
|
|
|
|
if(rank == nRank) {
|
|
|
|
return Find(s.second);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
void CMasternodeMan::ProcessMasternodeConnections()
|
|
|
|
{
|
2015-03-24 03:03:34 +01:00
|
|
|
//we don't care about this for regtest
|
2015-04-03 00:51:08 +02:00
|
|
|
if(Params().NetworkID() == CBaseChainParams::REGTEST) return;
|
2015-03-06 23:17:51 +01:00
|
|
|
|
2015-03-02 00:09:33 +01:00
|
|
|
LOCK(cs_vNodes);
|
|
|
|
|
|
|
|
if(!darkSendPool.pSubmittedToMasternode) return;
|
2015-03-22 04:10:34 +01:00
|
|
|
|
2015-03-02 00:09:33 +01:00
|
|
|
BOOST_FOREACH(CNode* pnode, vNodes)
|
|
|
|
{
|
|
|
|
if(darkSendPool.pSubmittedToMasternode->addr == pnode->addr) continue;
|
|
|
|
|
|
|
|
if(pnode->fDarkSendMaster){
|
2015-03-05 09:10:15 +01:00
|
|
|
LogPrintf("Closing Masternode connection %s \n", pnode->addr.ToString().c_str());
|
2015-03-02 00:09:33 +01:00
|
|
|
pnode->CloseSocketDisconnect();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-02-23 21:01:21 +01:00
|
|
|
void CMasternodeMan::ProcessMessage(CNode* pfrom, std::string& strCommand, CDataStream& vRecv)
|
|
|
|
{
|
|
|
|
|
2015-03-05 09:10:15 +01:00
|
|
|
if(fLiteMode) return; //disable all Darksend/Masternode related functionality
|
2015-02-23 21:01:21 +01:00
|
|
|
if(IsInitialBlockDownload()) return;
|
|
|
|
|
2015-03-31 23:21:59 +02:00
|
|
|
LOCK(cs_process_message);
|
2015-02-23 21:01:21 +01:00
|
|
|
|
2015-04-17 17:10:38 +02:00
|
|
|
if (strCommand == "mnb") { //Masternode Broadcast
|
|
|
|
CMasternodeBroadcast mnb;
|
2015-07-14 07:25:07 +02:00
|
|
|
vRecv >> mnb;
|
2015-02-23 21:01:21 +01:00
|
|
|
|
2015-04-22 16:33:44 +02:00
|
|
|
if(mapSeenMasternodeBroadcast.count(mnb.GetHash())) return; //seen
|
|
|
|
mapSeenMasternodeBroadcast[mnb.GetHash()] = mnb;
|
|
|
|
|
2015-04-17 17:10:38 +02:00
|
|
|
int nDoS = 0;
|
2015-07-14 07:25:07 +02:00
|
|
|
if(!mnb.CheckAndUpdate(nDoS)){
|
2015-02-23 21:01:21 +01:00
|
|
|
|
2015-04-17 17:10:38 +02:00
|
|
|
if(nDoS > 0)
|
|
|
|
Misbehaving(pfrom->GetId(), nDoS);
|
2015-02-23 21:01:21 +01:00
|
|
|
|
2015-04-17 17:10:38 +02:00
|
|
|
//failed
|
2015-02-23 21:01:21 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2015-03-05 09:10:15 +01:00
|
|
|
// make sure the vout that was signed is related to the transaction that spawned the Masternode
|
|
|
|
// - this is expensive, so it's only done once per Masternode
|
2015-04-17 17:10:38 +02:00
|
|
|
if(!darkSendSigner.IsVinAssociatedWithPubkey(mnb.vin, mnb.pubkey)) {
|
2015-05-04 17:04:09 +02:00
|
|
|
LogPrintf("mnb - Got mismatched pubkey and vin\n");
|
2015-04-17 17:10:38 +02:00
|
|
|
Misbehaving(pfrom->GetId(), 33);
|
2015-02-23 21:01:21 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// make sure it's still unspent
|
|
|
|
// - this is checked later by .check() in many places and by ThreadCheckDarkSendPool()
|
2015-07-14 07:25:07 +02:00
|
|
|
if(mnb.CheckInputsAndAdd(nDoS)) {
|
2015-02-23 21:01:21 +01:00
|
|
|
// use this as a peer
|
2015-04-17 17:10:38 +02:00
|
|
|
addrman.Add(CAddress(mnb.addr), pfrom->addr, 2*60*60);
|
2015-07-17 11:17:15 +02:00
|
|
|
masternodeSync.AddedMasternodeList();
|
2015-02-23 21:01:21 +01:00
|
|
|
} else {
|
2015-07-14 07:25:07 +02:00
|
|
|
LogPrintf("mnb - Rejected Masternode entry %s\n", mnb.addr.ToString());
|
2015-02-23 21:01:21 +01:00
|
|
|
|
2015-04-17 17:10:38 +02:00
|
|
|
if (nDoS > 0)
|
|
|
|
Misbehaving(pfrom->GetId(), nDoS);
|
2015-02-23 21:01:21 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-04-17 17:10:38 +02:00
|
|
|
else if (strCommand == "mnp") { //Masternode Ping
|
|
|
|
CMasternodePing mnp;
|
|
|
|
vRecv >> mnp;
|
2015-02-23 21:01:21 +01:00
|
|
|
|
2015-07-14 07:25:07 +02:00
|
|
|
if(fDebug) LogPrintf("mnp - Masternode ping, vin: %s\n", mnp.vin.ToString());
|
|
|
|
|
2015-04-22 16:33:44 +02:00
|
|
|
if(mapSeenMasternodePing.count(mnp.GetHash())) return; //seen
|
|
|
|
mapSeenMasternodePing[mnp.GetHash()] = mnp;
|
2015-02-23 21:01:21 +01:00
|
|
|
|
2015-04-17 17:10:38 +02:00
|
|
|
int nDoS = 0;
|
2015-07-14 07:25:07 +02:00
|
|
|
if(mnp.CheckAndUpdate(nDoS)) return;
|
|
|
|
|
|
|
|
if(nDoS > 0) {
|
|
|
|
// if anything significant failed, mark that node and return
|
|
|
|
Misbehaving(pfrom->GetId(), nDoS);
|
2015-02-23 21:01:21 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2015-07-19 18:38:23 +02:00
|
|
|
//search existing Masternode list, if it's known -- don't ask for the mnb
|
|
|
|
CMasternode* pmn = mnodeman.Find(mnp.vin);
|
|
|
|
if(pmn != NULL) return;
|
|
|
|
|
2015-07-14 07:25:07 +02:00
|
|
|
// we wasn't able to accept mnp but nothing significant happened,
|
|
|
|
// we might just have to ask for a masternode entry once
|
2015-04-17 17:10:38 +02:00
|
|
|
std::map<COutPoint, int64_t>::iterator i = mWeAskedForMasternodeListEntry.find(mnp.vin.prevout);
|
2015-02-26 00:21:28 +01:00
|
|
|
if (i != mWeAskedForMasternodeListEntry.end())
|
2015-02-23 21:01:21 +01:00
|
|
|
{
|
|
|
|
int64_t t = (*i).second;
|
|
|
|
if (GetTime() < t) return; // we've asked recently
|
|
|
|
}
|
|
|
|
|
2015-06-23 17:40:08 +02:00
|
|
|
// ask for the mnb info once from the node that sent mnp
|
2015-02-23 21:01:21 +01:00
|
|
|
|
2015-07-14 07:25:07 +02:00
|
|
|
LogPrintf("mnp - Asking source node for missing entry, vin: %s\n", mnp.vin.ToString());
|
2015-04-17 17:10:38 +02:00
|
|
|
pfrom->PushMessage("dseg", mnp.vin);
|
|
|
|
int64_t askAgain = GetTime() + MASTERNODE_MIN_MNP_SECONDS;
|
|
|
|
mWeAskedForMasternodeListEntry[mnp.vin.prevout] = askAgain;
|
2015-02-23 21:01:21 +01:00
|
|
|
|
2015-03-05 09:10:15 +01:00
|
|
|
} else if (strCommand == "dseg") { //Get Masternode list or specific entry
|
2015-02-23 21:01:21 +01:00
|
|
|
|
|
|
|
CTxIn vin;
|
|
|
|
vRecv >> vin;
|
|
|
|
|
|
|
|
if(vin == CTxIn()) { //only should ask for this once
|
|
|
|
//local network
|
2015-07-02 17:07:30 +02:00
|
|
|
bool isLocal = (pfrom->addr.IsRFC1918() || pfrom->addr.IsLocal());
|
|
|
|
|
|
|
|
if(!isLocal && Params().NetworkID() == CBaseChainParams::MAIN) {
|
2015-02-26 00:21:28 +01:00
|
|
|
std::map<CNetAddr, int64_t>::iterator i = mAskedUsForMasternodeList.find(pfrom->addr);
|
2015-04-03 00:51:08 +02:00
|
|
|
if (i != mAskedUsForMasternodeList.end()){
|
2015-02-23 21:01:21 +01:00
|
|
|
int64_t t = (*i).second;
|
|
|
|
if (GetTime() < t) {
|
|
|
|
Misbehaving(pfrom->GetId(), 34);
|
|
|
|
LogPrintf("dseg - peer already asked me for the list\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
2015-02-26 00:21:28 +01:00
|
|
|
int64_t askAgain = GetTime() + MASTERNODES_DSEG_SECONDS;
|
|
|
|
mAskedUsForMasternodeList[pfrom->addr] = askAgain;
|
2015-02-23 21:01:21 +01:00
|
|
|
}
|
|
|
|
} //else, asking for a specific node which is ok
|
|
|
|
|
|
|
|
int i = 0;
|
|
|
|
BOOST_FOREACH(CMasternode& mn, vMasternodes) {
|
|
|
|
if(mn.addr.IsRFC1918()) continue; //local network
|
|
|
|
|
2015-04-03 00:51:08 +02:00
|
|
|
if(mn.IsEnabled()) {
|
2015-03-05 09:10:15 +01:00
|
|
|
if(fDebug) LogPrintf("dseg - Sending Masternode entry - %s \n", mn.addr.ToString().c_str());
|
2015-02-23 21:01:21 +01:00
|
|
|
if(vin == CTxIn()){
|
2015-05-04 17:04:09 +02:00
|
|
|
CDataStream ss(SER_NETWORK, PROTOCOL_VERSION);
|
|
|
|
ss.reserve(1000);
|
|
|
|
ss << CMasternodeBroadcast(mn);
|
|
|
|
pfrom->PushMessage("mnb", ss);
|
2015-02-23 21:01:21 +01:00
|
|
|
} else if (vin == mn.vin) {
|
2015-05-04 17:04:09 +02:00
|
|
|
CDataStream ss(SER_NETWORK, PROTOCOL_VERSION);
|
|
|
|
ss.reserve(1000);
|
|
|
|
ss << CMasternodeBroadcast(mn);
|
|
|
|
pfrom->PushMessage("mnb", ss);
|
|
|
|
|
2015-03-05 09:10:15 +01:00
|
|
|
LogPrintf("dseg - Sent 1 Masternode entries to %s\n", pfrom->addr.ToString().c_str());
|
2015-02-23 21:01:21 +01:00
|
|
|
return;
|
|
|
|
}
|
2015-07-14 07:25:07 +02:00
|
|
|
i++;
|
2015-02-23 21:01:21 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-03-05 09:10:15 +01:00
|
|
|
LogPrintf("dseg - Sent %d Masternode entries to %s\n", i, pfrom->addr.ToString().c_str());
|
2015-02-23 21:01:21 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|
2015-03-01 00:56:52 +01:00
|
|
|
|
2015-04-08 04:07:25 +02:00
|
|
|
void CMasternodeMan::Remove(CTxIn vin)
|
|
|
|
{
|
|
|
|
LOCK(cs);
|
|
|
|
|
|
|
|
vector<CMasternode>::iterator it = vMasternodes.begin();
|
|
|
|
while(it != vMasternodes.end()){
|
|
|
|
if((*it).vin == vin){
|
|
|
|
if(fDebug) LogPrintf("CMasternodeMan: Removing Masternode %s - %i now\n", (*it).addr.ToString().c_str(), size() - 1);
|
|
|
|
vMasternodes.erase(it);
|
|
|
|
break;
|
|
|
|
}
|
2015-05-04 17:04:09 +02:00
|
|
|
++it;
|
2015-04-08 04:07:25 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-03-05 00:46:50 +01:00
|
|
|
std::string CMasternodeMan::ToString() const
|
2015-03-01 00:56:52 +01:00
|
|
|
{
|
|
|
|
std::ostringstream info;
|
|
|
|
|
2015-03-05 09:10:15 +01:00
|
|
|
info << "Masternodes: " << (int)vMasternodes.size() <<
|
|
|
|
", peers who asked us for Masternode list: " << (int)mAskedUsForMasternodeList.size() <<
|
|
|
|
", peers we asked for Masternode list: " << (int)mWeAskedForMasternodeList.size() <<
|
2015-03-06 18:25:48 +01:00
|
|
|
", entries in Masternode list we asked for: " << (int)mWeAskedForMasternodeListEntry.size() <<
|
|
|
|
", nDsqCount: " << (int)nDsqCount;
|
2015-03-01 00:56:52 +01:00
|
|
|
|
|
|
|
return info.str();
|
|
|
|
}
|