2016-04-09 21:57:53 +02:00
|
|
|
// Copyright (c) 2014-2016 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.
|
2016-04-09 22:31:01 +02:00
|
|
|
#ifndef GOVERANCE_H
|
|
|
|
#define GOVERANCE_H
|
2016-04-09 21:57:53 +02:00
|
|
|
|
|
|
|
#include "main.h"
|
|
|
|
#include "sync.h"
|
|
|
|
#include "net.h"
|
|
|
|
#include "key.h"
|
|
|
|
#include "util.h"
|
|
|
|
#include "base58.h"
|
|
|
|
#include "masternode.h"
|
2016-04-14 00:41:40 +02:00
|
|
|
#include "governance-vote.h"
|
2016-04-09 21:57:53 +02:00
|
|
|
#include <boost/lexical_cast.hpp>
|
|
|
|
#include "init.h"
|
|
|
|
|
2016-04-16 19:19:17 +02:00
|
|
|
#include <stdio.h>
|
|
|
|
#include <string.h>
|
|
|
|
|
2016-04-09 21:57:53 +02:00
|
|
|
using namespace std;
|
|
|
|
extern CCriticalSection cs_budget;
|
|
|
|
|
2016-04-15 04:54:11 +02:00
|
|
|
// note: is there a reason these are static?
|
|
|
|
// http://stackoverflow.com/questions/3709207/c-semantics-of-static-const-vs-const
|
|
|
|
static const CAmount BUDGET_FEE_TX = (5*COIN);
|
|
|
|
static const int64_t BUDGET_FEE_CONFIRMATIONS = 6;
|
|
|
|
static const int64_t BUDGET_VOTE_UPDATE_MIN = 60*60;
|
|
|
|
static const int64_t CONTRACT_ACTIVATION_TIME = 60*60*24*14;
|
|
|
|
|
|
|
|
|
2016-04-09 22:31:01 +02:00
|
|
|
class CGovernanceManager;
|
2016-04-16 19:19:17 +02:00
|
|
|
class CGovernanceObject;
|
2016-04-09 21:57:53 +02:00
|
|
|
class CBudgetVote;
|
2016-04-14 20:53:46 +02:00
|
|
|
class CNode;
|
2016-04-09 21:57:53 +02:00
|
|
|
|
|
|
|
// todo - 12.1 - change BUDGET_ to GOVERNANCE_ (cherry pick)
|
2016-04-10 02:43:15 +02:00
|
|
|
static const CAmount GOVERNANCE_FEE_TX = (5*COIN);
|
|
|
|
static const int64_t GOVERNANCE_FEE_CONFIRMATIONS = 6;
|
|
|
|
static const int64_t GOVERNANCE_UPDATE_MIN = 60*60;
|
2016-04-09 21:57:53 +02:00
|
|
|
|
2016-04-16 19:19:17 +02:00
|
|
|
extern std::vector<CGovernanceObject> vecImmatureBudgetProposals;
|
2016-04-12 18:00:19 +02:00
|
|
|
extern std::map<uint256, int64_t> askedForSourceProposalOrBudget;
|
2016-04-10 16:46:19 +02:00
|
|
|
extern CGovernanceManager governance;
|
2016-04-09 21:57:53 +02:00
|
|
|
|
|
|
|
//Check the collateral transaction for the budget proposal/finalized budget
|
2016-04-12 18:00:19 +02:00
|
|
|
extern bool IsCollateralValid(uint256 nTxCollateralHash, uint256 nExpectedHash, std::string& strError, int64_t& nTime, int& nConf, CAmount minFee);
|
2016-04-09 21:57:53 +02:00
|
|
|
|
|
|
|
|
|
|
|
//
|
2016-04-09 22:31:01 +02:00
|
|
|
// Governance Manager : Contains all proposals for the budget
|
2016-04-09 21:57:53 +02:00
|
|
|
//
|
2016-04-09 22:31:01 +02:00
|
|
|
class CGovernanceManager
|
2016-04-09 21:57:53 +02:00
|
|
|
{
|
|
|
|
private:
|
|
|
|
|
|
|
|
//hold txes until they mature enough to use
|
|
|
|
map<uint256, CTransaction> mapCollateral;
|
|
|
|
// Keep track of current block index
|
|
|
|
const CBlockIndex *pCurrentBlockIndex;
|
|
|
|
|
2016-04-19 18:51:15 +02:00
|
|
|
int64_t nTimeLastDiff;
|
|
|
|
|
2016-04-09 21:57:53 +02:00
|
|
|
public:
|
|
|
|
// critical section to protect the inner data structures
|
|
|
|
mutable CCriticalSection cs;
|
|
|
|
|
|
|
|
// keep track of the scanning errors I've seen
|
2016-04-16 19:19:17 +02:00
|
|
|
map<uint256, CGovernanceObject> mapProposals;
|
2016-04-09 21:57:53 +02:00
|
|
|
|
2016-04-14 00:41:40 +02:00
|
|
|
// todo - 12.1 - move to private for better encapsulation
|
2016-04-16 19:19:17 +02:00
|
|
|
std::map<uint256, CGovernanceObject> mapSeenMasternodeBudgetProposals;
|
2016-04-09 21:57:53 +02:00
|
|
|
std::map<uint256, CBudgetVote> mapSeenMasternodeBudgetVotes;
|
|
|
|
std::map<uint256, CBudgetVote> mapOrphanMasternodeBudgetVotes;
|
2016-04-14 03:52:26 +02:00
|
|
|
// parent hash vote hash vote
|
2016-04-14 20:07:59 +02:00
|
|
|
std::map<uint256, std::map<uint256, CBudgetVote> > mapVotes;
|
2016-04-09 21:57:53 +02:00
|
|
|
|
2016-04-09 22:31:01 +02:00
|
|
|
CGovernanceManager() {
|
2016-04-09 21:57:53 +02:00
|
|
|
mapProposals.clear();
|
|
|
|
}
|
|
|
|
|
|
|
|
void ClearSeen() {
|
|
|
|
mapSeenMasternodeBudgetProposals.clear();
|
|
|
|
mapSeenMasternodeBudgetVotes.clear();
|
|
|
|
}
|
|
|
|
|
|
|
|
int CountProposalInventoryItems()
|
|
|
|
{
|
|
|
|
return mapSeenMasternodeBudgetProposals.size() + mapSeenMasternodeBudgetVotes.size();
|
|
|
|
}
|
|
|
|
|
|
|
|
int sizeProposals() {return (int)mapProposals.size();}
|
|
|
|
|
2016-04-14 00:41:40 +02:00
|
|
|
// description: incremental sync with our peers
|
|
|
|
// note: incremental syncing seems excessive, well just have clients ask for specific objects and their votes
|
|
|
|
// note: 12.1 - remove
|
|
|
|
//void ResetSync();
|
|
|
|
//void MarkSynced();
|
|
|
|
void Sync(CNode* node, uint256 nProp);
|
2016-04-09 21:57:53 +02:00
|
|
|
|
|
|
|
void ProcessMessage(CNode* pfrom, std::string& strCommand, CDataStream& vRecv);
|
|
|
|
void NewBlock();
|
2016-04-09 22:31:01 +02:00
|
|
|
|
2016-04-16 19:19:17 +02:00
|
|
|
CGovernanceObject *FindProposal(const std::string &strName);
|
|
|
|
CGovernanceObject *FindProposal(uint256 nHash);
|
2016-04-09 21:57:53 +02:00
|
|
|
|
2016-04-26 06:08:36 +02:00
|
|
|
std::vector<CGovernanceObject*> GetAllProposals(int64_t nMoreThanTime);
|
2016-04-09 22:31:01 +02:00
|
|
|
|
2016-04-09 21:57:53 +02:00
|
|
|
bool IsBudgetPaymentBlock(int nBlockHeight);
|
2016-04-16 19:19:17 +02:00
|
|
|
bool AddProposal(CGovernanceObject& budgetProposal);
|
2016-04-09 21:57:53 +02:00
|
|
|
bool UpdateProposal(CBudgetVote& vote, CNode* pfrom, std::string& strError);
|
2016-04-14 03:52:26 +02:00
|
|
|
bool AddOrUpdateVote(CBudgetVote& vote, std::string& strError);
|
2016-04-09 21:57:53 +02:00
|
|
|
bool PropExists(uint256 nHash);
|
|
|
|
std::string GetRequiredPaymentsString(int nBlockHeight);
|
2016-04-14 22:01:15 +02:00
|
|
|
void CleanAndRemove(bool fSignatureCheck);
|
|
|
|
int CountMatchingVotes(int nVoteTypeIn, int nVoteOutcomeIn);
|
2016-04-09 21:57:53 +02:00
|
|
|
|
|
|
|
void CheckOrphanVotes();
|
|
|
|
void Clear(){
|
|
|
|
LOCK(cs);
|
|
|
|
|
|
|
|
LogPrintf("Budget object cleared\n");
|
|
|
|
mapProposals.clear();
|
|
|
|
mapSeenMasternodeBudgetProposals.clear();
|
|
|
|
mapSeenMasternodeBudgetVotes.clear();
|
|
|
|
mapOrphanMasternodeBudgetVotes.clear();
|
|
|
|
}
|
|
|
|
void CheckAndRemove();
|
|
|
|
std::string ToString() const;
|
|
|
|
|
|
|
|
ADD_SERIALIZE_METHODS;
|
|
|
|
|
|
|
|
template <typename Stream, typename Operation>
|
|
|
|
inline void SerializationOp(Stream& s, Operation ser_action, int nType, int nVersion) {
|
|
|
|
READWRITE(mapSeenMasternodeBudgetProposals);
|
|
|
|
READWRITE(mapSeenMasternodeBudgetVotes);
|
|
|
|
READWRITE(mapOrphanMasternodeBudgetVotes);
|
|
|
|
READWRITE(mapProposals);
|
2016-04-14 03:52:26 +02:00
|
|
|
READWRITE(mapVotes);
|
2016-04-09 21:57:53 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void UpdatedBlockTip(const CBlockIndex *pindex);
|
2016-04-19 18:51:15 +02:00
|
|
|
int64_t GetLastDiffTime() {return nTimeLastDiff;}
|
|
|
|
void UpdateLastDiffTime(int64_t nTimeIn) {nTimeLastDiff=nTimeIn;}
|
2016-04-09 21:57:53 +02:00
|
|
|
};
|
|
|
|
|
2016-04-14 03:52:26 +02:00
|
|
|
/**
|
2016-04-16 19:19:17 +02:00
|
|
|
* Governance objects can hold any type of data
|
2016-04-14 03:52:26 +02:00
|
|
|
* --------------------------------------------
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2016-04-16 19:19:17 +02:00
|
|
|
class CGovernanceObjectRegister
|
|
|
|
{
|
|
|
|
private:
|
|
|
|
int nType;
|
|
|
|
std::string strReg;
|
|
|
|
|
|
|
|
public:
|
|
|
|
CGovernanceObjectRegister(int nTypeIn, char* strRegIn)
|
|
|
|
{
|
|
|
|
nType = nTypeIn;
|
|
|
|
strReg = strRegIn;
|
|
|
|
}
|
|
|
|
|
|
|
|
ADD_SERIALIZE_METHODS;
|
|
|
|
|
|
|
|
template <typename Stream, typename Operation>
|
|
|
|
inline void SerializationOp(Stream& s, Operation ser_action, int nType, int nVersion) {
|
|
|
|
//for syncing with other clients
|
|
|
|
READWRITE(nType);
|
2016-04-19 18:51:15 +02:00
|
|
|
READWRITE(LIMITED_STRING(strReg, 255));
|
2016-04-16 19:19:17 +02:00
|
|
|
}
|
|
|
|
};
|
2016-04-14 03:52:26 +02:00
|
|
|
|
|
|
|
/**
|
2016-04-16 19:19:17 +02:00
|
|
|
* Generic Governance Object
|
2016-04-14 03:52:26 +02:00
|
|
|
*
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2016-04-16 19:19:17 +02:00
|
|
|
class CGovernanceObject
|
2016-04-09 21:57:53 +02:00
|
|
|
{
|
|
|
|
private:
|
|
|
|
// critical section to protect the inner data structures
|
|
|
|
mutable CCriticalSection cs;
|
|
|
|
CAmount nAlloted;
|
|
|
|
|
|
|
|
public:
|
2016-04-19 18:51:15 +02:00
|
|
|
|
|
|
|
uint256 nHashParent; //parent object, 0 is root
|
|
|
|
int nRevision; //object revision in the system
|
2016-04-16 19:19:17 +02:00
|
|
|
std::string strName; //org name, username, prop name, etc.
|
2016-04-26 06:08:36 +02:00
|
|
|
int64_t nTime; //time this object was created
|
|
|
|
uint256 nFeeTXHash; //fee-tx
|
2016-04-15 11:10:38 +02:00
|
|
|
|
2016-04-19 18:51:15 +02:00
|
|
|
// caching
|
|
|
|
bool fValid;
|
|
|
|
uint256 nHash;
|
|
|
|
|
2016-04-16 19:19:17 +02:00
|
|
|
// Registers, these can be used for anything
|
|
|
|
// -- check governance wiki for correct usage
|
|
|
|
std::map<int, CGovernanceObjectRegister> mapRegister;
|
|
|
|
|
2016-04-17 02:29:41 +02:00
|
|
|
|
|
|
|
CGovernanceObject();
|
2016-04-26 06:08:36 +02:00
|
|
|
CGovernanceObject(uint256 nHashParentIn, int nRevisionIn, std::string strNameIn, int64_t nTime, uint256 nFeeTXHashIn);
|
|
|
|
|
|
|
|
void swap(CGovernanceObject& first, CGovernanceObject& second) // nothrow
|
|
|
|
{
|
|
|
|
// enable ADL (not necessary in our case, but good practice)
|
|
|
|
using std::swap;
|
|
|
|
|
|
|
|
// by swapping the members of two classes,
|
|
|
|
// the two classes are effectively swapped
|
|
|
|
swap(first.strName, second.strName);
|
|
|
|
swap(first.nHashParent, second.nHashParent);
|
|
|
|
swap(first.nRevision, second.nRevision);
|
|
|
|
swap(first.nTime, second.nTime);
|
|
|
|
swap(first.nFeeTXHash, second.nFeeTXHash);
|
|
|
|
swap(first.fValid, second.fValid);
|
|
|
|
swap(first.nHash, second.nHash);
|
|
|
|
first.mapRegister.swap(second.mapRegister);
|
|
|
|
}
|
2016-04-17 02:29:41 +02:00
|
|
|
|
|
|
|
bool HasMinimumRequiredSupport();
|
|
|
|
bool IsValid(const CBlockIndex* pindex, std::string& strError, bool fCheckCollateral=true);
|
|
|
|
bool IsEstablished();
|
|
|
|
bool NetworkWillPay();
|
|
|
|
|
|
|
|
std::string GetName() {return strName; }
|
2016-04-26 06:08:36 +02:00
|
|
|
|
|
|
|
// get vote counts on each outcome
|
|
|
|
int GetAbsoluteYesCount(int nVoteOutcomeIn);
|
|
|
|
int GetYesCount(int nVoteOutcomeIn);
|
|
|
|
int GetNoCount(int nVoteOutcomeIn);
|
|
|
|
int GetAbstainCount(int nVoteOutcomeIn);
|
2016-04-17 02:29:41 +02:00
|
|
|
|
|
|
|
void CleanAndRemove(bool fSignatureCheck);
|
|
|
|
void Relay();
|
|
|
|
|
|
|
|
uint256 GetHash(){
|
|
|
|
CHashWriter ss(SER_GETHASH, PROTOCOL_VERSION);
|
|
|
|
ss << strName;
|
2016-04-26 06:08:36 +02:00
|
|
|
// ss << mapRegister;
|
2016-04-17 02:29:41 +02:00
|
|
|
uint256 h1 = ss.GetHash();
|
|
|
|
|
|
|
|
return h1;
|
|
|
|
}
|
|
|
|
|
2016-04-15 11:10:38 +02:00
|
|
|
/**
|
2016-04-17 02:29:41 +02:00
|
|
|
* AddRegister - Example usage:
|
2016-04-16 19:19:17 +02:00
|
|
|
* --------------------------------------------------------
|
|
|
|
*
|
|
|
|
* We don't really care what's in these, as long as the masternode network
|
|
|
|
* believes they're accurate. Otherwise the masternodes will vote them down
|
|
|
|
* and we'll delete them from memory (fee-loss attack).
|
2016-04-15 11:10:38 +02:00
|
|
|
*
|
|
|
|
* - This system is designed to allow virtually any usage
|
|
|
|
* - No protocol changes are needed
|
|
|
|
* - Abuse will just be simply automatically deleted
|
|
|
|
* - Masternodes could read this data and do all sorts of things
|
|
|
|
*
|
|
|
|
* Contractor: Mailing address, contact info (5 strings)
|
|
|
|
* Company: BitcoinAddress, UserHash, repeat (automate core team payments?)
|
|
|
|
* Contract: CAmount, nDenomination(int)
|
|
|
|
* Proposal:
|
|
|
|
* MasternodePaymentsBlock: BlockStart, Masternode1, 2, 3...
|
|
|
|
* Arbitration: UserId1, UserId2, TxHash, ContractHash
|
2016-04-16 19:19:17 +02:00
|
|
|
|
2016-04-15 11:10:38 +02:00
|
|
|
*/
|
|
|
|
|
2016-04-16 19:19:17 +02:00
|
|
|
bool AddRegister(std::string& strError, int nTypeIn, std::string strIn)
|
|
|
|
{
|
|
|
|
if(strIn.size() > 64)
|
|
|
|
{
|
|
|
|
strError = "Too big.";
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
char regbuff[64];
|
|
|
|
strncpy(regbuff, strIn.c_str(), sizeof(regbuff));
|
|
|
|
|
|
|
|
CGovernanceObjectRegister newRegister(nTypeIn, regbuff);
|
|
|
|
mapRegister.insert(make_pair(1 , newRegister));
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2016-04-09 21:57:53 +02:00
|
|
|
ADD_SERIALIZE_METHODS;
|
|
|
|
|
|
|
|
template <typename Stream, typename Operation>
|
2016-04-17 02:29:41 +02:00
|
|
|
inline void SerializationOp(Stream& s, Operation ser_action, int nType, int nVersion)
|
|
|
|
{
|
2016-04-26 06:08:36 +02:00
|
|
|
/**
|
|
|
|
* Store all major data items in serialization for other clients
|
|
|
|
* --
|
|
|
|
*
|
|
|
|
*
|
|
|
|
* uint256 nHashParent; //parent object, 0 is root
|
|
|
|
* int nRevision; //object revision in the system
|
|
|
|
* std::string strName; //org name, username, prop name, etc.
|
|
|
|
* int64_t nTime; //time this object was created
|
|
|
|
* uint256 nFeeTXHash; //fee-tx
|
|
|
|
*
|
|
|
|
* // caching
|
|
|
|
* bool fValid;
|
|
|
|
* uint256 nHash;
|
|
|
|
*
|
|
|
|
* // Registers, these can be used for anything
|
|
|
|
* // -- check governance wiki for correct usage
|
|
|
|
* std::map<int, CGovernanceObjectRegister> mapRegister;
|
|
|
|
*
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
READWRITE(nHashParent);
|
|
|
|
READWRITE(nRevision);
|
|
|
|
READWRITE(LIMITED_STRING(strName, 64));
|
2016-04-09 21:57:53 +02:00
|
|
|
READWRITE(nTime);
|
|
|
|
READWRITE(nFeeTXHash);
|
2016-04-26 06:08:36 +02:00
|
|
|
|
|
|
|
// todo - 12.1 - serialize map
|
|
|
|
//READWRITE(mapRegister);
|
2016-04-09 21:57:53 +02:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
#endif
|