2017-06-06 19:40:14 +02:00
pragma solidity ^ 0 . 4 . 11 ;
2017-09-13 14:41:08 +02:00
import " ./ILiquidPledgingPlugin.sol " ;
2017-09-29 12:11:13 +02:00
/// @dev This is declares a few functions from `Vault` so that the
/// `LiquidPledgingBase` contract can interface with the `Vault` contract
2017-07-13 19:12:45 +02:00
contract Vault {
function authorizePayment ( bytes32 _ref , address _dest , uint _amount ) ;
2017-08-13 13:23:00 +02:00
function ( ) payable ;
2017-07-13 19:12:45 +02:00
}
2017-06-06 19:40:14 +02:00
2017-06-26 19:54:28 +02:00
contract LiquidPledgingBase {
2017-10-03 12:20:23 +02:00
// Limits inserted to prevent large loops that could prevent canceling
2017-07-13 19:21:53 +02:00
uint constant MAX_DELEGATES = 20 ;
2017-10-03 13:37:45 +02:00
uint constant MAX_SUBCAMPAIGN_LEVEL = 20 ;
uint constant MAX_INTERCAMPAIGN_LEVEL = 20 ;
2017-06-06 19:40:14 +02:00
2017-10-04 10:24:35 +02:00
enum PledgeAdminType { Giver , Delegate , Campaign }
2017-10-04 10:29:41 +02:00
enum PaymentState { Pledged , Paying , Paid } // TODO name change Pledged
2017-09-29 12:11:13 +02:00
2017-10-04 10:24:35 +02:00
/// @dev This struct defines the details of each the PledgeAdmin, these
/// PledgeAdmins can own pledges and act as delegates
struct PledgeAdmin { // TODO name change PledgeAdmin
PledgeAdminType adminType ; // Giver, Delegate or Campaign
2017-09-29 12:11:13 +02:00
address addr ; // account or contract address for admin
2017-10-03 12:20:23 +02:00
string name ;
2017-10-04 11:40:26 +02:00
string url ;
2017-09-29 12:11:13 +02:00
uint64 commitTime ; // In seconds, used for Givers' & Delegates' vetos
2017-10-03 13:37:45 +02:00
uint64 parentCampaign ; // Only for campaigns
2017-09-29 12:11:13 +02:00
bool canceled ; //Always false except for canceled campaigns
ILiquidPledgingPlugin plugin ; // if the plugin is 0x0 then nothing happens if its a contract address than that smart contract is called via the milestone contract
2017-06-06 19:40:14 +02:00
}
2017-10-03 14:42:21 +02:00
struct Pledge {
2017-06-06 19:40:14 +02:00
uint amount ;
2017-10-04 10:24:35 +02:00
uint64 owner ; // PledgeAdmin
2017-09-29 12:11:13 +02:00
uint64 [ ] delegationChain ; // list of index numbers
2017-10-04 10:33:33 +02:00
uint64 intendedCampaign ; // TODO change the name only used for when delegates are precommiting to a campaign
uint64 commitTime ; // When the intendedCampaign will become the owner
2017-10-03 14:42:21 +02:00
uint64 oldPledge ; // this points to the Pledge[] index that the Pledge was derived from
2017-06-06 19:40:14 +02:00
PaymentState paymentState ;
}
2017-10-03 14:42:21 +02:00
Pledge [ ] pledges ;
2017-10-04 10:24:35 +02:00
PledgeAdmin [ ] admins ; //The list of pledgeAdmins 0 means there is no admin
2017-06-06 19:40:14 +02:00
Vault public vault ;
2017-10-03 14:42:21 +02:00
// this mapping allows you to search for a specific pledge's index number by the hash of that pledge
mapping ( bytes32 => uint64 ) hPledge2idx ; //TODO Fix typo
2017-06-06 19:40:14 +02:00
/////
// Modifiers
/////
modifier onlyVault ( ) {
2017-07-13 19:12:45 +02:00
require ( msg . sender == address ( vault ) ) ;
2017-06-06 19:40:14 +02:00
_ ;
}
//////
// Constructor
//////
2017-09-29 12:11:13 +02:00
/// @notice The Constructor creates the `LiquidPledgingBase` on the blockchain
/// @param _vault Where the ETH is stored that the pledges represent
2017-06-06 19:40:14 +02:00
function LiquidPledgingBase ( address _vault ) {
2017-10-04 10:24:35 +02:00
admins . length = 1 ; // we reserve the 0 admin
2017-10-03 14:42:21 +02:00
pledges . length = 1 ; // we reserve the 0 pledge
2017-06-06 19:40:14 +02:00
vault = Vault ( _vault ) ;
}
///////
2017-10-04 10:24:35 +02:00
// Adminss functions
2017-06-06 19:40:14 +02:00
//////
2017-10-03 12:20:23 +02:00
/// @notice Creates a giver.
2017-10-04 11:40:26 +02:00
function addGiver ( string name , string url , uint64 commitTime , ILiquidPledgingPlugin plugin
2017-10-03 12:20:23 +02:00
) returns ( uint64 idGiver ) {
2017-09-14 20:06:58 +02:00
2017-10-04 10:24:35 +02:00
idGiver = uint64 ( admins . length ) ;
2017-09-14 20:06:58 +02:00
2017-10-04 10:24:35 +02:00
admins . push ( PledgeAdmin (
PledgeAdminType . Giver ,
2017-06-06 19:40:14 +02:00
msg . sender ,
name ,
2017-10-04 11:40:26 +02:00
url ,
2017-06-06 19:40:14 +02:00
commitTime ,
2017-08-18 12:47:22 -04:00
0 ,
2017-09-13 14:41:08 +02:00
false ,
plugin ) ) ;
2017-09-28 08:49:10 -07:00
2017-10-03 12:20:23 +02:00
GiverAdded ( idGiver ) ;
2017-06-06 19:40:14 +02:00
}
2017-10-03 12:20:23 +02:00
event GiverAdded ( uint64 indexed idGiver ) ;
2017-06-06 19:40:14 +02:00
2017-10-03 12:20:23 +02:00
///@notice Changes the address, name or commitTime associated with a specific giver
function updateGiver (
uint64 idGiver ,
2017-06-06 19:40:14 +02:00
address newAddr ,
string newName ,
2017-10-04 11:40:26 +02:00
string newUrl ,
2017-06-06 19:40:14 +02:00
uint64 newCommitTime )
{
2017-10-04 10:24:35 +02:00
PledgeAdmin storage giver = findAdmin ( idGiver ) ;
require ( giver . adminType == PledgeAdminType . Giver ) ; //Must be a Giver
2017-10-03 14:42:21 +02:00
require ( giver . addr == msg . sender ) ; //current addr had to originate this tx
2017-10-03 12:20:23 +02:00
giver . addr = newAddr ;
giver . name = newName ;
2017-10-04 11:40:26 +02:00
giver . url = newUrl ;
2017-10-03 12:20:23 +02:00
giver . commitTime = newCommitTime ;
GiverUpdated ( idGiver ) ;
2017-06-06 19:40:14 +02:00
}
2017-10-03 12:20:23 +02:00
event GiverUpdated ( uint64 indexed idGiver ) ;
2017-06-06 19:40:14 +02:00
2017-09-29 12:11:13 +02:00
/// @notice Creates a new Delegate
2017-10-04 11:40:26 +02:00
function addDelegate ( string name , string url , uint64 commitTime , ILiquidPledgingPlugin plugin ) returns ( uint64 idDelegate ) { //TODO return index number
2017-09-14 20:06:58 +02:00
2017-10-04 10:24:35 +02:00
idDelegate = uint64 ( admins . length ) ;
2017-09-14 20:06:58 +02:00
2017-10-04 10:24:35 +02:00
admins . push ( PledgeAdmin (
PledgeAdminType . Delegate ,
2017-06-06 19:40:14 +02:00
msg . sender ,
name ,
2017-10-04 11:40:26 +02:00
url ,
2017-09-13 14:41:08 +02:00
commitTime ,
2017-06-06 19:40:14 +02:00
0 ,
2017-09-13 14:41:08 +02:00
false ,
plugin ) ) ;
2017-06-06 19:40:14 +02:00
2017-09-28 08:50:57 -07:00
DelegateAdded ( idDelegate ) ;
2017-06-06 19:40:14 +02:00
}
2017-09-28 08:49:10 -07:00
event DelegateAdded ( uint64 indexed idDelegate ) ;
2017-06-06 19:40:14 +02:00
2017-09-29 12:11:13 +02:00
///@notice Changes the address, name or commitTime associated with a specific delegate
2017-09-13 14:41:08 +02:00
function updateDelegate (
uint64 idDelegate ,
address newAddr ,
string newName ,
2017-10-04 11:40:26 +02:00
string newUrl ,
2017-09-18 15:43:09 +02:00
uint64 newCommitTime ) {
2017-10-04 10:24:35 +02:00
PledgeAdmin storage delegate = findAdmin ( idDelegate ) ;
require ( delegate . adminType == PledgeAdminType . Delegate ) ;
2017-07-13 19:12:45 +02:00
require ( delegate . addr == msg . sender ) ;
2017-06-06 19:40:14 +02:00
delegate . addr = newAddr ;
delegate . name = newName ;
2017-10-04 11:40:26 +02:00
delegate . url = newUrl ;
2017-09-13 14:41:08 +02:00
delegate . commitTime = newCommitTime ;
2017-06-06 19:40:14 +02:00
DelegateUpdated ( idDelegate ) ;
}
2017-07-09 12:04:02 -05:00
event DelegateUpdated ( uint64 indexed idDelegate ) ;
2017-06-06 19:40:14 +02:00
2017-09-29 12:11:13 +02:00
/// @notice Creates a new Campaign
2017-10-04 11:40:26 +02:00
function addCampaign ( string name , string url , address campaignAdmin , uint64 parentCampaign , uint64 commitTime , ILiquidPledgingPlugin plugin ) returns ( uint64 idCampaign ) {
2017-10-03 13:37:45 +02:00
if ( parentCampaign != 0 ) {
2017-10-04 13:02:19 +02:00
PledgeAdmin storage pa = findAdmin ( parentCampaign ) ;
require ( pa . adminType == PledgeAdminType . Campaign ) ;
require ( pa . addr == msg . sender ) ;
require ( getCampaignLevel ( pa ) < MAX_SUBCAMPAIGN_LEVEL ) ;
2017-08-18 12:47:22 -04:00
}
2017-09-14 20:06:58 +02:00
2017-10-04 10:24:35 +02:00
idCampaign = uint64 ( admins . length ) ;
2017-09-14 20:06:58 +02:00
2017-10-04 10:24:35 +02:00
admins . push ( PledgeAdmin (
PledgeAdminType . Campaign ,
campaignAdmin ,
2017-06-06 19:40:14 +02:00
name ,
2017-10-04 11:40:26 +02:00
url ,
2017-06-06 19:40:14 +02:00
commitTime ,
2017-10-03 13:37:45 +02:00
parentCampaign ,
2017-09-13 14:41:08 +02:00
false ,
plugin ) ) ;
2017-06-06 19:40:14 +02:00
2017-10-03 13:37:45 +02:00
CampaignAdded ( idCampaign ) ;
2017-06-06 19:40:14 +02:00
}
2017-10-03 13:37:45 +02:00
event CampaignAdded ( uint64 indexed idCampaign ) ;
2017-06-06 19:40:14 +02:00
2017-09-29 12:11:13 +02:00
///@notice Changes the address, name or commitTime associated with a specific Campaign
2017-10-03 13:37:45 +02:00
function updateCampaign (
uint64 idCampaign ,
2017-09-18 15:43:09 +02:00
address newAddr ,
string newName ,
2017-10-04 11:40:26 +02:00
string newUrl ,
2017-09-18 15:43:09 +02:00
uint64 newCommitTime )
{
2017-10-04 10:24:35 +02:00
PledgeAdmin storage campaign = findAdmin ( idCampaign ) ;
require ( campaign . adminType == PledgeAdminType . Campaign ) ;
2017-10-03 13:37:45 +02:00
require ( campaign . addr == msg . sender ) ;
campaign . addr = newAddr ;
campaign . name = newName ;
2017-10-04 11:40:26 +02:00
campaign . url = newUrl ;
2017-10-03 13:37:45 +02:00
campaign . commitTime = newCommitTime ;
CampaignUpdated ( idCampaign ) ;
2017-06-06 19:40:14 +02:00
}
2017-10-04 10:24:35 +02:00
event CampaignUpdated ( uint64 indexed idAdmin ) ;
2017-06-06 19:40:14 +02:00
//////////
// Public constant functions
//////////
2017-10-03 14:42:21 +02:00
/// @notice Public constant that states how many pledgess are in the system
function numberOfPledges ( ) constant returns ( uint ) {
return pledges . length - 1 ;
2017-06-06 19:40:14 +02:00
}
2017-10-03 14:42:21 +02:00
/// @notice Public constant that states the details of the specified Pledge
function getPledge ( uint64 idPledge ) constant returns (
2017-06-06 19:40:14 +02:00
uint amount ,
uint64 owner ,
uint64 nDelegates ,
2017-10-04 10:33:33 +02:00
uint64 intendedCampaign ,
2017-06-27 13:08:23 +02:00
uint64 commitTime ,
2017-10-03 14:42:21 +02:00
uint64 oldPledge ,
2017-06-06 19:40:14 +02:00
PaymentState paymentState
) {
2017-10-03 14:42:21 +02:00
Pledge storage n = findPledge ( idPledge ) ;
2017-06-06 19:40:14 +02:00
amount = n . amount ;
owner = n . owner ;
nDelegates = uint64 ( n . delegationChain . length ) ;
2017-10-04 10:33:33 +02:00
intendedCampaign = n . intendedCampaign ;
2017-06-27 13:08:23 +02:00
commitTime = n . commitTime ;
2017-10-03 14:42:21 +02:00
oldPledge = n . oldPledge ;
2017-06-06 19:40:14 +02:00
paymentState = n . paymentState ;
}
2017-09-29 12:11:13 +02:00
/// @notice Public constant that states the delegates one by one, because
/// an array cannot be returned
2017-10-03 14:42:21 +02:00
function getPledgeDelegate ( uint64 idPledge , uint idxDelegate ) constant returns (
2017-06-06 19:40:14 +02:00
uint64 idDelegate ,
address addr ,
string name
) {
2017-10-03 14:42:21 +02:00
Pledge storage n = findPledge ( idPledge ) ;
2017-06-26 19:54:28 +02:00
idDelegate = n . delegationChain [ idxDelegate - 1 ] ;
2017-10-04 10:24:35 +02:00
PledgeAdmin storage delegate = findAdmin ( idDelegate ) ;
2017-06-06 19:40:14 +02:00
addr = delegate . addr ;
name = delegate . name ;
}
2017-09-29 12:11:13 +02:00
/// @notice Public constant that states the number of admins in the system
2017-10-04 10:24:35 +02:00
function numberOfPledgeAdmins ( ) constant returns ( uint ) {
return admins . length - 1 ;
2017-06-06 19:40:14 +02:00
}
2017-10-03 12:20:23 +02:00
/// @notice Public constant that states the details of the specified admin
2017-10-04 10:24:35 +02:00
function getPledgeAdmin ( uint64 idAdmin ) constant returns (
PledgeAdminType adminType ,
2017-06-06 19:40:14 +02:00
address addr ,
string name ,
2017-10-04 11:40:26 +02:00
string url ,
2017-06-06 19:40:14 +02:00
uint64 commitTime ,
2017-10-03 13:37:45 +02:00
uint64 parentCampaign ,
2017-09-26 12:06:45 -07:00
bool canceled ,
address plugin )
2017-06-06 19:40:14 +02:00
{
2017-10-04 10:24:35 +02:00
PledgeAdmin storage m = findAdmin ( idAdmin ) ;
adminType = m . adminType ;
2017-06-06 19:40:14 +02:00
addr = m . addr ;
name = m . name ;
2017-10-04 11:40:26 +02:00
url = m . url ;
2017-06-06 19:40:14 +02:00
commitTime = m . commitTime ;
2017-10-03 13:37:45 +02:00
parentCampaign = m . parentCampaign ;
2017-06-06 19:40:14 +02:00
canceled = m . canceled ;
2017-09-26 12:26:26 -07:00
plugin = address ( m . plugin ) ;
2017-06-06 19:40:14 +02:00
}
////////
// Private methods
///////
2017-10-03 14:42:21 +02:00
/// @notice All pledges technically exist... but if the pledge hasn't been
2017-09-29 12:11:13 +02:00
/// created in this system yet then it wouldn't be in the hash array
2017-10-03 14:42:21 +02:00
/// hPledge2idx[]; this creates a Pledge with and amount of 0 if one is not
2017-10-03 12:20:23 +02:00
/// created already...
2017-10-04 12:55:46 +02:00
function findOrCreatePledge (
2017-06-06 19:40:14 +02:00
uint64 owner ,
uint64 [ ] delegationChain ,
2017-10-04 10:33:33 +02:00
uint64 intendedCampaign ,
2017-06-27 13:08:23 +02:00
uint64 commitTime ,
2017-10-03 14:42:21 +02:00
uint64 oldPledge ,
2017-06-06 19:40:14 +02:00
PaymentState paid
) internal returns ( uint64 )
{
2017-10-04 10:33:33 +02:00
bytes32 hPledge = sha3 ( owner , delegationChain , intendedCampaign , commitTime , oldPledge , paid ) ;
2017-10-03 14:42:21 +02:00
uint64 idx = hPledge2idx [ hPledge ] ;
2017-06-06 19:40:14 +02:00
if ( idx > 0 ) return idx ;
2017-10-03 14:42:21 +02:00
idx = uint64 ( pledges . length ) ;
hPledge2idx [ hPledge ] = idx ;
2017-10-04 10:33:33 +02:00
pledges . push ( Pledge ( 0 , owner , delegationChain , intendedCampaign , commitTime , oldPledge , paid ) ) ;
2017-06-06 19:40:14 +02:00
return idx ;
}
2017-10-04 10:24:35 +02:00
function findAdmin ( uint64 idAdmin ) internal returns ( PledgeAdmin storage ) {
require ( idAdmin < admins . length ) ;
return admins [ idAdmin ] ;
2017-06-06 19:40:14 +02:00
}
2017-10-03 14:42:21 +02:00
function findPledge ( uint64 idPledge ) internal returns ( Pledge storage ) {
require ( idPledge < pledges . length ) ;
return pledges [ idPledge ] ;
2017-06-06 19:40:14 +02:00
}
2017-07-09 12:04:02 -05:00
// a constant for the case that a delegate is requested that is not a delegate in the system
uint64 constant NOTFOUND = 0xFFFFFFFFFFFFFFFF ;
2017-07-13 19:21:53 +02:00
2017-07-09 12:04:02 -05:00
// helper function that searches the delegationChain fro a specific delegate and
2017-09-29 12:11:13 +02:00
// level of delegation returns their idx in the delegation chain which reflect their level of authority
2017-10-03 14:42:21 +02:00
function getDelegateIdx ( Pledge n , uint64 idDelegate ) internal returns ( uint64 ) {
2017-07-09 12:04:02 -05:00
for ( uint i = 0 ; i < n . delegationChain . length ; i ++ ) {
if ( n . delegationChain [ i ] == idDelegate ) return uint64 ( i ) ;
}
return NOTFOUND ;
}
2017-10-03 14:42:21 +02:00
// helper function that returns the pledge level solely to check that transfers
2017-10-03 13:37:45 +02:00
// between Campaigns not violate MAX_INTERCAMPAIGN_LEVEL
2017-10-03 14:42:21 +02:00
function getPledgeLevel ( Pledge n ) internal returns ( uint ) {
if ( n . oldPledge == 0 ) return 0 ; //changed
Pledge storage oldN = findPledge ( n . oldPledge ) ;
return getPledgeLevel ( oldN ) + 1 ;
2017-07-09 12:04:02 -05:00
}
2017-08-18 12:47:22 -04:00
2017-09-13 14:41:08 +02:00
// helper function that returns the max commit time of the owner and all the
// delegates
2017-10-03 14:42:21 +02:00
function maxCommitTime ( Pledge n ) internal returns ( uint commitTime ) {
2017-10-04 10:24:35 +02:00
PledgeAdmin storage m = findAdmin ( n . owner ) ;
2017-09-13 14:41:08 +02:00
commitTime = m . commitTime ;
for ( uint i = 0 ; i < n . delegationChain . length ; i ++ ) {
2017-10-04 10:24:35 +02:00
m = findAdmin ( n . delegationChain [ i ] ) ;
2017-09-13 14:41:08 +02:00
if ( m . commitTime > commitTime ) commitTime = m . commitTime ;
}
}
2017-08-18 12:47:22 -04:00
2017-10-03 13:37:45 +02:00
// helper function that returns the campaign level solely to check that there
// are not too many Campaigns that violate MAX_SUBCAMPAIGNS_LEVEL
2017-10-04 10:24:35 +02:00
function getCampaignLevel ( PledgeAdmin m ) internal returns ( uint ) {
assert ( m . adminType == PledgeAdminType . Campaign ) ;
2017-10-03 13:37:45 +02:00
if ( m . parentCampaign == 0 ) return ( 1 ) ;
2017-10-04 10:24:35 +02:00
PledgeAdmin storage parentNM = findAdmin ( m . parentCampaign ) ;
2017-10-03 13:37:45 +02:00
return getCampaignLevel ( parentNM ) ;
2017-08-18 12:47:22 -04:00
}
2017-10-03 13:37:45 +02:00
function isCampaignCanceled ( uint64 campaignId ) constant returns ( bool ) {
2017-10-04 10:24:35 +02:00
PledgeAdmin storage m = findAdmin ( campaignId ) ;
if ( m . adminType == PledgeAdminType . Giver ) return false ;
assert ( m . adminType == PledgeAdminType . Campaign ) ;
2017-08-18 12:47:22 -04:00
if ( m . canceled ) return true ;
2017-10-03 13:37:45 +02:00
if ( m . parentCampaign == 0 ) return false ;
return isCampaignCanceled ( m . parentCampaign ) ;
2017-08-18 12:47:22 -04:00
}
2017-10-03 13:37:45 +02:00
// @notice A helper function for canceling campaigns
2017-10-03 14:42:21 +02:00
// @param idPledge the pledge that may or may not be canceled
function getOldestPledgeNotCanceled ( uint64 idPledge ) internal constant returns ( uint64 ) { //todo rename
if ( idPledge == 0 ) return 0 ;
Pledge storage n = findPledge ( idPledge ) ;
2017-10-04 10:24:35 +02:00
PledgeAdmin storage admin = findAdmin ( n . owner ) ;
if ( admin . adminType == PledgeAdminType . Giver ) return idPledge ;
2017-08-18 12:47:22 -04:00
2017-10-04 10:24:35 +02:00
assert ( admin . adminType == PledgeAdminType . Campaign ) ;
2017-08-18 12:47:22 -04:00
2017-10-03 14:42:21 +02:00
if ( ! isCampaignCanceled ( n . owner ) ) return idPledge ;
2017-08-18 12:47:22 -04:00
2017-10-03 14:42:21 +02:00
return getOldestPledgeNotCanceled ( n . oldPledge ) ;
2017-06-06 19:40:14 +02:00
}
2017-10-04 10:24:35 +02:00
function checkAdminOwner ( PledgeAdmin m ) internal constant {
2017-09-18 15:43:09 +02:00
require ( ( msg . sender == m . addr ) || ( msg . sender == address ( m . plugin ) ) ) ;
}
2017-06-06 19:40:14 +02:00
}