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
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-04-09 21:57:53 +02:00
using namespace std ;
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-04-09 21:57:53 +02:00
extern CCriticalSection cs_budget ;
2016-04-09 22:31:01 +02:00
class CGovernanceManager ;
2016-04-16 19:19:17 +02:00
class CGovernanceObject ;
2016-05-23 19:53:05 +02:00
class CGovernanceVote ;
2016-04-14 20:53:46 +02:00
class CNode ;
2016-04-09 21:57:53 +02:00
2016-05-23 20:16:46 +02:00
static const CAmount GOVERNANCE_FEE_TX = ( 0.1 * COIN ) ;
2016-08-17 09:08:25 +02:00
static const int64_t GOVERNANCE_FEE_CONFIRMATIONS = 1 ; //todo 12.1 -- easy testing
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-23 20:10:20 +02:00
extern std : : map < uint256 , int64_t > mapAskedForGovernanceObject ;
2016-04-10 16:46:19 +02:00
extern CGovernanceManager governance ;
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-04-09 21:57:53 +02:00
//Check the collateral transaction for the budget proposal/finalized budget
2016-05-31 22:00:01 +02:00
extern bool IsCollateralValid ( uint256 nTxCollateralHash , uint256 nExpectedHash , std : : string & strError , 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
{
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-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
// note: move to private for better encapsulation
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-08-17 09:08:25 +02:00
public :
// critical section to protect the inner data structures
mutable CCriticalSection cs ;
CGovernanceManager ( )
: mapCollateral ( ) ,
pCurrentBlockIndex ( NULL ) ,
nTimeLastDiff ( 0 ) ,
nCachedBlockHeight ( 0 ) ,
mapObjects ( ) ,
cs ( )
{ }
2016-04-09 21:57:53 +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-05-13 18:03:01 +02:00
int sizeProposals ( ) { return ( int ) mapObjects . 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-05-24 20:11:59 +02:00
CGovernanceObject * FindGovernanceObject ( const std : : string & strName ) ;
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-05-23 23:39:10 +02:00
int CountMatchingVotes ( CGovernanceObject & govobj , int nVoteSignalIn , int 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-04-09 21:57:53 +02:00
void Clear ( ) {
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-04-09 21:57:53 +02:00
}
2016-08-17 09:08:25 +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-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-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 ) ;
// int GetVoteCountByHash(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-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-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
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-08-17 09:08:25 +02:00
bool fCachedLocalValidity ; // is valid by blockchain
2016-05-23 23:39:10 +02:00
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-08-17 09:08:25 +02:00
CGovernanceObject ( uint256 nHashParentIn , int nRevisionIn , std : : string strNameIn , 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-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 ) ;
void UpdateLocalValidity ( const CBlockIndex * pCurrentBlockIndex ) ;
void UpdateSentinelVariables ( const CBlockIndex * pCurrentBlockIndex ) ;
int GetObjectType ( ) ;
int GetObjectSubtype ( ) ;
std : : string GetName ( ) { return strName ; }
2016-04-26 06:08:36 +02:00
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-05-23 19:53:05 +02:00
int GetAbsoluteYesCount ( int nVoteSignalIn ) ;
2016-06-08 08:57:16 +02:00
int GetAbsoluteNoCount ( int nVoteSignalIn ) ;
2016-05-23 19:53:05 +02:00
int GetYesCount ( int nVoteSignalIn ) ;
int GetNoCount ( int nVoteSignalIn ) ;
int GetAbstainCount ( int nVoteSignalIn ) ;
2016-04-17 02:29:41 +02:00
2016-08-17 09:08:25 +02:00
// FUNCTIONS FOR DEALING WITH DATA STRING
2016-05-13 18:03:01 +02:00
2016-08-17 09:08:25 +02:00
void LoadData ( ) ;
bool SetData ( std : : string & strError , std : : string strDataIn ) ;
bool GetData ( UniValue & objResult ) ;
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 ) ;
READWRITE ( LIMITED_STRING ( strName , 64 ) ) ;
2016-04-09 21:57:53 +02:00
READWRITE ( nTime ) ;
2016-08-17 09:08:25 +02:00
READWRITE ( nCollateralHash ) ;
2016-05-23 19:53:05 +02:00
READWRITE ( strData ) ;
2016-08-17 09:08:25 +02:00
READWRITE ( nObjectType ) ;
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
}
} ;
# endif