2016-12-20 14:26:45 +01:00
// Copyright (c) 2014-2017 The Dash Core developers
2016-11-24 19:12:05 +01:00
// Distributed under the MIT/X11 software license, see the accompanying
// file COPYING or http://www.opensource.org/licenses/mit-license.php.
# ifndef GOVERNANCE_OBJECT_H
# define GOVERNANCE_OBJECT_H
//#define ENABLE_DASH_DEBUG
2016-12-20 14:27:59 +01:00
# include "cachemultimap.h"
2016-11-24 19:12:05 +01:00
# include "governance-exceptions.h"
# include "governance-vote.h"
# include "governance-votedb.h"
2016-12-20 14:27:59 +01:00
# include "key.h"
# include "net.h"
# include "sync.h"
# include "util.h"
2018-02-15 15:43:17 +01:00
# include "utilstrencodings.h"
2016-11-24 19:12:05 +01:00
2016-12-20 14:27:59 +01:00
# include <univalue.h>
2016-11-24 19:12:05 +01:00
class CGovernanceManager ;
class CGovernanceTriggerManager ;
class CGovernanceObject ;
class CGovernanceVote ;
static const int MAX_GOVERNANCE_OBJECT_DATA_SIZE = 16 * 1024 ;
2017-03-14 07:22:00 +01:00
static const int MIN_GOVERNANCE_PEER_PROTO_VERSION = 70206 ;
2017-02-02 09:50:44 +01:00
static const int GOVERNANCE_FILTER_PROTO_VERSION = 70206 ;
static const double GOVERNANCE_FILTER_FP_RATE = 0.001 ;
2016-11-24 19:12:05 +01:00
static const int GOVERNANCE_OBJECT_UNKNOWN = 0 ;
static const int GOVERNANCE_OBJECT_PROPOSAL = 1 ;
static const int GOVERNANCE_OBJECT_TRIGGER = 2 ;
static const int GOVERNANCE_OBJECT_WATCHDOG = 3 ;
2017-02-02 09:51:02 +01:00
static const CAmount GOVERNANCE_PROPOSAL_FEE_TX = ( 5.0 * COIN ) ;
2016-11-24 19:12:05 +01:00
static const int64_t GOVERNANCE_FEE_CONFIRMATIONS = 6 ;
2017-07-05 02:31:50 +02:00
static const int64_t GOVERNANCE_MIN_RELAY_FEE_CONFIRMATIONS = 1 ;
2016-11-24 19:12:05 +01:00
static const int64_t GOVERNANCE_UPDATE_MIN = 60 * 60 ;
static const int64_t GOVERNANCE_DELETION_DELAY = 10 * 60 ;
static const int64_t GOVERNANCE_ORPHAN_EXPIRATION_TIME = 10 * 60 ;
// FOR SEEN MAP ARRAYS - GOVERNANCE OBJECTS AND VOTES
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
typedef std : : pair < CGovernanceVote , int64_t > vote_time_pair_t ;
inline bool operator < ( const vote_time_pair_t & p1 , const vote_time_pair_t & p2 )
{
return ( p1 . first < p2 . first ) ;
}
struct vote_instance_t {
vote_outcome_enum_t eOutcome ;
int64_t nTime ;
2017-02-23 13:29:00 +01:00
int64_t nCreationTime ;
2016-11-24 19:12:05 +01:00
2017-02-23 13:29:00 +01:00
vote_instance_t ( vote_outcome_enum_t eOutcomeIn = VOTE_OUTCOME_NONE , int64_t nTimeIn = 0 , int64_t nCreationTimeIn = 0 )
2016-11-24 19:12:05 +01:00
: eOutcome ( eOutcomeIn ) ,
2017-02-23 13:29:00 +01:00
nTime ( nTimeIn ) ,
nCreationTime ( nCreationTimeIn )
2016-11-24 19:12:05 +01:00
{ }
ADD_SERIALIZE_METHODS ;
template < typename Stream , typename Operation >
2017-09-19 21:36:55 +02:00
inline void SerializationOp ( Stream & s , Operation ser_action )
2016-11-24 19:12:05 +01:00
{
int nOutcome = int ( eOutcome ) ;
READWRITE ( nOutcome ) ;
READWRITE ( nTime ) ;
2017-02-23 13:29:00 +01:00
READWRITE ( nCreationTime ) ;
2016-11-24 19:12:05 +01:00
if ( ser_action . ForRead ( ) ) {
eOutcome = vote_outcome_enum_t ( nOutcome ) ;
}
}
} ;
typedef std : : map < int , vote_instance_t > vote_instance_m_t ;
typedef vote_instance_m_t : : iterator vote_instance_m_it ;
typedef vote_instance_m_t : : const_iterator vote_instance_m_cit ;
struct vote_rec_t {
vote_instance_m_t mapInstances ;
ADD_SERIALIZE_METHODS ;
template < typename Stream , typename Operation >
2017-09-19 21:36:55 +02:00
inline void SerializationOp ( Stream & s , Operation ser_action )
2016-11-24 19:12:05 +01:00
{
READWRITE ( mapInstances ) ;
}
} ;
/**
* Governance Object
*
*/
class CGovernanceObject
{
friend class CGovernanceManager ;
friend class CGovernanceTriggerManager ;
2018-02-12 19:35:10 +01:00
friend class CSuperblock ;
2016-11-24 19:12:05 +01:00
public : // Types
2017-09-11 16:13:48 +02:00
typedef std : : map < COutPoint , vote_rec_t > vote_m_t ;
2016-11-24 19:12:05 +01:00
typedef vote_m_t : : iterator vote_m_it ;
typedef vote_m_t : : const_iterator vote_m_cit ;
2018-02-06 12:08:43 +01:00
typedef CacheMultiMap < COutPoint , vote_time_pair_t > vote_cmm_t ;
2016-11-24 19:12:05 +01:00
private :
/// critical section to protect the inner data structures
mutable CCriticalSection cs ;
/// Object typecode
int nObjectType ;
/// parent object, 0 is root
uint256 nHashParent ;
/// object revision in the system
int nRevision ;
/// time this object was created
int64_t nTime ;
/// time this object was marked for deletion
int64_t nDeletionTime ;
/// fee-tx
uint256 nCollateralHash ;
/// Data field - can be used for anything
2018-02-15 15:43:17 +01:00
std : : vector < unsigned char > vchData ;
2016-11-24 19:12:05 +01:00
/// Masternode info for signed objects
2018-02-15 08:29:44 +01:00
COutPoint masternodeOutpoint ;
2016-11-24 19:12:05 +01:00
std : : vector < unsigned char > vchSig ;
/// is valid by blockchain
bool fCachedLocalValidity ;
std : : string strLocalValidityError ;
// VARIOUS FLAGS FOR OBJECT / SET VIA MASTERNODE VOTING
/// true == minimum network support has been reached for this object to be funded (doesn't mean it will for sure though)
bool fCachedFunding ;
2017-12-05 23:17:45 +01:00
/// true == minimum network has been reached flagging this object as a valid and understood governance object (e.g, the serialized data is correct format, etc)
2016-11-24 19:12:05 +01:00
bool fCachedValid ;
/// true == minimum network support has been reached saying this object should be deleted from the system entirely
bool fCachedDelete ;
/** 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 )
*/
bool fCachedEndorsed ;
/// object was updated and cached values should be updated soon
bool fDirtyCache ;
/// Object is no longer of interest
bool fExpired ;
/// Failed to parse object data
bool fUnparsable ;
vote_m_t mapCurrentMNVotes ;
/// Limited map of votes orphaned by MN
2018-02-06 12:08:43 +01:00
vote_cmm_t cmmapOrphanVotes ;
2016-11-24 19:12:05 +01:00
CGovernanceObjectVoteFile fileVotes ;
public :
CGovernanceObject ( ) ;
2018-02-15 15:43:17 +01:00
CGovernanceObject ( const uint256 & nHashParentIn , int nRevisionIn , int64_t nTime , const uint256 & nCollateralHashIn , const std : : string & strDataHexIn ) ;
2016-11-24 19:12:05 +01:00
CGovernanceObject ( const CGovernanceObject & other ) ;
void swap ( CGovernanceObject & first , CGovernanceObject & second ) ; // nothrow
// Public Getter methods
int64_t GetCreationTime ( ) const {
return nTime ;
}
int64_t GetDeletionTime ( ) const {
return nDeletionTime ;
}
int GetObjectType ( ) const {
return nObjectType ;
}
const uint256 & GetCollateralHash ( ) const {
return nCollateralHash ;
}
2018-02-15 08:29:44 +01:00
const COutPoint & GetMasternodeOutpoint ( ) const {
return masternodeOutpoint ;
2016-11-24 19:12:05 +01:00
}
bool IsSetCachedFunding ( ) const {
return fCachedFunding ;
}
bool IsSetCachedValid ( ) const {
return fCachedValid ;
}
bool IsSetCachedDelete ( ) const {
return fCachedDelete ;
}
bool IsSetCachedEndorsed ( ) const {
return fCachedEndorsed ;
}
bool IsSetDirtyCache ( ) const {
return fDirtyCache ;
}
bool IsSetExpired ( ) const {
return fExpired ;
}
void InvalidateVoteCache ( ) {
fDirtyCache = true ;
}
2018-03-20 12:04:59 +01:00
const CGovernanceObjectVoteFile & GetVoteFile ( ) const {
2016-11-24 19:12:05 +01:00
return fileVotes ;
}
// Signature related functions
2018-02-15 08:29:44 +01:00
void SetMasternodeOutpoint ( const COutPoint & outpoint ) ;
2018-02-12 13:49:00 +01:00
bool Sign ( const CKey & keyMasternode , const CPubKey & pubKeyMasternode ) ;
2018-03-20 12:04:59 +01:00
bool CheckSignature ( const CPubKey & pubKeyMasternode ) const ;
2016-11-24 19:12:05 +01:00
2016-12-20 04:25:20 +01:00
std : : string GetSignatureMessage ( ) const ;
2018-02-16 15:54:53 +01:00
uint256 GetSignatureHash ( ) const ;
2016-12-20 04:25:20 +01:00
2016-11-24 19:12:05 +01:00
// CORE OBJECT FUNCTIONS
2018-03-20 12:04:59 +01:00
bool IsValidLocally ( std : : string & strError , bool fCheckCollateral ) const ;
2016-11-24 19:12:05 +01:00
2018-03-20 12:04:59 +01:00
bool IsValidLocally ( std : : string & strError , bool & fMissingMasternode , bool & fMissingConfirmations , bool fCheckCollateral ) const ;
2016-11-24 19:12:05 +01:00
/// Check the collateral transaction for the budget proposal/finalized budget
2018-03-20 12:04:59 +01:00
bool IsCollateralValid ( std : : string & strError , bool & fMissingConfirmations ) const ;
2016-11-24 19:12:05 +01:00
2017-01-03 19:32:52 +01:00
void UpdateLocalValidity ( ) ;
2016-11-24 19:12:05 +01:00
2016-12-11 07:17:38 +01:00
void UpdateSentinelVariables ( ) ;
2016-11-24 19:12:05 +01:00
2018-03-20 12:04:59 +01:00
CAmount GetMinCollateralFee ( ) const ;
2016-11-24 19:12:05 +01:00
UniValue GetJSONObject ( ) ;
2017-09-19 16:51:38 +02:00
void Relay ( CConnman & connman ) ;
2016-11-24 19:12:05 +01:00
2016-12-11 07:17:38 +01:00
uint256 GetHash ( ) const ;
2016-11-24 19:12:05 +01:00
// GET VOTE COUNT FOR SIGNAL
int CountMatchingVotes ( vote_signal_enum_t eVoteSignalIn , vote_outcome_enum_t eVoteOutcomeIn ) const ;
int GetAbsoluteYesCount ( vote_signal_enum_t eVoteSignalIn ) const ;
int GetAbsoluteNoCount ( vote_signal_enum_t eVoteSignalIn ) const ;
int GetYesCount ( vote_signal_enum_t eVoteSignalIn ) const ;
int GetNoCount ( vote_signal_enum_t eVoteSignalIn ) const ;
int GetAbstainCount ( vote_signal_enum_t eVoteSignalIn ) const ;
2018-03-20 12:04:59 +01:00
bool GetCurrentMNVotes ( const COutPoint & mnCollateralOutpoint , vote_rec_t & voteRecord ) const ;
2016-11-24 19:12:05 +01:00
// FUNCTIONS FOR DEALING WITH DATA STRING
2018-02-15 15:43:17 +01:00
std : : string GetDataAsHexString ( ) const ;
std : : string GetDataAsPlainString ( ) const ;
2016-11-24 19:12:05 +01:00
// SERIALIZER
ADD_SERIALIZE_METHODS ;
template < typename Stream , typename Operation >
2017-09-19 21:36:55 +02:00
inline void SerializationOp ( Stream & s , Operation ser_action )
2016-11-24 19:12:05 +01:00
{
// SERIALIZE DATA FOR SAVING/LOADING OR NETWORK FUNCTIONS
2018-02-15 08:29:44 +01:00
int nVersion = s . GetVersion ( ) ;
2016-11-24 19:12:05 +01:00
READWRITE ( nHashParent ) ;
READWRITE ( nRevision ) ;
READWRITE ( nTime ) ;
READWRITE ( nCollateralHash ) ;
2018-02-16 15:54:53 +01:00
if ( nVersion = = 70208 & & ( s . GetType ( ) & SER_NETWORK ) ) {
2018-02-15 15:43:17 +01:00
// converting from/to old format
std : : string strDataHex ;
if ( ser_action . ForRead ( ) ) {
READWRITE ( LIMITED_STRING ( strDataHex , MAX_GOVERNANCE_OBJECT_DATA_SIZE ) ) ;
vchData = ParseHex ( strDataHex ) ;
} else {
strDataHex = HexStr ( vchData ) ;
READWRITE ( LIMITED_STRING ( strDataHex , MAX_GOVERNANCE_OBJECT_DATA_SIZE ) ) ;
}
} else {
// using new format directly
READWRITE ( vchData ) ;
}
2016-11-24 19:12:05 +01:00
READWRITE ( nObjectType ) ;
2018-02-16 15:54:53 +01:00
if ( nVersion = = 70208 & & ( s . GetType ( ) & SER_NETWORK ) ) {
2018-02-15 08:29:44 +01:00
// converting from/to old format
CTxIn txin ;
if ( ser_action . ForRead ( ) ) {
READWRITE ( txin ) ;
masternodeOutpoint = txin . prevout ;
} else {
txin = CTxIn ( masternodeOutpoint ) ;
READWRITE ( txin ) ;
}
} else {
// using new format directly
READWRITE ( masternodeOutpoint ) ;
}
2018-02-16 15:54:53 +01:00
if ( ! ( s . GetType ( ) & SER_GETHASH ) ) {
READWRITE ( vchSig ) ;
}
2017-09-19 21:36:55 +02:00
if ( s . GetType ( ) & SER_DISK ) {
2016-11-24 19:12:05 +01:00
// Only include these for the disk file format
LogPrint ( " gobject " , " CGovernanceObject::SerializationOp Reading/writing votes from/to disk \n " ) ;
2017-02-17 01:48:42 +01:00
READWRITE ( nDeletionTime ) ;
READWRITE ( fExpired ) ;
2016-11-24 19:12:05 +01:00
READWRITE ( mapCurrentMNVotes ) ;
READWRITE ( fileVotes ) ;
LogPrint ( " gobject " , " CGovernanceObject::SerializationOp hash = %s, vote count = %d \n " , GetHash ( ) . ToString ( ) , fileVotes . GetVoteCount ( ) ) ;
}
// AFTER DESERIALIZATION OCCURS, CACHED VARIABLES MUST BE CALCULATED MANUALLY
}
2017-01-17 21:02:38 +01:00
CGovernanceObject & operator = ( CGovernanceObject from )
{
swap ( * this , from ) ;
return * this ;
}
2016-11-24 19:12:05 +01:00
private :
// FUNCTIONS FOR DEALING WITH DATA STRING
void LoadData ( ) ;
void GetData ( UniValue & objResult ) ;
bool ProcessVote ( CNode * pfrom ,
const CGovernanceVote & vote ,
2017-09-19 16:51:38 +02:00
CGovernanceException & exception ,
CConnman & connman ) ;
2016-11-24 19:12:05 +01:00
/// Called when MN's which have voted on this object have been removed
void ClearMasternodeVotes ( ) ;
2017-09-19 16:51:38 +02:00
void CheckOrphanVotes ( CConnman & connman ) ;
2016-11-24 19:12:05 +01:00
} ;
# endif