Source Code
Overview
ETH Balance
0 ETH
Eth Value
$0.00Latest 25 from a total of 456 transactions
| Transaction Hash |
Method
|
Block
|
From
|
|
To
|
||||
|---|---|---|---|---|---|---|---|---|---|
| Execute Proposal | 15689798 | 1236 days ago | IN | 0 ETH | 0.00451745 | ||||
| Vote | 15669713 | 1239 days ago | IN | 0 ETH | 0.00157995 | ||||
| Vote | 15668317 | 1239 days ago | IN | 0 ETH | 0.00213234 | ||||
| Vote | 15667798 | 1239 days ago | IN | 0 ETH | 0.00102651 | ||||
| Vote | 15666745 | 1239 days ago | IN | 0 ETH | 0.00094006 | ||||
| Vote | 15665912 | 1239 days ago | IN | 0 ETH | 0.00056556 | ||||
| Vote | 15664779 | 1239 days ago | IN | 0 ETH | 0.00105673 | ||||
| Vote | 15664428 | 1239 days ago | IN | 0 ETH | 0.00061861 | ||||
| Vote | 15663467 | 1240 days ago | IN | 0 ETH | 0.00039567 | ||||
| Vote | 15662951 | 1240 days ago | IN | 0 ETH | 0.00041 | ||||
| Vote | 15662916 | 1240 days ago | IN | 0 ETH | 0.00046739 | ||||
| Vote | 15661498 | 1240 days ago | IN | 0 ETH | 0.00049509 | ||||
| Vote | 15661073 | 1240 days ago | IN | 0 ETH | 0.00059533 | ||||
| Vote | 15659875 | 1240 days ago | IN | 0 ETH | 0.00046387 | ||||
| Vote | 15659711 | 1240 days ago | IN | 0 ETH | 0.00025466 | ||||
| Vote | 15659534 | 1240 days ago | IN | 0 ETH | 0.00041311 | ||||
| Vote | 15659407 | 1240 days ago | IN | 0 ETH | 0.00035815 | ||||
| Vote | 15658781 | 1240 days ago | IN | 0 ETH | 0.00030304 | ||||
| Vote | 15658456 | 1240 days ago | IN | 0 ETH | 0.00019051 | ||||
| Vote | 15658455 | 1240 days ago | IN | 0 ETH | 0.00030841 | ||||
| Vote | 15657816 | 1240 days ago | IN | 0 ETH | 0.0006331 | ||||
| Vote | 15657452 | 1240 days ago | IN | 0 ETH | 0.00046794 | ||||
| Vote | 15656940 | 1241 days ago | IN | 0 ETH | 0.00049424 | ||||
| Vote | 15655972 | 1241 days ago | IN | 0 ETH | 0.00066063 | ||||
| Vote | 15654802 | 1241 days ago | IN | 0 ETH | 0.00078506 |
View more zero value Internal Transactions in Advanced View mode
Advanced mode:
Loading...
Loading
Loading...
Loading
Cross-Chain Transactions
Loading...
Loading
Contract Name:
Voting
Compiler Version
v0.8.6+commit.11564f7e
Optimization Enabled:
Yes with 1600 runs
Other Settings:
default evmVersion
Contract Source Code (Solidity Standard Json-Input format)
// SPDX-License-Identifier: UNLICENSED
pragma solidity ^0.8.0;
pragma abicoder v2;
// OpenZeppelin v4
import { Staking } from "./Staking.sol";
import { Delegator } from "./Delegator.sol";
/**
* @title Voting
* @author Railgun Contributors
* @notice Governance contract for railgun, handles voting.
*/
contract Voting {
// Time offsets from publish time, offset times are relative to voteCallTime
uint256 public constant SPONSOR_WINDOW = 30 days;
uint256 public constant VOTING_START_OFFSET = 2 days; // Should be > interval size of staking snapshots
uint256 public constant VOTING_YAY_END_OFFSET = 5 days;
uint256 public constant VOTING_NAY_END_OFFSET = 6 days;
uint256 public constant EXECUTION_START_OFFSET = 7 days;
uint256 public constant EXECUTION_END_OFFSET = 14 days;
// Threshold constants
uint256 public constant QUORUM = 4000000e18; // 4 million, 18 decimal places
uint256 public constant PROPOSAL_SPONSOR_THRESHOLD = 1000000e18; // 1 million, 18 decimal places
// Proposal has been created
event Proposal(uint256 indexed id, address indexed proposer);
// Proposal has been sponsored
event Sponsorship(uint256 indexed id, address indexed sponsor, uint256 amount);
// Proposal has been unsponsored
event SponsorshipRevocation(uint256 indexed id, address indexed sponsor, uint256 amount);
// Proposal vote called
event VoteCall(uint256 indexed id);
// Vote cast on proposal
event VoteCast(uint256 indexed id, address indexed voter, bool affirmative, uint256 votes);
// Proposal executed
event Execution(uint256 indexed id);
// Function call
struct Call {
address callContract;
bytes data;
uint256 value;
}
// Governance proposals
struct ProposalStruct {
// Proposal Data
address proposer;
string proposalDocument; // IPFS hash
Call[] actions;
// Event timestamps
uint256 publishTime;
uint256 voteCallTime; // If vote call time is 0, proposal hasn't gone to vote
// Sponsorship info
uint256 sponsorship;
mapping(address => uint256) sponsors;
// Execution status
bool executed;
// Vote data
// Amount of voting power used for accounts, used for fractional voting from contracts
mapping(address => uint256) voted;
uint256 yayVotes;
uint256 nayVotes;
// Staking snapshots
uint256 sponsorInterval;
uint256 votingInterval;
}
// Proposals id => proposal data
ProposalStruct[] public proposals;
/* solhint-disable var-name-mixedcase */
Staking public immutable STAKING_CONTRACT;
Delegator public immutable DELEGATOR_CONTRACT;
/* solhint-enable var-name-mixedcase */
/**
* @notice Sets governance token ID and delegator contract
*/
constructor(Staking _stakingContract, Delegator _delegator) {
STAKING_CONTRACT = _stakingContract;
DELEGATOR_CONTRACT = _delegator;
}
/**
* @notice Gets length of proposals array
* @return length
*/
function proposalsLength() external view returns (uint256) {
return proposals.length;
}
/**
* @notice Gets actions from proposal document
* @dev Gets actions from proposal as nested arrays won't be returned on public getter
* @param _id - Proposal to get actions of
* @return actions
*/
function getActions(uint256 _id) external view returns (Call[] memory) {
return proposals[_id].actions;
}
/**
* @notice Gets sponsor amount an account has given to a proposal
* @dev Gets actions from proposal as mappings wont be returned on public getter
* @param _id - Proposal to get sponsor amount of
* @param _account - Account to get sponsor amount for
* @return sponsor amount
*/
function getSponsored(uint256 _id, address _account) external view returns (uint256) {
return proposals[_id].sponsors[_account];
}
/**
* @notice Creates governance proposal
* @param _proposalDocument - IPFS multihash of proposal document
* @param _actions - actions to take
*/
function createProposal(string calldata _proposalDocument, Call[] calldata _actions) external returns (uint256) {
// Don't allow proposals with no actions
require(_actions.length > 0, "Voting: No actions specified");
uint256 proposalID = proposals.length;
ProposalStruct storage proposal = proposals.push();
// Store proposer
proposal.proposer = msg.sender;
// Store proposal document
proposal.proposalDocument = _proposalDocument;
// Store published time
proposal.publishTime = block.timestamp;
// Store sponsor voting snapshot interval
proposal.sponsorInterval = STAKING_CONTRACT.currentInterval();
// Loop over actions and copy manually as solidity doesn't support copying structs
for (uint256 i = 0; i < _actions.length; i++) {
proposal.actions.push(Call(
_actions[i].callContract,
_actions[i].data,
_actions[i].value
));
}
// Emit event
emit Proposal(proposalID, msg.sender);
return proposalID;
}
/**
* @notice Sponsor proposal
* @param _id - id of proposal to sponsor
* @param _amount - amount to sponsor with
* @param _hint - hint for snapshot search
*/
function sponsorProposal(uint256 _id, uint256 _amount, uint256 _hint) external {
ProposalStruct storage proposal = proposals[_id];
// Check proposal hasn't already gone to vote
require(proposal.voteCallTime == 0, "Voting: Gone to vote");
// Check proposal is still in sponsor window
require(block.timestamp < proposal.publishTime + SPONSOR_WINDOW, "Voting: Sponsoring window passed");
// Get address sponsor voting power
Staking.AccountSnapshot memory snapshot = STAKING_CONTRACT.accountSnapshotAt(
msg.sender,
proposal.sponsorInterval,
_hint
);
// Can't sponsor with more than voting power
require(proposal.sponsors[msg.sender] + _amount <= snapshot.votingPower, "Voting: Not enough voting power");
// Update address sponsorship amount on proposal
proposal.sponsors[msg.sender] += _amount;
// Update sponsor total
proposal.sponsorship += _amount;
// Emit event
emit Sponsorship(_id, msg.sender, _amount);
}
/**
* @notice Unsponsor proposal
* @param _id - id of proposal to sponsor
* @param _amount - amount to sponsor with
*/
function unsponsorProposal(uint256 _id, uint256 _amount) external {
ProposalStruct storage proposal = proposals[_id];
// Check proposal hasn't already gone to vote
require(proposal.voteCallTime == 0, "Voting: Gone to vote");
// Check proposal is still in sponsor window
require(block.timestamp < proposal.publishTime + SPONSOR_WINDOW, "Voting: Sponsoring window passed");
// Can't unsponsor more than sponsored
require(_amount <= proposal.sponsors[msg.sender], "Voting: Amount greater than sponsored");
// Update address sponsorship amount on proposal
proposal.sponsors[msg.sender] -= _amount;
// Update sponsor total
proposal.sponsorship -= _amount;
// Emit event
emit SponsorshipRevocation(_id, msg.sender, _amount);
}
/**
* @notice Call vote
* @param _id - id of proposal to call to vote
*/
function callVote(uint256 _id) external {
ProposalStruct storage proposal = proposals[_id];
// Check proposal hasn't exceeded sponsor window
require(block.timestamp < proposal.publishTime + SPONSOR_WINDOW, "Voting: Sponsoring window passed");
// Check proposal hasn't already gone to vote
require(proposal.voteCallTime == 0, "Voting: Proposal already gone to vote");
// Proposal must meet sponsorship threshold
require(proposal.sponsorship >= PROPOSAL_SPONSOR_THRESHOLD, "Voting: Sponsor threshold not met");
// Log vote time (also marks proposal as ready to vote)
proposal.voteCallTime = block.timestamp;
// Log governance token snapshot interval
// VOTING_START_OFFSET must be greater than snapshot interval of governance token for this to work correctly
proposal.votingInterval = STAKING_CONTRACT.currentInterval();
// Emit event
emit VoteCall(_id);
}
/**
* @notice Vote on proposal
* @param _id - id of proposal to call to vote
* @param _amount - amount of voting power to allocate
* @param _affirmative - whether to vote yay (true) or nay (false) on this proposal
* @param _hint - hint for snapshot search
*/
function vote(uint256 _id, uint256 _amount, bool _affirmative, uint256 _hint) external {
ProposalStruct storage proposal = proposals[_id];
// Check vote has been called
require(proposal.voteCallTime > 0, "Voting: Vote hasn't been called for this proposal");
// Check Voting window has opened
require(block.timestamp > proposal.voteCallTime + VOTING_START_OFFSET, "Voting: Voting window hasn't opened");
// Check voting window hasn't closed (voting window length conditional on )
if(_affirmative) {
require(block.timestamp < proposal.voteCallTime + VOTING_YAY_END_OFFSET, "Voting: Affirmative voting window has closed");
} else {
require(block.timestamp < proposal.voteCallTime + VOTING_NAY_END_OFFSET, "Voting: Negative voting window has closed");
}
// Get address voting power
Staking.AccountSnapshot memory snapshot = STAKING_CONTRACT.accountSnapshotAt(
msg.sender,
proposal.votingInterval,
_hint
);
// Check address isn't voting with more voting power than it has
require(proposal.voted[msg.sender] + _amount <= snapshot.votingPower, "Voting: Not enough voting power to cast this vote");
// Update account voted amount
proposal.voted[msg.sender] += _amount;
// Update voting totals
if (_affirmative) {
proposal.yayVotes += _amount;
} else {
proposal.nayVotes += _amount;
}
// Emit event
emit VoteCast(_id, msg.sender, _affirmative, _amount);
}
/**
* @notice Execute proposal
* @param _id - id of proposal to execute
*/
function executeProposal(uint256 _id) external {
ProposalStruct storage proposal = proposals[_id];
// Check proposal has been called to vote
require(proposal.voteCallTime > 0, "Voting: Vote hasn't been called for this proposal");
// Check quorum has been reached
require(proposal.yayVotes + proposal.nayVotes >= QUORUM, "Voting: Quorum hasn't been reached");
// Check vote passed
require(proposal.yayVotes > proposal.nayVotes, "Voting: Proposal hasn't passed vote");
// Check we're in execution window
require(block.timestamp > proposal.voteCallTime + EXECUTION_START_OFFSET, "Voting: Execution window hasn't opened");
require(block.timestamp < proposal.voteCallTime + EXECUTION_END_OFFSET, "Voting: Execution window has closed");
// Check proposal hasn't been executed before
require(!proposal.executed, "Voting: Proposal has already been executed");
// Mark proposal as executed
proposal.executed = true;
Call[] storage actions = proposal.actions;
// Loop over actions and execute
for (uint256 i = 0; i < actions.length; i++) {
// Execute action
(bool successful, bytes memory returnData) = DELEGATOR_CONTRACT.callContract(
actions[i].callContract,
actions[i].data,
actions[i].value
);
// If an action fails to execute, catch and bubble up reason with revert
if (!successful) {
bytes memory revertData = abi.encode(i, returnData);
// solhint-disable-next-line no-inline-assembly
assembly {
revert (add (32, revertData), mload (revertData))
}
}
}
// Emit event
emit Execution(_id);
}
}// SPDX-License-Identifier: UNLICENSED
pragma solidity ^0.8.0;
pragma abicoder v2;
// OpenZeppelin v4
import { Math } from "@openzeppelin/contracts/utils/math/Math.sol";
import { IERC20 } from "@openzeppelin/contracts/token/ERC20/IERC20.sol";
import { SafeERC20 } from "@openzeppelin/contracts/token/ERC20/utils/SafeERC20.sol";
/**
* @title Snapshot
* @author Railgun Contributors
* @notice Governance contract for railgun, handles staking, voting power, and snapshotting
* @dev Snapshots cannot be taken during interval 0
* wait till interval 1 before utilising snapshots
*/
contract Staking {
using SafeERC20 for IERC20;
// Constants
uint256 public constant STAKE_LOCKTIME = 30 days;
uint256 public constant SNAPSHOT_INTERVAL = 1 days;
// Staking token
IERC20 public stakingToken;
// Time of deployment
// solhint-disable-next-line var-name-mixedcase
uint256 public immutable DEPLOY_TIME = block.timestamp;
// New stake screated
event Stake(address indexed account, uint256 indexed stakeID, uint256 amount);
// Stake unlocked (coins removed from voting pool, 30 day delay before claiming is allowed)
event Unlock(address indexed account, uint256 indexed stakeID);
// Stake claimed
event Claim(address indexed account, uint256 indexed stakeID);
// Delegate claimed
event Delegate(address indexed owner, address indexed _from, address indexed to, uint256 stakeID, uint256 amount);
// Total staked
uint256 public totalStaked = 0;
// Snapshots for globals
struct GlobalsSnapshot {
uint256 interval;
uint256 totalVotingPower;
uint256 totalStaked;
}
GlobalsSnapshot[] private globalsSnapshots;
// Stake
struct StakeStruct {
address delegate; // Address stake voting power is delegated to
uint256 amount; // Amount of tokens on this stake
uint256 staketime; // Time this stake was created
uint256 locktime; // Time this stake can be claimed (if 0, unlock hasn't been initiated)
uint256 claimedTime; // Time this stake was claimed (if 0, stake hasn't been claimed)
}
// Stake mapping
// address => stakeID => stake
mapping(address => StakeStruct[]) public stakes;
// Voting power for each account
mapping(address => uint256) public votingPower;
// Snapshots for accounts
struct AccountSnapshot {
uint256 interval;
uint256 votingPower;
}
mapping(address => AccountSnapshot[]) private accountSnapshots;
/**
* @notice Sets staking token
* @param _stakingToken - time to get interval of
*/
constructor(IERC20 _stakingToken) {
stakingToken = _stakingToken;
// Use address 0 to store inverted totalVotingPower
votingPower[address(0)] = type(uint256).max;
}
/**
* @notice Gets total voting power in system
* @return totalVotingPower
*/
function totalVotingPower() public view returns (uint256) {
return ~votingPower[address(0)];
}
/**
* @notice Gets length of stakes array for address
* @param _account - address to retrieve stakes array of
* @return length
*/
function stakesLength(address _account) external view returns (uint256) {
return stakes[_account].length;
}
/**
* @notice Gets interval at time
* @param _time - time to get interval of
* @return interval
*/
function intervalAtTime(uint256 _time) public view returns (uint256) {
require(_time >= DEPLOY_TIME, "Staking: Requested time is before contract was deployed");
return (_time - DEPLOY_TIME) / SNAPSHOT_INTERVAL;
}
/**
* @notice Gets current interval
* @return interval
*/
function currentInterval() public view returns (uint256) {
return intervalAtTime(block.timestamp);
}
/**
* @notice Returns interval of latest global snapshot
* @return Latest global snapshot interval
*/
function latestGlobalsSnapshotInterval() public view returns (uint256) {
if (globalsSnapshots.length > 0) {
// If a snapshot exists return the interval it was taken
return globalsSnapshots[globalsSnapshots.length - 1].interval;
} else {
// Else default to 0
return 0;
}
}
/**
* @notice Returns interval of latest account snapshot
* @param _account - account to get latest snapshot of
* @return Latest account snapshot interval
*/
function latestAccountSnapshotInterval(address _account) public view returns (uint256) {
if (accountSnapshots[_account].length > 0) {
// If a snapshot exists return the interval it was taken
return accountSnapshots[_account][accountSnapshots[_account].length - 1].interval;
} else {
// Else default to 0
return 0;
}
}
/**
* @notice Returns length of snapshot array
* @param _account - account to get snapshot array length of
* @return Snapshot array length
*/
function accountSnapshotLength(address _account) external view returns (uint256) {
return accountSnapshots[_account].length;
}
/**
* @notice Returns length of snapshot array
* @return Snapshot array length
*/
function globalsSnapshotLength() external view returns (uint256) {
return globalsSnapshots.length;
}
/**
* @notice Returns global snapshot at index
* @param _index - account to get latest snapshot of
* @return Globals snapshot
*/
function globalsSnapshot(uint256 _index) external view returns (GlobalsSnapshot memory) {
return globalsSnapshots[_index];
}
/**
* @notice Returns account snapshot at index
* @param _account - account to get snapshot of
* @param _index - index to get snapshot at
* @return Account snapshot
*/
function accountSnapshot(address _account, uint256 _index) external view returns (AccountSnapshot memory) {
return accountSnapshots[_account][_index];
}
/**
* @notice Checks if accoutn and globals snapshots need updating and updates
* @param _account - Account to take snapshot for
*/
function snapshot(address _account) internal {
uint256 _currentInterval = currentInterval();
// If latest global snapshot is less than current interval, push new snapshot
if(latestGlobalsSnapshotInterval() < _currentInterval) {
globalsSnapshots.push(GlobalsSnapshot(
_currentInterval,
totalVotingPower(),
totalStaked
));
}
// If latest account snapshot is less than current interval, push new snapshot
// Skip if account is 0 address
if(_account != address(0) && latestAccountSnapshotInterval(_account) < _currentInterval) {
accountSnapshots[_account].push(AccountSnapshot(
_currentInterval,
votingPower[_account]
));
}
}
/**
* @notice Moves voting power in response to delegation or stake/unstake
* @param _from - account to move voting power fom
* @param _to - account to move voting power to
* @param _amount - amount of voting power to move
*/
function moveVotingPower(address _from, address _to, uint256 _amount) internal {
votingPower[_from] -= _amount;
votingPower[_to] += _amount;
}
/**
* @notice Updates vote delegation
* @param _stakeID - stake to delegate
* @param _to - address to delegate to
*/
function delegate(uint256 _stakeID, address _to) public {
StakeStruct storage _stake = stakes[msg.sender][_stakeID];
require(
_stake.staketime != 0,
"Staking: Stake doesn't exist"
);
require(
_stake.locktime == 0,
"Staking: Stake unlocked"
);
require(
_to != address(0),
"Staking: Can't delegate to 0 address"
);
if (_stake.delegate != _to) {
// Check if snapshot needs to be taken
snapshot(_stake.delegate); // From
snapshot(_to); // To
// Move voting power to delegatee
moveVotingPower(
_stake.delegate,
_to,
_stake.amount
);
// Emit event
emit Delegate(msg.sender, _stake.delegate, _to, _stakeID, _stake.amount);
// Update delegation
_stake.delegate = _to;
}
}
/**
* @notice Delegates voting power of stake back to self
* @param _stakeID - stake to delegate back to self
*/
function undelegate(uint256 _stakeID) external {
delegate(_stakeID, msg.sender);
}
/**
* @notice Gets global state at interval
* @param _interval - interval to get state at
* @return state
*/
function globalsSnapshotAtSearch(uint256 _interval) internal view returns (GlobalsSnapshot memory) {
require(_interval <= currentInterval(), "Staking: Interval out of bounds");
// Index of element
uint256 index;
// High/low for binary serach to find index
// https://en.wikipedia.org/wiki/Binary_search_algorithm
uint256 low = 0;
uint256 high = globalsSnapshots.length;
while (low < high) {
uint256 mid = Math.average(low, high);
// Note that mid will always be strictly less than high (i.e. it will be a valid array index)
// because Math.average rounds down (it does integer division with truncation).
if (globalsSnapshots[mid].interval > _interval) {
high = mid;
} else {
low = mid + 1;
}
}
// At this point `low` is the exclusive upper bound. Find the inclusive upper bounds and set to index
if (low > 0 && globalsSnapshots[low - 1].interval == _interval) {
return globalsSnapshots[low - 1];
} else {
index = low;
}
// If index is equal to snapshot array length, then no update was made after the requested
// snapshot interval. This means the latest value is the right one.
if (index == globalsSnapshots.length) {
return GlobalsSnapshot(
_interval,
totalVotingPower(),
totalStaked
);
} else {
return globalsSnapshots[index];
}
}
/**
* @notice Gets global state at interval
* @param _interval - interval to get state at
* @param _hint - off-chain computed index of interval
* @return state
*/
function globalsSnapshotAt(uint256 _interval, uint256 _hint) external view returns (GlobalsSnapshot memory) {
require(_interval <= currentInterval(), "Staking: Interval out of bounds");
// Check if hint is correct, else fall back to binary search
if (
_hint <= globalsSnapshots.length
&& (_hint == 0 || globalsSnapshots[_hint - 1].interval < _interval)
&& (_hint == globalsSnapshots.length || globalsSnapshots[_hint].interval >= _interval)
) {
// The hint is correct
if (_hint < globalsSnapshots.length)
return globalsSnapshots[_hint];
else
return GlobalsSnapshot (_interval, totalVotingPower(), totalStaked);
} else return globalsSnapshotAtSearch (_interval);
}
/**
* @notice Gets account state at interval
* @param _account - account to get state for
* @param _interval - interval to get state at
* @return state
*/
function accountSnapshotAtSearch(address _account, uint256 _interval) internal view returns (AccountSnapshot memory) {
require(_interval <= currentInterval(), "Staking: Interval out of bounds");
// Get account snapshots array
AccountSnapshot[] storage snapshots = accountSnapshots[_account];
// Index of element
uint256 index;
// High/low for binary serach to find index
// https://en.wikipedia.org/wiki/Binary_search_algorithm
uint256 low = 0;
uint256 high = snapshots.length;
while (low < high) {
uint256 mid = Math.average(low, high);
// Note that mid will always be strictly less than high (i.e. it will be a valid array index)
// because Math.average rounds down (it does integer division with truncation).
if (snapshots[mid].interval > _interval) {
high = mid;
} else {
low = mid + 1;
}
}
// At this point `low` is the exclusive upper bound. Find the inclusive upper bounds and set to index
if (low > 0 && snapshots[low - 1].interval == _interval) {
return snapshots[low - 1];
} else {
index = low;
}
// If index is equal to snapshot array length, then no update was made after the requested
// snapshot interval. This means the latest value is the right one.
if (index == snapshots.length) {
return AccountSnapshot(
_interval,
votingPower[_account]
);
} else {
return snapshots[index];
}
}
/**
* @notice Gets account state at interval
* @param _account - account to get state for
* @param _interval - interval to get state at
* @param _hint - off-chain computed index of interval
* @return state
*/
function accountSnapshotAt(address _account, uint256 _interval, uint256 _hint) external view returns (AccountSnapshot memory) {
require(_interval <= currentInterval(), "Staking: Interval out of bounds");
// Get account snapshots array
AccountSnapshot[] storage snapshots = accountSnapshots[_account];
// Check if hint is correct, else fall back to binary search
if (
_hint <= snapshots.length
&& (_hint == 0 || snapshots[_hint - 1].interval < _interval)
&& (_hint == snapshots.length || snapshots[_hint].interval >= _interval)
) {
// The hint is correct
if (_hint < snapshots.length)
return snapshots[_hint];
else
return AccountSnapshot(_interval, votingPower[_account]);
} else return accountSnapshotAtSearch(_account, _interval);
}
/**
* @notice Stake tokens
* @dev This contract should be approve()'d for _amount
* @param _amount - Amount to stake
* @return stake ID
*/
function stake(uint256 _amount) public returns (uint256) {
// Check if amount is not 0
require(_amount > 0, "Staking: Amount not set");
// Check if snapshot needs to be taken
snapshot(msg.sender);
// Get stakeID
uint256 stakeID = stakes[msg.sender].length;
// Set stake values
stakes[msg.sender].push(StakeStruct(
msg.sender,
_amount,
block.timestamp,
0,
0
));
// Increment global staked
totalStaked += _amount;
// Add voting power
moveVotingPower(
address(0),
msg.sender,
_amount
);
// Transfer tokens
stakingToken.safeTransferFrom(msg.sender, address(this), _amount);
// Emit event
emit Stake(msg.sender, stakeID, _amount);
return stakeID;
}
/**
* @notice Unlock stake tokens
* @param _stakeID - Stake to unlock
*/
function unlock(uint256 _stakeID) public {
require(
stakes[msg.sender][_stakeID].staketime != 0,
"Staking: Stake doesn't exist"
);
require(
stakes[msg.sender][_stakeID].locktime == 0,
"Staking: Stake already unlocked"
);
// Check if snapshot needs to be taken
snapshot(msg.sender);
// Set stake locktime
stakes[msg.sender][_stakeID].locktime = block.timestamp + STAKE_LOCKTIME;
// Remove voting power
moveVotingPower(
stakes[msg.sender][_stakeID].delegate,
address(0),
stakes[msg.sender][_stakeID].amount
);
// Emit event
emit Unlock(msg.sender, _stakeID);
}
/**
* @notice Claim stake token
* @param _stakeID - Stake to claim
*/
function claim(uint256 _stakeID) public {
require(
stakes[msg.sender][_stakeID].locktime != 0
&& stakes[msg.sender][_stakeID].locktime < block.timestamp,
"Staking: Stake not unlocked"
);
require(
stakes[msg.sender][_stakeID].claimedTime == 0,
"Staking: Stake already claimed"
);
// Check if snapshot needs to be taken
snapshot(msg.sender);
// Set stake claimed time
stakes[msg.sender][_stakeID].claimedTime = block.timestamp;
// Decrement global staked
totalStaked -= stakes[msg.sender][_stakeID].amount;
// Transfer tokens
stakingToken.safeTransfer(
msg.sender,
stakes[msg.sender][_stakeID].amount
);
// Emit event
emit Claim(msg.sender, _stakeID);
}
}// SPDX-License-Identifier: UNLICENSED
pragma solidity ^0.8.0;
pragma abicoder v2;
// OpenZeppelin v4
import { Ownable } from "@openzeppelin/contracts/access/Ownable.sol";
/**
* @title Delegator
* @author Railgun Contributors
* @notice 'Owner' contract for all railgun contracts
* delegates permissions to other contracts (voter, role)
*/
contract Delegator is Ownable {
/*
Mapping structure is calling address => contract => function signature
0 is used as a wildcard, so permission for contract 0 is permission for
any contract, and permission for function signature 0 is permission for
any function.
Comments below use * to signify wildcard and . notation to seperate address/contract/function.
caller.*.* allows caller to call any function on any contract
caller.X.* allows caller to call any function on contract X
caller.*.Y allows caller to call function Y on any contract
*/
mapping(
address => mapping(
address => mapping(bytes4 => bool)
)
) public permissions;
event GrantPermission(address indexed caller, address indexed contractAddress, bytes4 indexed selector);
event RevokePermission(address indexed caller, address indexed contractAddress, bytes4 indexed selector);
/**
* @notice Sets initial admin
*/
constructor(address _admin) {
Ownable.transferOwnership(_admin);
}
/**
* @notice Sets permission bit
* @dev See comment on permissions mapping for wildcard format
* @param _caller - caller to set permissions for
* @param _contract - contract to set permissions for
* @param _selector - selector to set permissions for
* @param _permission - permission bit to set
*/
function setPermission(
address _caller,
address _contract,
bytes4 _selector,
bool _permission
) public onlyOwner {
// If permission set is different to new permission then we execute, otherwise skip
if (permissions[_caller][_contract][_selector] != _permission) {
// Set permission bit
permissions[_caller][_contract][_selector] = _permission;
// Emit event
if (_permission) {
emit GrantPermission(_caller, _contract, _selector);
} else {
emit RevokePermission(_caller, _contract, _selector);
}
}
}
/**
* @notice Checks if caller has permission to execute function
* @param _caller - caller to check permissions for
* @param _contract - contract to check
* @param _selector - function signature to check
* @return if caller has permission
*/
function checkPermission(address _caller, address _contract, bytes4 _selector) public view returns (bool) {
/*
See comment on permissions mapping for structure
Comments below use * to signify wildcard and . notation to seperate contract/function
*/
return (
_caller == Ownable.owner()
|| permissions[_caller][_contract][_selector] // Owner always has global permissions
|| permissions[_caller][_contract][0x0] // Permission for function is given
|| permissions[_caller][address(0)][_selector] // Permission for _contract.* is given
|| permissions[_caller][address(0)][0x0] // Global permission is given
);
}
/**
* @notice Calls function
* @dev calls to functions on this contract are intercepted and run directly
* this is so the voting contract doesn't need to have special cases for calling
* functions other than this one.
* @param _contract - contract to call
* @param _data - calldata to pass to contract
* @return success - whether call succeeded
* @return returnData - return data from function call
*/
function callContract(address _contract, bytes calldata _data, uint256 _value) public returns (bool success, bytes memory returnData) {
// Get selector
bytes4 selector = bytes4(_data);
// Intercept calls to this contract
if (_contract == address(this)) {
if (selector == this.setPermission.selector) {
// Decode call data
(
address caller,
address calledContract,
bytes4 _permissionSelector,
bool permission
) = abi.decode(abi.encodePacked(_data[4:]), (address, address, bytes4, bool));
// Call setPermission
setPermission(caller, calledContract, _permissionSelector, permission);
// Return success with empty returndata bytes
bytes memory empty;
return (true, empty);
} else if (selector == this.transferOwnership.selector) {
// Decode call data
(
address newOwner
) = abi.decode(abi.encodePacked(_data[4:]), (address));
// Call transferOwnership
Ownable.transferOwnership(newOwner);
// Return success with empty returndata bytes
bytes memory empty;
return (true, empty);
} else if (selector == this.renounceOwnership.selector) {
// Call renounceOwnership
Ownable.renounceOwnership();
// Return success with empty returndata bytes
bytes memory empty;
return (true, empty);
} else {
// Return failed with empty returndata bytes
bytes memory empty;
return (false, empty);
}
}
// Check permissions
require(checkPermission(msg.sender, _contract, selector), "Delegator: Caller doesn't have permission");
// Call external contract and return
// solhint-disable-next-line avoid-low-level-calls
return _contract.call{value: _value}(_data);
}
}// SPDX-License-Identifier: MIT
pragma solidity ^0.8.0;
/**
* @dev Standard math utilities missing in the Solidity language.
*/
library Math {
/**
* @dev Returns the largest of two numbers.
*/
function max(uint256 a, uint256 b) internal pure returns (uint256) {
return a >= b ? a : b;
}
/**
* @dev Returns the smallest of two numbers.
*/
function min(uint256 a, uint256 b) internal pure returns (uint256) {
return a < b ? a : b;
}
/**
* @dev Returns the average of two numbers. The result is rounded towards
* zero.
*/
function average(uint256 a, uint256 b) internal pure returns (uint256) {
// (a + b) / 2 can overflow, so we distribute
return (a / 2) + (b / 2) + ((a % 2 + b % 2) / 2);
}
}// SPDX-License-Identifier: MIT
pragma solidity ^0.8.0;
/**
* @dev Interface of the ERC20 standard as defined in the EIP.
*/
interface IERC20 {
/**
* @dev Returns the amount of tokens in existence.
*/
function totalSupply() external view returns (uint256);
/**
* @dev Returns the amount of tokens owned by `account`.
*/
function balanceOf(address account) external view returns (uint256);
/**
* @dev Moves `amount` tokens from the caller's account to `recipient`.
*
* Returns a boolean value indicating whether the operation succeeded.
*
* Emits a {Transfer} event.
*/
function transfer(address recipient, uint256 amount) external returns (bool);
/**
* @dev Returns the remaining number of tokens that `spender` will be
* allowed to spend on behalf of `owner` through {transferFrom}. This is
* zero by default.
*
* This value changes when {approve} or {transferFrom} are called.
*/
function allowance(address owner, address spender) external view returns (uint256);
/**
* @dev Sets `amount` as the allowance of `spender` over the caller's tokens.
*
* Returns a boolean value indicating whether the operation succeeded.
*
* IMPORTANT: Beware that changing an allowance with this method brings the risk
* that someone may use both the old and the new allowance by unfortunate
* transaction ordering. One possible solution to mitigate this race
* condition is to first reduce the spender's allowance to 0 and set the
* desired value afterwards:
* https://github.com/ethereum/EIPs/issues/20#issuecomment-263524729
*
* Emits an {Approval} event.
*/
function approve(address spender, uint256 amount) external returns (bool);
/**
* @dev Moves `amount` tokens from `sender` to `recipient` using the
* allowance mechanism. `amount` is then deducted from the caller's
* allowance.
*
* Returns a boolean value indicating whether the operation succeeded.
*
* Emits a {Transfer} event.
*/
function transferFrom(address sender, address recipient, uint256 amount) external returns (bool);
/**
* @dev Emitted when `value` tokens are moved from one account (`from`) to
* another (`to`).
*
* Note that `value` may be zero.
*/
event Transfer(address indexed from, address indexed to, uint256 value);
/**
* @dev Emitted when the allowance of a `spender` for an `owner` is set by
* a call to {approve}. `value` is the new allowance.
*/
event Approval(address indexed owner, address indexed spender, uint256 value);
}// SPDX-License-Identifier: MIT
pragma solidity ^0.8.0;
import "../IERC20.sol";
import "../../../utils/Address.sol";
/**
* @title SafeERC20
* @dev Wrappers around ERC20 operations that throw on failure (when the token
* contract returns false). Tokens that return no value (and instead revert or
* throw on failure) are also supported, non-reverting calls are assumed to be
* successful.
* To use this library you can add a `using SafeERC20 for IERC20;` statement to your contract,
* which allows you to call the safe operations as `token.safeTransfer(...)`, etc.
*/
library SafeERC20 {
using Address for address;
function safeTransfer(IERC20 token, address to, uint256 value) internal {
_callOptionalReturn(token, abi.encodeWithSelector(token.transfer.selector, to, value));
}
function safeTransferFrom(IERC20 token, address from, address to, uint256 value) internal {
_callOptionalReturn(token, abi.encodeWithSelector(token.transferFrom.selector, from, to, value));
}
/**
* @dev Deprecated. This function has issues similar to the ones found in
* {IERC20-approve}, and its usage is discouraged.
*
* Whenever possible, use {safeIncreaseAllowance} and
* {safeDecreaseAllowance} instead.
*/
function safeApprove(IERC20 token, address spender, uint256 value) internal {
// safeApprove should only be called when setting an initial allowance,
// or when resetting it to zero. To increase and decrease it, use
// 'safeIncreaseAllowance' and 'safeDecreaseAllowance'
// solhint-disable-next-line max-line-length
require((value == 0) || (token.allowance(address(this), spender) == 0),
"SafeERC20: approve from non-zero to non-zero allowance"
);
_callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, value));
}
function safeIncreaseAllowance(IERC20 token, address spender, uint256 value) internal {
uint256 newAllowance = token.allowance(address(this), spender) + value;
_callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, newAllowance));
}
function safeDecreaseAllowance(IERC20 token, address spender, uint256 value) internal {
unchecked {
uint256 oldAllowance = token.allowance(address(this), spender);
require(oldAllowance >= value, "SafeERC20: decreased allowance below zero");
uint256 newAllowance = oldAllowance - value;
_callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, newAllowance));
}
}
/**
* @dev Imitates a Solidity high-level call (i.e. a regular function call to a contract), relaxing the requirement
* on the return value: the return value is optional (but if data is returned, it must not be false).
* @param token The token targeted by the call.
* @param data The call data (encoded using abi.encode or one of its variants).
*/
function _callOptionalReturn(IERC20 token, bytes memory data) private {
// We need to perform a low level call here, to bypass Solidity's return data size checking mechanism, since
// we're implementing it ourselves. We use {Address.functionCall} to perform this call, which verifies that
// the target address contains contract code and also asserts for success in the low-level call.
bytes memory returndata = address(token).functionCall(data, "SafeERC20: low-level call failed");
if (returndata.length > 0) { // Return data is optional
// solhint-disable-next-line max-line-length
require(abi.decode(returndata, (bool)), "SafeERC20: ERC20 operation did not succeed");
}
}
}// SPDX-License-Identifier: MIT
pragma solidity ^0.8.0;
/**
* @dev Collection of functions related to the address type
*/
library Address {
/**
* @dev Returns true if `account` is a contract.
*
* [IMPORTANT]
* ====
* It is unsafe to assume that an address for which this function returns
* false is an externally-owned account (EOA) and not a contract.
*
* Among others, `isContract` will return false for the following
* types of addresses:
*
* - an externally-owned account
* - a contract in construction
* - an address where a contract will be created
* - an address where a contract lived, but was destroyed
* ====
*/
function isContract(address account) internal view returns (bool) {
// This method relies on extcodesize, which returns 0 for contracts in
// construction, since the code is only stored at the end of the
// constructor execution.
uint256 size;
// solhint-disable-next-line no-inline-assembly
assembly { size := extcodesize(account) }
return size > 0;
}
/**
* @dev Replacement for Solidity's `transfer`: sends `amount` wei to
* `recipient`, forwarding all available gas and reverting on errors.
*
* https://eips.ethereum.org/EIPS/eip-1884[EIP1884] increases the gas cost
* of certain opcodes, possibly making contracts go over the 2300 gas limit
* imposed by `transfer`, making them unable to receive funds via
* `transfer`. {sendValue} removes this limitation.
*
* https://diligence.consensys.net/posts/2019/09/stop-using-soliditys-transfer-now/[Learn more].
*
* IMPORTANT: because control is transferred to `recipient`, care must be
* taken to not create reentrancy vulnerabilities. Consider using
* {ReentrancyGuard} or the
* https://solidity.readthedocs.io/en/v0.5.11/security-considerations.html#use-the-checks-effects-interactions-pattern[checks-effects-interactions pattern].
*/
function sendValue(address payable recipient, uint256 amount) internal {
require(address(this).balance >= amount, "Address: insufficient balance");
// solhint-disable-next-line avoid-low-level-calls, avoid-call-value
(bool success, ) = recipient.call{ value: amount }("");
require(success, "Address: unable to send value, recipient may have reverted");
}
/**
* @dev Performs a Solidity function call using a low level `call`. A
* plain`call` is an unsafe replacement for a function call: use this
* function instead.
*
* If `target` reverts with a revert reason, it is bubbled up by this
* function (like regular Solidity function calls).
*
* Returns the raw returned data. To convert to the expected return value,
* use https://solidity.readthedocs.io/en/latest/units-and-global-variables.html?highlight=abi.decode#abi-encoding-and-decoding-functions[`abi.decode`].
*
* Requirements:
*
* - `target` must be a contract.
* - calling `target` with `data` must not revert.
*
* _Available since v3.1._
*/
function functionCall(address target, bytes memory data) internal returns (bytes memory) {
return functionCall(target, data, "Address: low-level call failed");
}
/**
* @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], but with
* `errorMessage` as a fallback revert reason when `target` reverts.
*
* _Available since v3.1._
*/
function functionCall(address target, bytes memory data, string memory errorMessage) internal returns (bytes memory) {
return functionCallWithValue(target, data, 0, errorMessage);
}
/**
* @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
* but also transferring `value` wei to `target`.
*
* Requirements:
*
* - the calling contract must have an ETH balance of at least `value`.
* - the called Solidity function must be `payable`.
*
* _Available since v3.1._
*/
function functionCallWithValue(address target, bytes memory data, uint256 value) internal returns (bytes memory) {
return functionCallWithValue(target, data, value, "Address: low-level call with value failed");
}
/**
* @dev Same as {xref-Address-functionCallWithValue-address-bytes-uint256-}[`functionCallWithValue`], but
* with `errorMessage` as a fallback revert reason when `target` reverts.
*
* _Available since v3.1._
*/
function functionCallWithValue(address target, bytes memory data, uint256 value, string memory errorMessage) internal returns (bytes memory) {
require(address(this).balance >= value, "Address: insufficient balance for call");
require(isContract(target), "Address: call to non-contract");
// solhint-disable-next-line avoid-low-level-calls
(bool success, bytes memory returndata) = target.call{ value: value }(data);
return _verifyCallResult(success, returndata, errorMessage);
}
/**
* @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
* but performing a static call.
*
* _Available since v3.3._
*/
function functionStaticCall(address target, bytes memory data) internal view returns (bytes memory) {
return functionStaticCall(target, data, "Address: low-level static call failed");
}
/**
* @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`],
* but performing a static call.
*
* _Available since v3.3._
*/
function functionStaticCall(address target, bytes memory data, string memory errorMessage) internal view returns (bytes memory) {
require(isContract(target), "Address: static call to non-contract");
// solhint-disable-next-line avoid-low-level-calls
(bool success, bytes memory returndata) = target.staticcall(data);
return _verifyCallResult(success, returndata, errorMessage);
}
/**
* @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
* but performing a delegate call.
*
* _Available since v3.4._
*/
function functionDelegateCall(address target, bytes memory data) internal returns (bytes memory) {
return functionDelegateCall(target, data, "Address: low-level delegate call failed");
}
/**
* @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`],
* but performing a delegate call.
*
* _Available since v3.4._
*/
function functionDelegateCall(address target, bytes memory data, string memory errorMessage) internal returns (bytes memory) {
require(isContract(target), "Address: delegate call to non-contract");
// solhint-disable-next-line avoid-low-level-calls
(bool success, bytes memory returndata) = target.delegatecall(data);
return _verifyCallResult(success, returndata, errorMessage);
}
function _verifyCallResult(bool success, bytes memory returndata, string memory errorMessage) private pure returns(bytes memory) {
if (success) {
return returndata;
} else {
// Look for revert reason and bubble it up if present
if (returndata.length > 0) {
// The easiest way to bubble the revert reason is using memory via assembly
// solhint-disable-next-line no-inline-assembly
assembly {
let returndata_size := mload(returndata)
revert(add(32, returndata), returndata_size)
}
} else {
revert(errorMessage);
}
}
}
}// SPDX-License-Identifier: MIT
pragma solidity ^0.8.0;
import "../utils/Context.sol";
/**
* @dev Contract module which provides a basic access control mechanism, where
* there is an account (an owner) that can be granted exclusive access to
* specific functions.
*
* By default, the owner account will be the one that deploys the contract. This
* can later be changed with {transferOwnership}.
*
* This module is used through inheritance. It will make available the modifier
* `onlyOwner`, which can be applied to your functions to restrict their use to
* the owner.
*/
abstract contract Ownable is Context {
address private _owner;
event OwnershipTransferred(address indexed previousOwner, address indexed newOwner);
/**
* @dev Initializes the contract setting the deployer as the initial owner.
*/
constructor () {
address msgSender = _msgSender();
_owner = msgSender;
emit OwnershipTransferred(address(0), msgSender);
}
/**
* @dev Returns the address of the current owner.
*/
function owner() public view virtual returns (address) {
return _owner;
}
/**
* @dev Throws if called by any account other than the owner.
*/
modifier onlyOwner() {
require(owner() == _msgSender(), "Ownable: caller is not the owner");
_;
}
/**
* @dev Leaves the contract without owner. It will not be possible to call
* `onlyOwner` functions anymore. Can only be called by the current owner.
*
* NOTE: Renouncing ownership will leave the contract without an owner,
* thereby removing any functionality that is only available to the owner.
*/
function renounceOwnership() public virtual onlyOwner {
emit OwnershipTransferred(_owner, address(0));
_owner = address(0);
}
/**
* @dev Transfers ownership of the contract to a new account (`newOwner`).
* Can only be called by the current owner.
*/
function transferOwnership(address newOwner) public virtual onlyOwner {
require(newOwner != address(0), "Ownable: new owner is the zero address");
emit OwnershipTransferred(_owner, newOwner);
_owner = newOwner;
}
}// SPDX-License-Identifier: MIT
pragma solidity ^0.8.0;
/*
* @dev Provides information about the current execution context, including the
* sender of the transaction and its data. While these are generally available
* via msg.sender and msg.data, they should not be accessed in such a direct
* manner, since when dealing with meta-transactions the account sending and
* paying for execution may not be the actual sender (as far as an application
* is concerned).
*
* This contract is only required for intermediate, library-like contracts.
*/
abstract contract Context {
function _msgSender() internal view virtual returns (address) {
return msg.sender;
}
function _msgData() internal view virtual returns (bytes calldata) {
this; // silence state mutability warning without generating bytecode - see https://github.com/ethereum/solidity/issues/2691
return msg.data;
}
}{
"optimizer": {
"enabled": true,
"runs": 1600
},
"outputSelection": {
"*": {
"*": [
"evm.bytecode",
"evm.deployedBytecode",
"abi"
]
}
},
"libraries": {}
}Contract Security Audit
- No Contract Security Audit Submitted- Submit Audit Here
Contract ABI
API[{"inputs":[{"internalType":"contract Staking","name":"_stakingContract","type":"address"},{"internalType":"contract Delegator","name":"_delegator","type":"address"}],"stateMutability":"nonpayable","type":"constructor"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"uint256","name":"id","type":"uint256"}],"name":"Execution","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"uint256","name":"id","type":"uint256"},{"indexed":true,"internalType":"address","name":"proposer","type":"address"}],"name":"Proposal","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"uint256","name":"id","type":"uint256"},{"indexed":true,"internalType":"address","name":"sponsor","type":"address"},{"indexed":false,"internalType":"uint256","name":"amount","type":"uint256"}],"name":"Sponsorship","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"uint256","name":"id","type":"uint256"},{"indexed":true,"internalType":"address","name":"sponsor","type":"address"},{"indexed":false,"internalType":"uint256","name":"amount","type":"uint256"}],"name":"SponsorshipRevocation","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"uint256","name":"id","type":"uint256"}],"name":"VoteCall","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"uint256","name":"id","type":"uint256"},{"indexed":true,"internalType":"address","name":"voter","type":"address"},{"indexed":false,"internalType":"bool","name":"affirmative","type":"bool"},{"indexed":false,"internalType":"uint256","name":"votes","type":"uint256"}],"name":"VoteCast","type":"event"},{"inputs":[],"name":"DELEGATOR_CONTRACT","outputs":[{"internalType":"contract Delegator","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"EXECUTION_END_OFFSET","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"EXECUTION_START_OFFSET","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"PROPOSAL_SPONSOR_THRESHOLD","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"QUORUM","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"SPONSOR_WINDOW","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"STAKING_CONTRACT","outputs":[{"internalType":"contract Staking","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"VOTING_NAY_END_OFFSET","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"VOTING_START_OFFSET","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"VOTING_YAY_END_OFFSET","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"_id","type":"uint256"}],"name":"callVote","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"string","name":"_proposalDocument","type":"string"},{"components":[{"internalType":"address","name":"callContract","type":"address"},{"internalType":"bytes","name":"data","type":"bytes"},{"internalType":"uint256","name":"value","type":"uint256"}],"internalType":"struct Voting.Call[]","name":"_actions","type":"tuple[]"}],"name":"createProposal","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_id","type":"uint256"}],"name":"executeProposal","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_id","type":"uint256"}],"name":"getActions","outputs":[{"components":[{"internalType":"address","name":"callContract","type":"address"},{"internalType":"bytes","name":"data","type":"bytes"},{"internalType":"uint256","name":"value","type":"uint256"}],"internalType":"struct Voting.Call[]","name":"","type":"tuple[]"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"_id","type":"uint256"},{"internalType":"address","name":"_account","type":"address"}],"name":"getSponsored","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"}],"name":"proposals","outputs":[{"internalType":"address","name":"proposer","type":"address"},{"internalType":"string","name":"proposalDocument","type":"string"},{"internalType":"uint256","name":"publishTime","type":"uint256"},{"internalType":"uint256","name":"voteCallTime","type":"uint256"},{"internalType":"uint256","name":"sponsorship","type":"uint256"},{"internalType":"bool","name":"executed","type":"bool"},{"internalType":"uint256","name":"yayVotes","type":"uint256"},{"internalType":"uint256","name":"nayVotes","type":"uint256"},{"internalType":"uint256","name":"sponsorInterval","type":"uint256"},{"internalType":"uint256","name":"votingInterval","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"proposalsLength","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"_id","type":"uint256"},{"internalType":"uint256","name":"_amount","type":"uint256"},{"internalType":"uint256","name":"_hint","type":"uint256"}],"name":"sponsorProposal","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_id","type":"uint256"},{"internalType":"uint256","name":"_amount","type":"uint256"}],"name":"unsponsorProposal","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_id","type":"uint256"},{"internalType":"uint256","name":"_amount","type":"uint256"},{"internalType":"bool","name":"_affirmative","type":"bool"},{"internalType":"uint256","name":"_hint","type":"uint256"}],"name":"vote","outputs":[],"stateMutability":"nonpayable","type":"function"}]Contract Creation Code
60c06040523480156200001157600080fd5b506040516200214938038062002149833981016040819052620000349162000053565b6001600160601b0319606092831b8116608052911b1660a052620000ab565b600080604083850312156200006757600080fd5b8251620000748162000092565b6020840151909250620000878162000092565b809150509250929050565b6001600160a01b0381168114620000a857600080fd5b50565b60805160601c60a05160601c61204f620000fa60003960008181610253015261077f0152600081816102ca01528181610bd601528181610d7e015281816112310152611544015261204f6000f3fe608060405234801561001057600080fd5b506004361061016c5760003560e01c806351ec8bd7116100cd578063a1d2a6a311610081578063de3479df11610066578063de3479df146102ec578063e1f811ac146102ff578063ef76ccda1461031257600080fd5b8063a1d2a6a3146102b2578063da19ddfb146102c557600080fd5b80636d079932116100b25780636d0799321461028d57806389508d15146102975780639f31490d146102a857600080fd5b806351ec8bd71461023b57806359383bf51461024e57600080fd5b80632e80d9b61161012457806342f87abf1161010957806342f87abf1461021657806344c7c8671461022057806349fe1f6d1461022857600080fd5b80632e80d9b6146101e4578063328dd982146101f657600080fd5b80630d61b519116101555780630d61b519146101bb5780630fe0554d146101d05780631af74bc3146101da57600080fd5b8063013cf08b1461017157806305fe07fb146101a3575b600080fd5b61018461017f366004611bb9565b610325565b60405161019a9a99989796959493929190611d93565b60405180910390f35b6101ad6212750081565b60405190815260200161019a565b6101ce6101c9366004611bb9565b61041e565b005b6101ad62278d0081565b6101ad6206978081565b6101ad6a034f086f3b33b68400000081565b610209610204366004611bb9565b610915565b60405161019a9190611df8565b6101ad6207e90081565b6000546101ad565b6101ce610236366004611bb9565b610a4a565b6101ad610249366004611aa5565b610c99565b6102757f000000000000000000000000000000000000000000000000000000000000000081565b6040516001600160a01b03909116815260200161019a565b6101ad6202a30081565b6101ad69d3c21bcecceda100000081565b6101ad62093a8081565b6101ce6102c0366004611c39565b610fc7565b6102757f000000000000000000000000000000000000000000000000000000000000000081565b6101ad6102fa366004611beb565b6113fa565b6101ce61030d366004611c78565b61143f565b6101ce610320366004611c17565b6116ba565b6000818154811061033557600080fd5b60009182526020909120600d9091020180546001820180546001600160a01b0390921693509061036490611f70565b80601f016020809104026020016040519081016040528092919081815260200182805461039090611f70565b80156103dd5780601f106103b2576101008083540402835291602001916103dd565b820191906000526020600020905b8154815290600101906020018083116103c057829003601f168201915b50505060038401546004850154600586015460078701546009880154600a890154600b8a0154600c909a01549899959894975092955060ff9091169390928a565b600080828154811061043257610432611fdc565b90600052602060002090600d0201905060008160040154116104c15760405162461bcd60e51b815260206004820152603160248201527f566f74696e673a20566f7465206861736e2774206265656e2063616c6c65642060448201527f666f7220746869732070726f706f73616c00000000000000000000000000000060648201526084015b60405180910390fd5b6a034f086f3b33b68400000081600a015482600901546104e19190611f11565b10156105555760405162461bcd60e51b815260206004820152602260248201527f566f74696e673a2051756f72756d206861736e2774206265656e20726561636860448201527f656400000000000000000000000000000000000000000000000000000000000060648201526084016104b8565b80600a01548160090154116105d25760405162461bcd60e51b815260206004820152602360248201527f566f74696e673a2050726f706f73616c206861736e277420706173736564207660448201527f6f7465000000000000000000000000000000000000000000000000000000000060648201526084016104b8565b62093a8081600401546105e59190611f11565b42116106595760405162461bcd60e51b815260206004820152602660248201527f566f74696e673a20457865637574696f6e2077696e646f77206861736e27742060448201527f6f70656e6564000000000000000000000000000000000000000000000000000060648201526084016104b8565b62127500816004015461066c9190611f11565b42106106e05760405162461bcd60e51b815260206004820152602360248201527f566f74696e673a20457865637574696f6e2077696e646f772068617320636c6f60448201527f736564000000000000000000000000000000000000000000000000000000000060648201526084016104b8565b600781015460ff161561075b5760405162461bcd60e51b815260206004820152602a60248201527f566f74696e673a2050726f706f73616c2068617320616c72656164792062656560448201527f6e2065786563757465640000000000000000000000000000000000000000000060648201526084016104b8565b60078101805460ff191660011790556002810160005b81548110156108e4576000807f00000000000000000000000000000000000000000000000000000000000000006001600160a01b031663c6b295c18585815481106107be576107be611fdc565b600091825260209091206003909102015486546001600160a01b03909116908790879081106107ef576107ef611fdc565b906000526020600020906003020160010187878154811061081257610812611fdc565b9060005260206000209060030201600201546040518463ffffffff1660e01b815260040161084293929190611cd0565b600060405180830381600087803b15801561085c57600080fd5b505af1158015610870573d6000803e3d6000fd5b505050506040513d6000823e601f3d908101601f1916820160405261089891908101906119e2565b91509150816108cf57600083826040516020016108b6929190611e82565b6040516020818303038152906040529050805181602001fd5b505080806108dc90611fab565b915050610771565b5060405183907f33e13ecb54c3076d8e8bb8c2881800a4d972b792045ffae98fdf46df365fed7590600090a2505050565b60606000828154811061092a5761092a611fdc565b90600052602060002090600d0201600201805480602002602001604051908101604052809291908181526020016000905b82821015610a3f57600084815260209081902060408051606081019091526003850290910180546001600160a01b0316825260018101805492939192918401916109a490611f70565b80601f01602080910402602001604051908101604052809291908181526020018280546109d090611f70565b8015610a1d5780601f106109f257610100808354040283529160200191610a1d565b820191906000526020600020905b815481529060010190602001808311610a0057829003601f168201915b505050505081526020016002820154815250508152602001906001019061095b565b505050509050919050565b6000808281548110610a5e57610a5e611fdc565b90600052602060002090600d0201905062278d008160030154610a819190611f11565b4210610acf5760405162461bcd60e51b815260206004820181905260248201527f566f74696e673a2053706f6e736f72696e672077696e646f772070617373656460448201526064016104b8565b600481015415610b475760405162461bcd60e51b815260206004820152602560248201527f566f74696e673a2050726f706f73616c20616c726561647920676f6e6520746f60448201527f20766f746500000000000000000000000000000000000000000000000000000060648201526084016104b8565b69d3c21bcecceda100000081600501541015610bcb5760405162461bcd60e51b815260206004820152602160248201527f566f74696e673a2053706f6e736f72207468726573686f6c64206e6f74206d6560448201527f740000000000000000000000000000000000000000000000000000000000000060648201526084016104b8565b4281600401819055507f00000000000000000000000000000000000000000000000000000000000000006001600160a01b031663363487bc6040518163ffffffff1660e01b815260040160206040518083038186803b158015610c2d57600080fd5b505afa158015610c41573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610c659190611bd2565b600c82015560405182907f4bebca4d6e9291871f8e494321fae9eba02c07d0844ff387b8efd20c7a41de3290600090a25050565b600081610ce85760405162461bcd60e51b815260206004820152601c60248201527f566f74696e673a204e6f20616374696f6e73207370656369666965640000000060448201526064016104b8565b60008054600181018255908052600d81027f290decd9548b62a8d60345a988386fc84ba6bc95484008f6362f93160ef3e563810180547fffffffffffffffffffffffff0000000000000000000000000000000000000000163317815590610d72907f290decd9548b62a8d60345a988386fc84ba6bc95484008f6362f93160ef3e564018888611897565b504281600301819055507f00000000000000000000000000000000000000000000000000000000000000006001600160a01b031663363487bc6040518163ffffffff1660e01b815260040160206040518083038186803b158015610dd557600080fd5b505afa158015610de9573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610e0d9190611bd2565b600b82015560005b84811015610f8f57816002016040518060600160405280888885818110610e3e57610e3e611fdc565b9050602002810190610e509190611ef1565b610e5e9060208101906119c0565b6001600160a01b03168152602001888885818110610e7e57610e7e611fdc565b9050602002810190610e909190611ef1565b610e9e906020810190611ea3565b8080601f016020809104026020016040519081016040528093929190818152602001838380828437600092019190915250505090825250602001888885818110610eea57610eea611fdc565b9050602002810190610efc9190611ef1565b6040013590528154600180820184556000938452602093849020835160039093020180547fffffffffffffffffffffffff0000000000000000000000000000000000000000166001600160a01b0390931692909217825582840151805193949293610f6f9392850192919091019061191b565b506040820151816002015550508080610f8790611fab565b915050610e15565b50604051339083907fe56c149b9a4f632c327b2bc8b691ebfd0d76354ca7b678faea1effbd9d9f223d90600090a35095945050505050565b6000808581548110610fdb57610fdb611fdc565b90600052602060002090600d0201905060008160040154116110655760405162461bcd60e51b815260206004820152603160248201527f566f74696e673a20566f7465206861736e2774206265656e2063616c6c65642060448201527f666f7220746869732070726f706f73616c00000000000000000000000000000060648201526084016104b8565b6202a30081600401546110789190611f11565b42116110ec5760405162461bcd60e51b815260206004820152602360248201527f566f74696e673a20566f74696e672077696e646f77206861736e2774206f706560448201527f6e6564000000000000000000000000000000000000000000000000000000000060648201526084016104b8565b821561117e576206978081600401546111059190611f11565b42106111795760405162461bcd60e51b815260206004820152602c60248201527f566f74696e673a2041666669726d617469766520766f74696e672077696e646f60448201527f772068617320636c6f736564000000000000000000000000000000000000000060648201526084016104b8565b611205565b6207e90081600401546111919190611f11565b42106112055760405162461bcd60e51b815260206004820152602960248201527f566f74696e673a204e6567617469766520766f74696e672077696e646f77206860448201527f617320636c6f736564000000000000000000000000000000000000000000000060648201526084016104b8565b600c81015460405163046a32f360e11b81523360048201526024810191909152604481018390526000907f00000000000000000000000000000000000000000000000000000000000000006001600160a01b0316906308d465e690606401604080518083038186803b15801561127a57600080fd5b505afa15801561128e573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906112b29190611b6a565b60208082015133600090815260088601909252604090912054919250906112da908790611f11565b111561134e5760405162461bcd60e51b815260206004820152603160248201527f566f74696e673a204e6f7420656e6f75676820766f74696e6720706f7765722060448201527f746f2063617374207468697320766f746500000000000000000000000000000060648201526084016104b8565b3360009081526008830160205260408120805487929061136f908490611f11565b90915550508315611399578482600901600082825461138e9190611f11565b909155506113b39050565b8482600a0160008282546113ad9190611f11565b90915550505b60408051851515815260208101879052339188917fcbdf6214089cba887ecbf35a0b6a734589959c9763342c756bb2a80ca2bc9f6e910160405180910390a3505050505050565b600080838154811061140e5761140e611fdc565b600091825260208083206001600160a01b03861684526006600d909302019190910190526040902054905092915050565b600080848154811061145357611453611fdc565b90600052602060002090600d0201905080600401546000146114b75760405162461bcd60e51b815260206004820152601460248201527f566f74696e673a20476f6e6520746f20766f746500000000000000000000000060448201526064016104b8565b62278d0081600301546114ca9190611f11565b42106115185760405162461bcd60e51b815260206004820181905260248201527f566f74696e673a2053706f6e736f72696e672077696e646f772070617373656460448201526064016104b8565b600b81015460405163046a32f360e11b81523360048201526024810191909152604481018390526000907f00000000000000000000000000000000000000000000000000000000000000006001600160a01b0316906308d465e690606401604080518083038186803b15801561158d57600080fd5b505afa1580156115a1573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906115c59190611b6a565b60208082015133600090815260068601909252604090912054919250906115ed908690611f11565b111561163b5760405162461bcd60e51b815260206004820152601f60248201527f566f74696e673a204e6f7420656e6f75676820766f74696e6720706f7765720060448201526064016104b8565b3360009081526006830160205260408120805486929061165c908490611f11565b92505081905550838260050160008282546116779190611f11565b9091555050604051848152339086907fdf1985be8066f51229b5a3b530a445682812a3c384a5414ed34fa80e892dce0a9060200160405180910390a35050505050565b60008083815481106116ce576116ce611fdc565b90600052602060002090600d0201905080600401546000146117325760405162461bcd60e51b815260206004820152601460248201527f566f74696e673a20476f6e6520746f20766f746500000000000000000000000060448201526064016104b8565b62278d0081600301546117459190611f11565b42106117935760405162461bcd60e51b815260206004820181905260248201527f566f74696e673a2053706f6e736f72696e672077696e646f772070617373656460448201526064016104b8565b33600090815260068201602052604090205482111561181a5760405162461bcd60e51b815260206004820152602560248201527f566f74696e673a20416d6f756e742067726561746572207468616e2073706f6e60448201527f736f72656400000000000000000000000000000000000000000000000000000060648201526084016104b8565b3360009081526006820160205260408120805484929061183b908490611f29565b92505081905550818160050160008282546118569190611f29565b9091555050604051828152339084907feabb9788e481eecaf208dddaa2d5c986c2177e4b1b00cc25792b87c9ed00e7b29060200160405180910390a3505050565b8280546118a390611f70565b90600052602060002090601f0160209004810192826118c5576000855561190b565b82601f106118de5782800160ff1982351617855561190b565b8280016001018555821561190b579182015b8281111561190b5782358255916020019190600101906118f0565b5061191792915061198f565b5090565b82805461192790611f70565b90600052602060002090601f016020900481019282611949576000855561190b565b82601f1061196257805160ff191683800117855561190b565b8280016001018555821561190b579182015b8281111561190b578251825591602001919060010190611974565b5b808211156119175760008155600101611990565b80356001600160a01b03811681146119bb57600080fd5b919050565b6000602082840312156119d257600080fd5b6119db826119a4565b9392505050565b600080604083850312156119f557600080fd5b8251611a0081612008565b602084015190925067ffffffffffffffff80821115611a1e57600080fd5b818501915085601f830112611a3257600080fd5b815181811115611a4457611a44611ff2565b604051601f8201601f19908116603f01168101908382118183101715611a6c57611a6c611ff2565b81604052828152886020848701011115611a8557600080fd5b611a96836020830160208801611f40565b80955050505050509250929050565b60008060008060408587031215611abb57600080fd5b843567ffffffffffffffff80821115611ad357600080fd5b818701915087601f830112611ae757600080fd5b813581811115611af657600080fd5b886020828501011115611b0857600080fd5b602092830196509450908601359080821115611b2357600080fd5b818701915087601f830112611b3757600080fd5b813581811115611b4657600080fd5b8860208260051b8501011115611b5b57600080fd5b95989497505060200194505050565b600060408284031215611b7c57600080fd5b6040516040810181811067ffffffffffffffff82111715611b9f57611b9f611ff2565b604052825181526020928301519281019290925250919050565b600060208284031215611bcb57600080fd5b5035919050565b600060208284031215611be457600080fd5b5051919050565b60008060408385031215611bfe57600080fd5b82359150611c0e602084016119a4565b90509250929050565b60008060408385031215611c2a57600080fd5b50508035926020909101359150565b60008060008060808587031215611c4f57600080fd5b84359350602085013592506040850135611c6881612008565b9396929550929360600135925050565b600080600060608486031215611c8d57600080fd5b505081359360208301359350604090920135919050565b60008151808452611cbc816020860160208601611f40565b601f01601f19169290920160200192915050565b6001600160a01b0384168152600060206060818401526000855481600182811c915080831680611d0157607f831692505b858310811415611d1f57634e487b7160e01b85526022600452602485fd5b6060880183905260808801818015611d3e5760018114611d4f57611d7a565b60ff19861682528782019650611d7a565b60008c81526020902060005b86811015611d7457815484820152908501908901611d5b565b83019750505b5050505050508092505050826040830152949350505050565b60006101406001600160a01b038d168352806020840152611db68184018d611ca4565b604084019b909b5250506060810197909752608087019590955292151560a086015260c085019190915260e08401526101008301526101209091015292915050565b60006020808301818452808551808352604092508286019150828160051b87010184880160005b83811015611e7457603f19898403018552815160606001600160a01b03825116855288820151818a870152611e5682870182611ca4565b92890151958901959095525094870194925090860190600101611e1f565b509098975050505050505050565b828152604060208201526000611e9b6040830184611ca4565b949350505050565b6000808335601e19843603018112611eba57600080fd5b83018035915067ffffffffffffffff821115611ed557600080fd5b602001915036819003821315611eea57600080fd5b9250929050565b60008235605e19833603018112611f0757600080fd5b9190910192915050565b60008219821115611f2457611f24611fc6565b500190565b600082821015611f3b57611f3b611fc6565b500390565b60005b83811015611f5b578181015183820152602001611f43565b83811115611f6a576000848401525b50505050565b600181811c90821680611f8457607f821691505b60208210811415611fa557634e487b7160e01b600052602260045260246000fd5b50919050565b6000600019821415611fbf57611fbf611fc6565b5060010190565b634e487b7160e01b600052601160045260246000fd5b634e487b7160e01b600052603260045260246000fd5b634e487b7160e01b600052604160045260246000fd5b801515811461201657600080fd5b5056fea2646970667358221220ab667865bf3a87c17ab6e6423d76bb7aab2783c2a3b9806ed925bdd5490dabff64736f6c63430008060033000000000000000000000000ee6a649aa3766bd117e12c161726b693a1b2ee20000000000000000000000000b6d513f6222ee92fff975e901bd792e2513fb53b
Deployed Bytecode
0x608060405234801561001057600080fd5b506004361061016c5760003560e01c806351ec8bd7116100cd578063a1d2a6a311610081578063de3479df11610066578063de3479df146102ec578063e1f811ac146102ff578063ef76ccda1461031257600080fd5b8063a1d2a6a3146102b2578063da19ddfb146102c557600080fd5b80636d079932116100b25780636d0799321461028d57806389508d15146102975780639f31490d146102a857600080fd5b806351ec8bd71461023b57806359383bf51461024e57600080fd5b80632e80d9b61161012457806342f87abf1161010957806342f87abf1461021657806344c7c8671461022057806349fe1f6d1461022857600080fd5b80632e80d9b6146101e4578063328dd982146101f657600080fd5b80630d61b519116101555780630d61b519146101bb5780630fe0554d146101d05780631af74bc3146101da57600080fd5b8063013cf08b1461017157806305fe07fb146101a3575b600080fd5b61018461017f366004611bb9565b610325565b60405161019a9a99989796959493929190611d93565b60405180910390f35b6101ad6212750081565b60405190815260200161019a565b6101ce6101c9366004611bb9565b61041e565b005b6101ad62278d0081565b6101ad6206978081565b6101ad6a034f086f3b33b68400000081565b610209610204366004611bb9565b610915565b60405161019a9190611df8565b6101ad6207e90081565b6000546101ad565b6101ce610236366004611bb9565b610a4a565b6101ad610249366004611aa5565b610c99565b6102757f000000000000000000000000b6d513f6222ee92fff975e901bd792e2513fb53b81565b6040516001600160a01b03909116815260200161019a565b6101ad6202a30081565b6101ad69d3c21bcecceda100000081565b6101ad62093a8081565b6101ce6102c0366004611c39565b610fc7565b6102757f000000000000000000000000ee6a649aa3766bd117e12c161726b693a1b2ee2081565b6101ad6102fa366004611beb565b6113fa565b6101ce61030d366004611c78565b61143f565b6101ce610320366004611c17565b6116ba565b6000818154811061033557600080fd5b60009182526020909120600d9091020180546001820180546001600160a01b0390921693509061036490611f70565b80601f016020809104026020016040519081016040528092919081815260200182805461039090611f70565b80156103dd5780601f106103b2576101008083540402835291602001916103dd565b820191906000526020600020905b8154815290600101906020018083116103c057829003601f168201915b50505060038401546004850154600586015460078701546009880154600a890154600b8a0154600c909a01549899959894975092955060ff9091169390928a565b600080828154811061043257610432611fdc565b90600052602060002090600d0201905060008160040154116104c15760405162461bcd60e51b815260206004820152603160248201527f566f74696e673a20566f7465206861736e2774206265656e2063616c6c65642060448201527f666f7220746869732070726f706f73616c00000000000000000000000000000060648201526084015b60405180910390fd5b6a034f086f3b33b68400000081600a015482600901546104e19190611f11565b10156105555760405162461bcd60e51b815260206004820152602260248201527f566f74696e673a2051756f72756d206861736e2774206265656e20726561636860448201527f656400000000000000000000000000000000000000000000000000000000000060648201526084016104b8565b80600a01548160090154116105d25760405162461bcd60e51b815260206004820152602360248201527f566f74696e673a2050726f706f73616c206861736e277420706173736564207660448201527f6f7465000000000000000000000000000000000000000000000000000000000060648201526084016104b8565b62093a8081600401546105e59190611f11565b42116106595760405162461bcd60e51b815260206004820152602660248201527f566f74696e673a20457865637574696f6e2077696e646f77206861736e27742060448201527f6f70656e6564000000000000000000000000000000000000000000000000000060648201526084016104b8565b62127500816004015461066c9190611f11565b42106106e05760405162461bcd60e51b815260206004820152602360248201527f566f74696e673a20457865637574696f6e2077696e646f772068617320636c6f60448201527f736564000000000000000000000000000000000000000000000000000000000060648201526084016104b8565b600781015460ff161561075b5760405162461bcd60e51b815260206004820152602a60248201527f566f74696e673a2050726f706f73616c2068617320616c72656164792062656560448201527f6e2065786563757465640000000000000000000000000000000000000000000060648201526084016104b8565b60078101805460ff191660011790556002810160005b81548110156108e4576000807f000000000000000000000000b6d513f6222ee92fff975e901bd792e2513fb53b6001600160a01b031663c6b295c18585815481106107be576107be611fdc565b600091825260209091206003909102015486546001600160a01b03909116908790879081106107ef576107ef611fdc565b906000526020600020906003020160010187878154811061081257610812611fdc565b9060005260206000209060030201600201546040518463ffffffff1660e01b815260040161084293929190611cd0565b600060405180830381600087803b15801561085c57600080fd5b505af1158015610870573d6000803e3d6000fd5b505050506040513d6000823e601f3d908101601f1916820160405261089891908101906119e2565b91509150816108cf57600083826040516020016108b6929190611e82565b6040516020818303038152906040529050805181602001fd5b505080806108dc90611fab565b915050610771565b5060405183907f33e13ecb54c3076d8e8bb8c2881800a4d972b792045ffae98fdf46df365fed7590600090a2505050565b60606000828154811061092a5761092a611fdc565b90600052602060002090600d0201600201805480602002602001604051908101604052809291908181526020016000905b82821015610a3f57600084815260209081902060408051606081019091526003850290910180546001600160a01b0316825260018101805492939192918401916109a490611f70565b80601f01602080910402602001604051908101604052809291908181526020018280546109d090611f70565b8015610a1d5780601f106109f257610100808354040283529160200191610a1d565b820191906000526020600020905b815481529060010190602001808311610a0057829003601f168201915b505050505081526020016002820154815250508152602001906001019061095b565b505050509050919050565b6000808281548110610a5e57610a5e611fdc565b90600052602060002090600d0201905062278d008160030154610a819190611f11565b4210610acf5760405162461bcd60e51b815260206004820181905260248201527f566f74696e673a2053706f6e736f72696e672077696e646f772070617373656460448201526064016104b8565b600481015415610b475760405162461bcd60e51b815260206004820152602560248201527f566f74696e673a2050726f706f73616c20616c726561647920676f6e6520746f60448201527f20766f746500000000000000000000000000000000000000000000000000000060648201526084016104b8565b69d3c21bcecceda100000081600501541015610bcb5760405162461bcd60e51b815260206004820152602160248201527f566f74696e673a2053706f6e736f72207468726573686f6c64206e6f74206d6560448201527f740000000000000000000000000000000000000000000000000000000000000060648201526084016104b8565b4281600401819055507f000000000000000000000000ee6a649aa3766bd117e12c161726b693a1b2ee206001600160a01b031663363487bc6040518163ffffffff1660e01b815260040160206040518083038186803b158015610c2d57600080fd5b505afa158015610c41573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610c659190611bd2565b600c82015560405182907f4bebca4d6e9291871f8e494321fae9eba02c07d0844ff387b8efd20c7a41de3290600090a25050565b600081610ce85760405162461bcd60e51b815260206004820152601c60248201527f566f74696e673a204e6f20616374696f6e73207370656369666965640000000060448201526064016104b8565b60008054600181018255908052600d81027f290decd9548b62a8d60345a988386fc84ba6bc95484008f6362f93160ef3e563810180547fffffffffffffffffffffffff0000000000000000000000000000000000000000163317815590610d72907f290decd9548b62a8d60345a988386fc84ba6bc95484008f6362f93160ef3e564018888611897565b504281600301819055507f000000000000000000000000ee6a649aa3766bd117e12c161726b693a1b2ee206001600160a01b031663363487bc6040518163ffffffff1660e01b815260040160206040518083038186803b158015610dd557600080fd5b505afa158015610de9573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610e0d9190611bd2565b600b82015560005b84811015610f8f57816002016040518060600160405280888885818110610e3e57610e3e611fdc565b9050602002810190610e509190611ef1565b610e5e9060208101906119c0565b6001600160a01b03168152602001888885818110610e7e57610e7e611fdc565b9050602002810190610e909190611ef1565b610e9e906020810190611ea3565b8080601f016020809104026020016040519081016040528093929190818152602001838380828437600092019190915250505090825250602001888885818110610eea57610eea611fdc565b9050602002810190610efc9190611ef1565b6040013590528154600180820184556000938452602093849020835160039093020180547fffffffffffffffffffffffff0000000000000000000000000000000000000000166001600160a01b0390931692909217825582840151805193949293610f6f9392850192919091019061191b565b506040820151816002015550508080610f8790611fab565b915050610e15565b50604051339083907fe56c149b9a4f632c327b2bc8b691ebfd0d76354ca7b678faea1effbd9d9f223d90600090a35095945050505050565b6000808581548110610fdb57610fdb611fdc565b90600052602060002090600d0201905060008160040154116110655760405162461bcd60e51b815260206004820152603160248201527f566f74696e673a20566f7465206861736e2774206265656e2063616c6c65642060448201527f666f7220746869732070726f706f73616c00000000000000000000000000000060648201526084016104b8565b6202a30081600401546110789190611f11565b42116110ec5760405162461bcd60e51b815260206004820152602360248201527f566f74696e673a20566f74696e672077696e646f77206861736e2774206f706560448201527f6e6564000000000000000000000000000000000000000000000000000000000060648201526084016104b8565b821561117e576206978081600401546111059190611f11565b42106111795760405162461bcd60e51b815260206004820152602c60248201527f566f74696e673a2041666669726d617469766520766f74696e672077696e646f60448201527f772068617320636c6f736564000000000000000000000000000000000000000060648201526084016104b8565b611205565b6207e90081600401546111919190611f11565b42106112055760405162461bcd60e51b815260206004820152602960248201527f566f74696e673a204e6567617469766520766f74696e672077696e646f77206860448201527f617320636c6f736564000000000000000000000000000000000000000000000060648201526084016104b8565b600c81015460405163046a32f360e11b81523360048201526024810191909152604481018390526000907f000000000000000000000000ee6a649aa3766bd117e12c161726b693a1b2ee206001600160a01b0316906308d465e690606401604080518083038186803b15801561127a57600080fd5b505afa15801561128e573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906112b29190611b6a565b60208082015133600090815260088601909252604090912054919250906112da908790611f11565b111561134e5760405162461bcd60e51b815260206004820152603160248201527f566f74696e673a204e6f7420656e6f75676820766f74696e6720706f7765722060448201527f746f2063617374207468697320766f746500000000000000000000000000000060648201526084016104b8565b3360009081526008830160205260408120805487929061136f908490611f11565b90915550508315611399578482600901600082825461138e9190611f11565b909155506113b39050565b8482600a0160008282546113ad9190611f11565b90915550505b60408051851515815260208101879052339188917fcbdf6214089cba887ecbf35a0b6a734589959c9763342c756bb2a80ca2bc9f6e910160405180910390a3505050505050565b600080838154811061140e5761140e611fdc565b600091825260208083206001600160a01b03861684526006600d909302019190910190526040902054905092915050565b600080848154811061145357611453611fdc565b90600052602060002090600d0201905080600401546000146114b75760405162461bcd60e51b815260206004820152601460248201527f566f74696e673a20476f6e6520746f20766f746500000000000000000000000060448201526064016104b8565b62278d0081600301546114ca9190611f11565b42106115185760405162461bcd60e51b815260206004820181905260248201527f566f74696e673a2053706f6e736f72696e672077696e646f772070617373656460448201526064016104b8565b600b81015460405163046a32f360e11b81523360048201526024810191909152604481018390526000907f000000000000000000000000ee6a649aa3766bd117e12c161726b693a1b2ee206001600160a01b0316906308d465e690606401604080518083038186803b15801561158d57600080fd5b505afa1580156115a1573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906115c59190611b6a565b60208082015133600090815260068601909252604090912054919250906115ed908690611f11565b111561163b5760405162461bcd60e51b815260206004820152601f60248201527f566f74696e673a204e6f7420656e6f75676820766f74696e6720706f7765720060448201526064016104b8565b3360009081526006830160205260408120805486929061165c908490611f11565b92505081905550838260050160008282546116779190611f11565b9091555050604051848152339086907fdf1985be8066f51229b5a3b530a445682812a3c384a5414ed34fa80e892dce0a9060200160405180910390a35050505050565b60008083815481106116ce576116ce611fdc565b90600052602060002090600d0201905080600401546000146117325760405162461bcd60e51b815260206004820152601460248201527f566f74696e673a20476f6e6520746f20766f746500000000000000000000000060448201526064016104b8565b62278d0081600301546117459190611f11565b42106117935760405162461bcd60e51b815260206004820181905260248201527f566f74696e673a2053706f6e736f72696e672077696e646f772070617373656460448201526064016104b8565b33600090815260068201602052604090205482111561181a5760405162461bcd60e51b815260206004820152602560248201527f566f74696e673a20416d6f756e742067726561746572207468616e2073706f6e60448201527f736f72656400000000000000000000000000000000000000000000000000000060648201526084016104b8565b3360009081526006820160205260408120805484929061183b908490611f29565b92505081905550818160050160008282546118569190611f29565b9091555050604051828152339084907feabb9788e481eecaf208dddaa2d5c986c2177e4b1b00cc25792b87c9ed00e7b29060200160405180910390a3505050565b8280546118a390611f70565b90600052602060002090601f0160209004810192826118c5576000855561190b565b82601f106118de5782800160ff1982351617855561190b565b8280016001018555821561190b579182015b8281111561190b5782358255916020019190600101906118f0565b5061191792915061198f565b5090565b82805461192790611f70565b90600052602060002090601f016020900481019282611949576000855561190b565b82601f1061196257805160ff191683800117855561190b565b8280016001018555821561190b579182015b8281111561190b578251825591602001919060010190611974565b5b808211156119175760008155600101611990565b80356001600160a01b03811681146119bb57600080fd5b919050565b6000602082840312156119d257600080fd5b6119db826119a4565b9392505050565b600080604083850312156119f557600080fd5b8251611a0081612008565b602084015190925067ffffffffffffffff80821115611a1e57600080fd5b818501915085601f830112611a3257600080fd5b815181811115611a4457611a44611ff2565b604051601f8201601f19908116603f01168101908382118183101715611a6c57611a6c611ff2565b81604052828152886020848701011115611a8557600080fd5b611a96836020830160208801611f40565b80955050505050509250929050565b60008060008060408587031215611abb57600080fd5b843567ffffffffffffffff80821115611ad357600080fd5b818701915087601f830112611ae757600080fd5b813581811115611af657600080fd5b886020828501011115611b0857600080fd5b602092830196509450908601359080821115611b2357600080fd5b818701915087601f830112611b3757600080fd5b813581811115611b4657600080fd5b8860208260051b8501011115611b5b57600080fd5b95989497505060200194505050565b600060408284031215611b7c57600080fd5b6040516040810181811067ffffffffffffffff82111715611b9f57611b9f611ff2565b604052825181526020928301519281019290925250919050565b600060208284031215611bcb57600080fd5b5035919050565b600060208284031215611be457600080fd5b5051919050565b60008060408385031215611bfe57600080fd5b82359150611c0e602084016119a4565b90509250929050565b60008060408385031215611c2a57600080fd5b50508035926020909101359150565b60008060008060808587031215611c4f57600080fd5b84359350602085013592506040850135611c6881612008565b9396929550929360600135925050565b600080600060608486031215611c8d57600080fd5b505081359360208301359350604090920135919050565b60008151808452611cbc816020860160208601611f40565b601f01601f19169290920160200192915050565b6001600160a01b0384168152600060206060818401526000855481600182811c915080831680611d0157607f831692505b858310811415611d1f57634e487b7160e01b85526022600452602485fd5b6060880183905260808801818015611d3e5760018114611d4f57611d7a565b60ff19861682528782019650611d7a565b60008c81526020902060005b86811015611d7457815484820152908501908901611d5b565b83019750505b5050505050508092505050826040830152949350505050565b60006101406001600160a01b038d168352806020840152611db68184018d611ca4565b604084019b909b5250506060810197909752608087019590955292151560a086015260c085019190915260e08401526101008301526101209091015292915050565b60006020808301818452808551808352604092508286019150828160051b87010184880160005b83811015611e7457603f19898403018552815160606001600160a01b03825116855288820151818a870152611e5682870182611ca4565b92890151958901959095525094870194925090860190600101611e1f565b509098975050505050505050565b828152604060208201526000611e9b6040830184611ca4565b949350505050565b6000808335601e19843603018112611eba57600080fd5b83018035915067ffffffffffffffff821115611ed557600080fd5b602001915036819003821315611eea57600080fd5b9250929050565b60008235605e19833603018112611f0757600080fd5b9190910192915050565b60008219821115611f2457611f24611fc6565b500190565b600082821015611f3b57611f3b611fc6565b500390565b60005b83811015611f5b578181015183820152602001611f43565b83811115611f6a576000848401525b50505050565b600181811c90821680611f8457607f821691505b60208210811415611fa557634e487b7160e01b600052602260045260246000fd5b50919050565b6000600019821415611fbf57611fbf611fc6565b5060010190565b634e487b7160e01b600052601160045260246000fd5b634e487b7160e01b600052603260045260246000fd5b634e487b7160e01b600052604160045260246000fd5b801515811461201657600080fd5b5056fea2646970667358221220ab667865bf3a87c17ab6e6423d76bb7aab2783c2a3b9806ed925bdd5490dabff64736f6c63430008060033
Constructor Arguments (ABI-Encoded and is the last bytes of the Contract Creation Code above)
000000000000000000000000ee6a649aa3766bd117e12c161726b693a1b2ee20000000000000000000000000b6d513f6222ee92fff975e901bd792e2513fb53b
-----Decoded View---------------
Arg [0] : _stakingContract (address): 0xEE6A649Aa3766bD117e12C161726b693A1B2Ee20
Arg [1] : _delegator (address): 0xB6d513f6222Ee92Fff975E901bd792E2513fB53B
-----Encoded View---------------
2 Constructor Arguments found :
Arg [0] : 000000000000000000000000ee6a649aa3766bd117e12c161726b693a1b2ee20
Arg [1] : 000000000000000000000000b6d513f6222ee92fff975e901bd792e2513fb53b
Loading...
Loading
Loading...
Loading
Net Worth in USD
$0.00
Net Worth in ETH
0
Multichain Portfolio | 34 Chains
| Chain | Token | Portfolio % | Price | Amount | Value |
|---|
Loading...
Loading
Loading...
Loading
Loading...
Loading
[ Download: CSV Export ]
A contract address hosts a smart contract, which is a set of code stored on the blockchain that runs when predetermined conditions are met. Learn more about addresses in our Knowledge Base.