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-09-12 09:40:00 +02:00
2016-04-09 22:31:01 +02:00
# ifndef GOVERANCE_H
# define GOVERANCE_H
2016-04-09 21:57:53 +02:00
2016-08-17 09:08:25 +02:00
//#define ENABLE_DASH_DEBUG
# include "util.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-05-25 18:56:58 +02:00
# include "masternodeman.h"
2016-04-09 21:57:53 +02:00
# include <boost/lexical_cast.hpp>
# include "init.h"
2016-08-17 09:08:25 +02:00
# include <univalue.h>
# include "utilstrencodings.h"
2016-04-09 21:57:53 +02:00
2016-04-16 19:19:17 +02:00
# include <stdio.h>
# include <string.h>
2016-09-12 09:40:00 +02:00
class CGovernanceManager ;
class CGovernanceObject ;
class CGovernanceVote ;
2016-08-17 09:08:25 +02:00
2016-09-17 21:37:48 +02:00
static const int MAX_GOVERNANCE_OBJECT_DATA_SIZE = 16 * 1024 ;
2016-09-28 22:03:54 +02:00
static const int MIN_GOVERNANCE_PEER_PROTO_VERSION = 70201 ;
2016-09-17 21:37:48 +02:00
2016-08-17 09:08:25 +02:00
static const int GOVERNANCE_OBJECT_UNKNOWN = 0 ;
static const int GOVERNANCE_OBJECT_PROPOSAL = 1 ;
static const int GOVERNANCE_OBJECT_TRIGGER = 2 ;
2016-09-05 01:44:10 +02:00
static const CAmount GOVERNANCE_PROPOSAL_FEE_TX = ( 0.33 * COIN ) ;
static const int64_t GOVERNANCE_FEE_CONFIRMATIONS = 6 ;
2016-04-10 02:43:15 +02:00
static const int64_t GOVERNANCE_UPDATE_MIN = 60 * 60 ;
2016-04-09 21:57:53 +02:00
2016-05-13 18:03:01 +02:00
// FOR SEEN MAP ARRAYS - GOVERNANCE OBJECTS AND VOTES
2016-08-17 09:08:25 +02:00
static const int SEEN_OBJECT_IS_VALID = 0 ;
static const int SEEN_OBJECT_ERROR_INVALID = 1 ;
static const int SEEN_OBJECT_ERROR_IMMATURE = 2 ;
static const int SEEN_OBJECT_EXECUTED = 3 ; //used for triggers
static const int SEEN_OBJECT_UNKNOWN = 4 ; // the default
2016-05-13 18:03:01 +02:00
2016-09-12 09:40:00 +02:00
extern std : : map < uint256 , int64_t > mapAskedForGovernanceObject ;
extern CGovernanceManager governance ;
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
{
2016-08-17 09:08:25 +02:00
public : // Types
typedef std : : map < uint256 , CGovernanceObject > object_m_t ;
typedef object_m_t : : iterator object_m_it ;
typedef object_m_t : : const_iterator object_m_cit ;
typedef std : : map < uint256 , int > count_m_t ;
typedef count_m_t : : iterator count_m_it ;
typedef count_m_t : : const_iterator count_m_cit ;
typedef std : : map < uint256 , CGovernanceVote > vote_m_t ;
typedef vote_m_t : : iterator vote_m_it ;
typedef vote_m_t : : const_iterator vote_m_cit ;
typedef std : : map < uint256 , CTransaction > transaction_m_t ;
typedef transaction_m_t : : iterator transaction_m_it ;
typedef transaction_m_t : : const_iterator transaction_m_cit ;
typedef object_m_t : : size_type size_type ;
2016-09-15 08:49:24 +02:00
typedef std : : map < COutPoint , int > txout_m_t ;
typedef txout_m_t : : iterator txout_m_it ;
typedef txout_m_t : : const_iterator txout_m_cit ;
2016-04-09 21:57:53 +02:00
private :
//hold txes until they mature enough to use
2016-08-17 09:08:25 +02:00
transaction_m_t mapCollateral ;
2016-04-09 21:57:53 +02:00
// Keep track of current block index
const CBlockIndex * pCurrentBlockIndex ;
2016-04-19 18:51:15 +02:00
int64_t nTimeLastDiff ;
2016-08-17 09:08:25 +02:00
int nCachedBlockHeight ;
2016-04-19 18:51:15 +02:00
2016-08-17 09:08:25 +02:00
// keep track of the scanning errors
object_m_t mapObjects ;
2016-04-09 21:57:53 +02:00
2016-08-17 09:08:25 +02:00
count_m_t mapSeenGovernanceObjects ;
count_m_t mapSeenVotes ;
vote_m_t mapOrphanVotes ;
2016-06-08 08:57:16 +02:00
// todo: one of these should point to the other
// -- must be carefully managed while adding/removing/updating
2016-08-17 09:08:25 +02:00
vote_m_t mapVotesByHash ;
vote_m_t mapVotesByType ;
2016-04-09 21:57:53 +02:00
2016-09-15 08:49:24 +02:00
txout_m_t mapLastMasternodeTrigger ;
2016-09-05 01:44:10 +02:00
2016-08-17 09:08:25 +02:00
public :
// critical section to protect the inner data structures
mutable CCriticalSection cs ;
2016-09-12 09:40:00 +02:00
2016-09-05 01:44:10 +02:00
CGovernanceManager ( ) ;
2016-04-09 21:57:53 +02:00
2016-09-12 09:40:00 +02:00
void ClearSeen ( )
{
2016-08-17 09:08:25 +02:00
LOCK ( cs ) ;
2016-05-23 19:53:05 +02:00
mapSeenGovernanceObjects . clear ( ) ;
mapSeenVotes . clear ( ) ;
2016-04-09 21:57:53 +02:00
}
int CountProposalInventoryItems ( )
{
2016-05-23 19:53:05 +02:00
return mapSeenGovernanceObjects . size ( ) + mapSeenVotes . size ( ) ;
2016-04-09 21:57:53 +02:00
}
2016-04-14 00:41:40 +02:00
void Sync ( CNode * node , uint256 nProp ) ;
2016-06-08 08:57:16 +02:00
void SyncParentObjectByVote ( CNode * pfrom , const CGovernanceVote & vote ) ;
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-06-08 08:57:16 +02:00
CGovernanceObject * FindGovernanceObject ( const uint256 & nHash ) ;
2016-08-17 09:08:25 +02:00
std : : vector < CGovernanceVote * > GetMatchingVotes ( const uint256 & nParentHash ) ;
std : : vector < CGovernanceObject * > GetAllNewerThan ( int64_t nMoreThanTime ) ;
2016-04-09 22:31:01 +02:00
2016-09-08 13:40:19 +02:00
int CountMatchingVotes ( CGovernanceObject & govobj , vote_signal_enum_t nVoteSignalIn , vote_outcome_enum_t nVoteOutcomeIn ) ;
2016-05-23 19:53:05 +02:00
2016-04-09 21:57:53 +02:00
bool IsBudgetPaymentBlock ( int nBlockHeight ) ;
2016-05-28 12:31:44 +02:00
bool AddGovernanceObject ( CGovernanceObject & govobj ) ;
2016-08-17 09:08:25 +02:00
bool AddOrUpdateVote ( const CGovernanceVote & vote , CNode * pfrom , std : : string & strError ) ;
2016-04-09 21:57:53 +02:00
std : : string GetRequiredPaymentsString ( int nBlockHeight ) ;
2016-04-14 22:01:15 +02:00
void CleanAndRemove ( bool fSignatureCheck ) ;
2016-08-17 09:08:25 +02:00
void UpdateCachesAndClean ( ) ;
void CheckAndRemove ( ) { UpdateCachesAndClean ( ) ; }
2016-04-09 21:57:53 +02:00
void CheckOrphanVotes ( ) ;
2016-08-17 09:08:25 +02:00
2016-09-12 09:40:00 +02:00
void Clear ( )
{
2016-04-09 21:57:53 +02:00
LOCK ( cs ) ;
2016-08-17 09:08:25 +02:00
LogPrint ( " gobject " , " Governance object manager was cleared \n " ) ;
2016-05-13 18:03:01 +02:00
mapObjects . clear ( ) ;
2016-05-23 19:53:05 +02:00
mapSeenGovernanceObjects . clear ( ) ;
mapSeenVotes . clear ( ) ;
2016-05-23 19:58:39 +02:00
mapOrphanVotes . clear ( ) ;
2016-06-08 08:57:16 +02:00
mapVotesByType . clear ( ) ;
mapVotesByHash . clear ( ) ;
2016-09-05 01:44:10 +02:00
mapLastMasternodeTrigger . clear ( ) ;
2016-04-09 21:57:53 +02:00
}
2016-09-12 09:40:00 +02:00
2016-04-09 21:57:53 +02:00
std : : string ToString ( ) const ;
ADD_SERIALIZE_METHODS ;
template < typename Stream , typename Operation >
inline void SerializationOp ( Stream & s , Operation ser_action , int nType , int nVersion ) {
2016-08-17 09:08:25 +02:00
LOCK ( cs ) ;
2016-05-23 19:53:05 +02:00
READWRITE ( mapSeenGovernanceObjects ) ;
READWRITE ( mapSeenVotes ) ;
2016-05-23 19:58:39 +02:00
READWRITE ( mapOrphanVotes ) ;
2016-05-13 18:03:01 +02:00
READWRITE ( mapObjects ) ;
2016-06-08 08:57:16 +02:00
READWRITE ( mapVotesByHash ) ;
READWRITE ( mapVotesByType ) ;
2016-09-05 01:44:10 +02:00
READWRITE ( mapLastMasternodeTrigger ) ;
2016-04-09 21:57:53 +02:00
}
void UpdatedBlockTip ( const CBlockIndex * pindex ) ;
2016-09-12 09:40:00 +02:00
int64_t GetLastDiffTime ( ) { return nTimeLastDiff ; }
void UpdateLastDiffTime ( int64_t nTimeIn ) { nTimeLastDiff = nTimeIn ; }
2016-08-17 09:08:25 +02:00
int GetCachedBlockHeight ( ) { return nCachedBlockHeight ; }
// Accessors for thread-safe access to maps
bool HaveObjectForHash ( uint256 nHash ) ;
bool HaveVoteForHash ( uint256 nHash ) ;
bool SerializeObjectForHash ( uint256 nHash , CDataStream & ss ) ;
bool SerializeVoteForHash ( uint256 nHash , CDataStream & ss ) ;
void AddSeenGovernanceObject ( uint256 nHash , int status ) ;
void AddSeenVote ( uint256 nHash , int status ) ;
2016-09-15 08:49:24 +02:00
bool MasternodeRateCheck ( const CTxIn & vin ) ;
2016-09-05 01:44:10 +02:00
2016-04-09 21:57:53 +02:00
} ;
2016-04-14 03:52:26 +02:00
/**
2016-08-17 09:08:25 +02:00
* 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 ;
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-26 06:08:36 +02:00
int64_t nTime ; //time this object was created
2016-08-17 09:08:25 +02:00
uint256 nCollateralHash ; //fee-tx
2016-05-23 19:53:05 +02:00
std : : string strData ; // Data field - can be used for anything
2016-08-17 09:08:25 +02:00
int nObjectType ;
2016-05-23 19:53:05 +02:00
2016-09-05 01:44:10 +02:00
// Masternode info for signed objects
CTxIn vinMasternode ;
std : : vector < unsigned char > vchSig ;
2016-09-15 08:49:24 +02:00
bool fCachedLocalValidity ; // is valid by blockchain
std : : string strLocalValidityError ;
2016-08-17 09:08:25 +02:00
// VARIOUS FLAGS FOR OBJECT / SET VIA MASTERNODE VOTING
2016-06-10 07:16:32 +02:00
bool fCachedFunding ; // true == minimum network support has been reached for this object to be funded (doesn't mean it will for sure though)
bool fCachedValid ; // true == minimum network has been reached flagging this object as a valid and understood goverance object (e.g, the serialized data is correct format, etc)
bool fCachedDelete ; // true == minimum network support has been reached saying this object should be deleted from the system entirely
bool fCachedEndorsed ; // true == minimum network support has been reached flagging this object as endorsed by an elected representative body (e.g. business review board / technecial review board /etc)
2016-08-17 09:08:25 +02:00
bool fDirtyCache ; // object was updated and cached values should be updated soon
bool fUnparsable ; // data field was unparsible, object will be rejected
bool fExpired ; // Object is no longer of interest
2016-04-19 18:51:15 +02:00
2016-04-17 02:29:41 +02:00
CGovernanceObject ( ) ;
2016-09-17 21:37:48 +02:00
CGovernanceObject ( uint256 nHashParentIn , int nRevisionIn , int64_t nTime , uint256 nCollateralHashIn , std : : string strDataIn ) ;
2016-04-26 13:42:34 +02:00
CGovernanceObject ( const CGovernanceObject & other ) ;
2016-08-17 09:08:25 +02:00
void swap ( CGovernanceObject & first , CGovernanceObject & second ) ; // nothrow
2016-04-26 13:42:34 +02:00
2016-09-05 01:44:10 +02:00
// Signature related functions
2016-09-15 08:49:24 +02:00
void SetMasternodeInfo ( const CTxIn & vin ) ;
bool Sign ( CKey & keyMasternode , CPubKey & pubkeyMasternode ) ;
bool CheckSignature ( CPubKey & pubkeyMasternode ) ;
2016-09-05 01:44:10 +02:00
2016-08-17 09:08:25 +02:00
// CORE OBJECT FUNCTIONS
2016-06-08 08:57:16 +02:00
2016-08-17 09:08:25 +02:00
bool IsValidLocally ( const CBlockIndex * pindex , std : : string & strError , bool fCheckCollateral ) ;
2016-09-05 01:44:10 +02:00
/// Check the collateral transaction for the budget proposal/finalized budget
bool IsCollateralValid ( std : : string & strError ) ;
2016-08-17 09:08:25 +02:00
void UpdateLocalValidity ( const CBlockIndex * pCurrentBlockIndex ) ;
void UpdateSentinelVariables ( const CBlockIndex * pCurrentBlockIndex ) ;
int GetObjectType ( ) ;
int GetObjectSubtype ( ) ;
2016-04-26 06:08:36 +02:00
2016-09-05 01:44:10 +02:00
CAmount GetMinCollateralFee ( ) ;
2016-08-17 09:08:25 +02:00
UniValue GetJSONObject ( ) ;
2016-04-17 02:29:41 +02:00
2016-08-17 09:08:25 +02:00
void Relay ( ) ;
uint256 GetHash ( ) ;
2016-04-17 02:29:41 +02:00
2016-08-17 09:08:25 +02:00
// GET VOTE COUNT FOR SIGNAL
2016-04-26 06:08:36 +02:00
2016-09-08 13:40:19 +02:00
int GetAbsoluteYesCount ( vote_signal_enum_t eVoteSignalIn ) ;
int GetAbsoluteNoCount ( vote_signal_enum_t eVoteSignalIn ) ;
int GetYesCount ( vote_signal_enum_t eVoteSignalIn ) ;
int GetNoCount ( vote_signal_enum_t eVoteSignalIn ) ;
int GetAbstainCount ( vote_signal_enum_t eVoteSignalIn ) ;
2016-04-17 02:29:41 +02:00
2016-09-12 09:40:00 +02:00
// FUNCTIONS FOR DEALING WITH DATA STRING
2016-05-13 18:03:01 +02:00
2016-08-17 09:08:25 +02:00
std : : string GetDataAsHex ( ) ;
std : : string GetDataAsString ( ) ;
2016-05-13 18:03:01 +02:00
2016-08-17 09:08:25 +02:00
// SERIALIZER
2016-04-16 19:19:17 +02:00
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-05-27 00:03:37 +02:00
// SERIALIZE DATA FOR SAVING/LOADING OR NETWORK FUNCTIONS
2016-04-26 06:08:36 +02:00
READWRITE ( nHashParent ) ;
READWRITE ( nRevision ) ;
2016-04-09 21:57:53 +02:00
READWRITE ( nTime ) ;
2016-08-17 09:08:25 +02:00
READWRITE ( nCollateralHash ) ;
2016-09-17 21:37:48 +02:00
READWRITE ( LIMITED_STRING ( strData , MAX_GOVERNANCE_OBJECT_DATA_SIZE ) ) ;
2016-08-17 09:08:25 +02:00
READWRITE ( nObjectType ) ;
2016-09-05 01:44:10 +02:00
READWRITE ( vinMasternode ) ;
READWRITE ( vchSig ) ;
2016-06-10 07:16:32 +02:00
// AFTER DESERIALIZATION OCCURS, CACHED VARIABLES MUST BE CALCULATED MANUALLY
2016-04-09 21:57:53 +02:00
}
2016-09-05 01:44:10 +02:00
2016-09-08 13:40:19 +02:00
private :
2016-09-12 09:40:00 +02:00
// FUNCTIONS FOR DEALING WITH DATA STRING
2016-09-08 13:40:19 +02:00
void LoadData ( ) ;
void GetData ( UniValue & objResult ) ;
2016-04-09 21:57:53 +02:00
} ;
# endif