Polygon Sponsored slots available. Book your slot here!
More Info
Private Name Tags
ContractCreator
Latest 25 from a total of 57 transactions
Transaction Hash |
Method
|
Block
|
From
|
To
|
|||||
---|---|---|---|---|---|---|---|---|---|
Vote | 63641601 | 34 days ago | IN | 0 POL | 0.00339577 | ||||
Vote | 63641544 | 34 days ago | IN | 0 POL | 0.00487446 | ||||
Call Vote | 63555685 | 36 days ago | IN | 0 POL | 0.00230178 | ||||
Sponsor Proposal | 63555648 | 36 days ago | IN | 0 POL | 0.00405927 | ||||
Create Proposal | 63153553 | 46 days ago | IN | 0 POL | 0.06939296 | ||||
Execute Proposal | 40143539 | 634 days ago | IN | 0 POL | 0.01978139 | ||||
Vote | 39941046 | 639 days ago | IN | 0 POL | 0.00684816 | ||||
Vote | 39926401 | 640 days ago | IN | 0 POL | 0.00637027 | ||||
Vote | 39925479 | 640 days ago | IN | 0 POL | 0.00667889 | ||||
Vote | 39925385 | 640 days ago | IN | 0 POL | 0.00743474 | ||||
Vote | 39925371 | 640 days ago | IN | 0 POL | 0.00758878 | ||||
Vote | 39925032 | 640 days ago | IN | 0 POL | 0.00871423 | ||||
Vote | 39924885 | 640 days ago | IN | 0 POL | 0.0072627 | ||||
Vote | 39924731 | 640 days ago | IN | 0 POL | 0.0057776 | ||||
Vote | 39924694 | 640 days ago | IN | 0 POL | 0.00691034 | ||||
Vote | 39878498 | 641 days ago | IN | 0 POL | 0.01194394 | ||||
Vote | 39878456 | 641 days ago | IN | 0 POL | 0.01441284 | ||||
Vote | 39870895 | 641 days ago | IN | 0 POL | 0.01222723 | ||||
Call Vote | 39789147 | 643 days ago | IN | 0 POL | 0.00934158 | ||||
Sponsor Proposal | 39789121 | 643 days ago | IN | 0 POL | 0.01417491 | ||||
Create Proposal | 39743588 | 644 days ago | IN | 0 POL | 0.03593911 | ||||
Execute Proposal | 38579745 | 675 days ago | IN | 0 POL | 0.00718577 | ||||
Vote | 38481905 | 677 days ago | IN | 0 POL | 0.00345465 | ||||
Vote | 38464131 | 678 days ago | IN | 0 POL | 0.00545615 | ||||
Vote | 38460182 | 678 days ago | IN | 0 POL | 0.00339017 |
Loading...
Loading
Contract Name:
Voting
Compiler Version
v0.8.12+commit.f00d7308
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.7; 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; uint256 public constant SPONSOR_LOCKOUT_TIME = 7 days; // Threshold constants uint256 public constant QUORUM = 2000000e18; // 2 million, 18 decimal places uint256 public constant PROPOSAL_SPONSOR_THRESHOLD = 500000e18; // 500 thousand, 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); // Proposal executed event VoteKeySet(address indexed account, address votingKey); // Errors error ExecutionFailed(uint256 index, bytes data); // Function call struct Call { address callContract; bytes data; uint256 value; } // Governance proposals struct ProposalStruct { // Execution status bool executed; // 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; // 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; // Voting keys mapping(address => address) public votingKey; // Last sponsored proposal data struct LastSponsored { uint256 lastSponsorTime; uint256 proposalID; } mapping(address => LastSponsored) public lastSponsored; /* solhint-disable var-name-mixedcase */ Staking public immutable STAKING_CONTRACT; Delegator public immutable DELEGATOR_CONTRACT; /* solhint-enable var-name-mixedcase */ // Only voting key modifier modifier onlyVotingKey(address _account) { // Only voting key or main key can call require( msg.sender == _account || msg.sender == votingKey[_account], "Voting: Caller not authorized" ); _; } /** * @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 Gets votes cast by an account on a particular proposal * @dev Gets votes from proposal as mappings wont be returned on public getter * @param _id - Proposal to get votes for * @param _account - Account to get votes for * @return votes amount */ function getVotes(uint256 _id, address _account) external view returns (uint256) { return proposals[_id].voted[_account]; } /** * @notice Sets voting key for account * @param _votingKey - voting key address */ function setVotingKey(address _votingKey) external { votingKey[msg.sender] = _votingKey; emit VoteKeySet(msg.sender, _votingKey); } /** * @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 _account - account to vote with * @param _hint - hint for snapshot search */ function sponsorProposal(uint256 _id, uint256 _amount, address _account, uint256 _hint) external onlyVotingKey(_account) { // Prevent proposal spam require( lastSponsored[_account].proposalID == _id || block.timestamp - lastSponsored[_account].lastSponsorTime > 7 days, "Voting: Can only sponsor one proposal per week" ); 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"); // Set last sponsored info lastSponsored[_account].proposalID = _id; lastSponsored[_account].lastSponsorTime = block.timestamp; // Get address sponsor voting power Staking.AccountSnapshot memory snapshot = STAKING_CONTRACT.accountSnapshotAt( _account, proposal.sponsorInterval, _hint ); // Can't sponsor with more than voting power require(proposal.sponsors[_account] + _amount <= snapshot.votingPower, "Voting: Not enough voting power"); // Update address sponsorship amount on proposal proposal.sponsors[_account] += _amount; // Update sponsor total proposal.sponsorship += _amount; // Emit event emit Sponsorship(_id, _account, _amount); } /** * @notice Unsponsor proposal * @param _id - id of proposal to sponsor * @param _account - account to vote with * @param _amount - amount to sponsor with */ function unsponsorProposal(uint256 _id, uint256 _amount, address _account) external onlyVotingKey(_account) { 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[_account], "Voting: Amount greater than sponsored"); // Update address sponsorship amount on proposal proposal.sponsors[_account] -= _amount; // Update sponsor total proposal.sponsorship -= _amount; // Emit event emit SponsorshipRevocation(_id, _account, _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 _account - account to vote with * @param _hint - hint for snapshot search */ function vote(uint256 _id, uint256 _amount, bool _affirmative, address _account, uint256 _hint) external onlyVotingKey(_account) { 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( _account, proposal.votingInterval, _hint ); // Check address isn't voting with more voting power than it has require(proposal.voted[_account] + _amount <= snapshot.votingPower, "Voting: Not enough voting power to cast this vote"); // Update account voted amount proposal.voted[_account] += _amount; // Update voting totals if (_affirmative) { proposal.yayVotes += _amount; } else { proposal.nayVotes += _amount; } // Emit event emit VoteCast(_id, _account, _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 >= 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[] memory 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) { revert ExecutionFailed(i, returnData); } } // Emit event emit Execution(_id); } }
// SPDX-License-Identifier: UNLICENSED pragma solidity ^0.8.7; 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 virtual 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.7; 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 // OpenZeppelin Contracts (last updated v4.5.0) (utils/math/Math.sol) 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. return (a & b) + (a ^ b) / 2; } /** * @dev Returns the ceiling of the division of two numbers. * * This differs from standard division with `/` in that it rounds up instead * of rounding down. */ function ceilDiv(uint256 a, uint256 b) internal pure returns (uint256) { // (a + b - 1) / b can overflow on addition, so we distribute. return a / b + (a % b == 0 ? 0 : 1); } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts v4.4.1 (token/ERC20/utils/SafeERC20.sol) 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' 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 require(abi.decode(returndata, (bool)), "SafeERC20: ERC20 operation did not succeed"); } } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.5.0) (token/ERC20/IERC20.sol) 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 `to`. * * Returns a boolean value indicating whether the operation succeeded. * * Emits a {Transfer} event. */ function transfer(address to, 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 `from` to `to` 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 from, address to, 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 // OpenZeppelin Contracts (last updated v4.5.0) (utils/Address.sol) pragma solidity ^0.8.1; /** * @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 * ==== * * [IMPORTANT] * ==== * You shouldn't rely on `isContract` to protect against flash loan attacks! * * Preventing calls from contracts is highly discouraged. It breaks composability, breaks support for smart wallets * like Gnosis Safe, and does not provide security since it can be circumvented by calling from a contract * constructor. * ==== */ function isContract(address account) internal view returns (bool) { // This method relies on extcodesize/address.code.length, which returns 0 // for contracts in construction, since the code is only stored at the end // of the constructor execution. return account.code.length > 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"); (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"); (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"); (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"); (bool success, bytes memory returndata) = target.delegatecall(data); return verifyCallResult(success, returndata, errorMessage); } /** * @dev Tool to verifies that a low level call was successful, and revert if it wasn't, either by bubbling the * revert reason using the provided one. * * _Available since v4.3._ */ function verifyCallResult( bool success, bytes memory returndata, string memory errorMessage ) internal 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 assembly { let returndata_size := mload(returndata) revert(add(32, returndata), returndata_size) } } else { revert(errorMessage); } } } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts v4.4.1 (access/Ownable.sol) 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() { _transferOwnership(_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 { _transferOwnership(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"); _transferOwnership(newOwner); } /** * @dev Transfers ownership of the contract to a new account (`newOwner`). * Internal function without access restriction. */ function _transferOwnership(address newOwner) internal virtual { address oldOwner = _owner; _owner = newOwner; emit OwnershipTransferred(oldOwner, newOwner); } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts v4.4.1 (utils/Context.sol) 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) { return msg.data; } }
{ "optimizer": { "enabled": true, "runs": 1600 }, "outputSelection": { "*": { "*": [ "evm.bytecode", "evm.deployedBytecode", "devdoc", "userdoc", "metadata", "abi" ] } }, "libraries": {} }
Contract Security Audit
- No Contract Security Audit Submitted- Submit Audit Here
[{"inputs":[{"internalType":"contract Staking","name":"_stakingContract","type":"address"},{"internalType":"contract Delegator","name":"_delegator","type":"address"}],"stateMutability":"nonpayable","type":"constructor"},{"inputs":[{"internalType":"uint256","name":"index","type":"uint256"},{"internalType":"bytes","name":"data","type":"bytes"}],"name":"ExecutionFailed","type":"error"},{"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"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"account","type":"address"},{"indexed":false,"internalType":"address","name":"votingKey","type":"address"}],"name":"VoteKeySet","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_LOCKOUT_TIME","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":"_id","type":"uint256"},{"internalType":"address","name":"_account","type":"address"}],"name":"getVotes","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"lastSponsored","outputs":[{"internalType":"uint256","name":"lastSponsorTime","type":"uint256"},{"internalType":"uint256","name":"proposalID","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"}],"name":"proposals","outputs":[{"internalType":"bool","name":"executed","type":"bool"},{"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":"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":"address","name":"_votingKey","type":"address"}],"name":"setVotingKey","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_id","type":"uint256"},{"internalType":"uint256","name":"_amount","type":"uint256"},{"internalType":"address","name":"_account","type":"address"},{"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"},{"internalType":"address","name":"_account","type":"address"}],"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":"address","name":"_account","type":"address"},{"internalType":"uint256","name":"_hint","type":"uint256"}],"name":"vote","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"votingKey","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"}]
Contract Creation Code
60c06040523480156200001157600080fd5b50604051620025b1380380620025b1833981016040819052620000349162000065565b6001600160a01b039182166080521660a052620000a4565b6001600160a01b03811681146200006257600080fd5b50565b600080604083850312156200007957600080fd5b825162000086816200004c565b602084015190925062000099816200004c565b809150509250929050565b60805160a0516124c4620000ed600039600081816102d9015261093201526000818161039f01528181610d6401528181611109015281816113cb01526118ac01526124c46000f3fe608060405234801561001057600080fd5b50600436106101a35760003560e01c806351ec8bd7116100ee5780639f31490d11610097578063da19ddfb11610071578063da19ddfb1461039a578063de3479df146103c1578063df4c119a14610354578063f4f624e5146103d457600080fd5b80639f31490d14610354578063b84882b31461035e578063b84bddf41461037157600080fd5b80636b10b120116100c85780636b10b120146103265780636d0799321461033957806389508d151461034357600080fd5b806351ec8bd7146102c157806359383bf5146102d4578063681973601461031357600080fd5b80632e80d9b61161015057806344c7c8671161012a57806344c7c8671461029357806349fe1f6d1461029b5780634f526875146102ae57600080fd5b80632e80d9b614610257578063328dd9821461026957806342f87abf1461028957600080fd5b80630fe0554d116101815780630fe0554d14610207578063191d79a7146102115780631af74bc31461024d57600080fd5b8063013cf08b146101a857806305fe07fb146101da5780630d61b519146101f2575b600080fd5b6101bb6101b6366004611e74565b6103e7565b6040516101d19a99989796959493929190611ee9565b60405180910390f35b6101e46212750081565b6040519081526020016101d1565b610205610200366004611e74565b6104dc565b005b6101e462278d0081565b61023861021f366004611f6b565b6002602052600090815260409020805460019091015482565b604080519283526020830191909152016101d1565b6101e46206978081565b6101e46a01a784379d99db4200000081565b61027c610277366004611e74565b610aa3565b6040516101d19190611f8d565b6101e46207e90081565b6000546101e4565b6102056102a9366004611e74565b610bd8565b6102056102bc366004612028565b610e18565b6101e46102cf366004612078565b6112e1565b6102fb7f000000000000000000000000000000000000000000000000000000000000000081565b6040516001600160a01b0390911681526020016101d1565b6101e461032136600461213d565b611605565b610205610334366004612169565b61164a565b6101e46202a30081565b6101e46969e10de76676d080000081565b6101e462093a8081565b61020561036c366004611f6b565b611a31565b6102fb61037f366004611f6b565b6001602052600090815260409020546001600160a01b031681565b6102fb7f000000000000000000000000000000000000000000000000000000000000000081565b6101e46103cf36600461213d565b611aab565b6102056103e23660046121a6565b611af0565b600081815481106103f757600080fd5b60009182526020909120600c90910201805460018201805460ff831694506101009092046001600160a01b0316929161042f906121db565b80601f016020809104026020016040519081016040528092919081815260200182805461045b906121db565b80156104a85780601f1061047d576101008083540402835291602001916104a8565b820191906000526020600020905b81548152906001019060200180831161048b57829003601f168201915b50505050509080600301549080600401549080600501549080600801549080600901549080600a01549080600b015490508a565b60008082815481106104f0576104f0612216565b90600052602060002090600c02019050600081600401541161057f5760405162461bcd60e51b815260206004820152603160248201527f566f74696e673a20566f7465206861736e2774206265656e2063616c6c65642060448201527f666f7220746869732070726f706f73616c00000000000000000000000000000060648201526084015b60405180910390fd5b6a01a784379d99db42000000816008015410156106045760405162461bcd60e51b815260206004820152602260248201527f566f74696e673a2051756f72756d206861736e2774206265656e20726561636860448201527f65640000000000000000000000000000000000000000000000000000000000006064820152608401610576565b80600901548160080154116106815760405162461bcd60e51b815260206004820152602360248201527f566f74696e673a2050726f706f73616c206861736e277420706173736564207660448201527f6f746500000000000000000000000000000000000000000000000000000000006064820152608401610576565b62093a8081600401546106949190612242565b42116107085760405162461bcd60e51b815260206004820152602660248201527f566f74696e673a20457865637574696f6e2077696e646f77206861736e27742060448201527f6f70656e656400000000000000000000000000000000000000000000000000006064820152608401610576565b62127500816004015461071b9190612242565b421061078f5760405162461bcd60e51b815260206004820152602360248201527f566f74696e673a20457865637574696f6e2077696e646f772068617320636c6f60448201527f73656400000000000000000000000000000000000000000000000000000000006064820152608401610576565b805460ff16156108075760405162461bcd60e51b815260206004820152602a60248201527f566f74696e673a2050726f706f73616c2068617320616c72656164792062656560448201527f6e206578656375746564000000000000000000000000000000000000000000006064820152608401610576565b805460ff191660011781556002810180546040805160208084028201810190925282815260009390929091849084015b8282101561091b57600084815260209081902060408051606081019091526003850290910180546001600160a01b031682526001810180549293919291840191610880906121db565b80601f01602080910402602001604051908101604052809291908181526020018280546108ac906121db565b80156108f95780601f106108ce576101008083540402835291602001916108f9565b820191906000526020600020905b8154815290600101906020018083116108dc57829003601f168201915b5050505050815260200160028201548152505081526020019060010190610837565b50505050905060005b8151811015610a72576000807f00000000000000000000000000000000000000000000000000000000000000006001600160a01b031663c6b295c185858151811061097157610971612216565b60200260200101516000015186868151811061098f5761098f612216565b6020026020010151602001518787815181106109ad576109ad612216565b6020026020010151604001516040518463ffffffff1660e01b81526004016109d79392919061225a565b6000604051808303816000875af11580156109f6573d6000803e3d6000fd5b505050506040513d6000823e601f3d908101601f19168201604052610a1e91908101906122a2565b9150915081610a5d5782816040517f2c4029e9000000000000000000000000000000000000000000000000000000008152600401610576929190612365565b50508080610a6a90612386565b915050610924565b5060405183907f33e13ecb54c3076d8e8bb8c2881800a4d972b792045ffae98fdf46df365fed7590600090a2505050565b606060008281548110610ab857610ab8612216565b90600052602060002090600c0201600201805480602002602001604051908101604052809291908181526020016000905b82821015610bcd57600084815260209081902060408051606081019091526003850290910180546001600160a01b031682526001810180549293919291840191610b32906121db565b80601f0160208091040260200160405190810160405280929190818152602001828054610b5e906121db565b8015610bab5780601f10610b8057610100808354040283529160200191610bab565b820191906000526020600020905b815481529060010190602001808311610b8e57829003601f168201915b5050505050815260200160028201548152505081526020019060010190610ae9565b505050509050919050565b6000808281548110610bec57610bec612216565b90600052602060002090600c0201905062278d008160030154610c0f9190612242565b4210610c5d5760405162461bcd60e51b815260206004820181905260248201527f566f74696e673a2053706f6e736f72696e672077696e646f77207061737365646044820152606401610576565b600481015415610cd55760405162461bcd60e51b815260206004820152602560248201527f566f74696e673a2050726f706f73616c20616c726561647920676f6e6520746f60448201527f20766f74650000000000000000000000000000000000000000000000000000006064820152608401610576565b6969e10de76676d080000081600501541015610d595760405162461bcd60e51b815260206004820152602160248201527f566f74696e673a2053706f6e736f72207468726573686f6c64206e6f74206d6560448201527f74000000000000000000000000000000000000000000000000000000000000006064820152608401610576565b4281600401819055507f00000000000000000000000000000000000000000000000000000000000000006001600160a01b031663363487bc6040518163ffffffff1660e01b8152600401602060405180830381865afa158015610dc0573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610de491906123a1565b600b82015560405182907f4bebca4d6e9291871f8e494321fae9eba02c07d0844ff387b8efd20c7a41de3290600090a25050565b81336001600160a01b0382161480610e4957506001600160a01b038181166000908152600160205260409020541633145b610e955760405162461bcd60e51b815260206004820152601d60248201527f566f74696e673a2043616c6c6572206e6f7420617574686f72697a65640000006044820152606401610576565b6000808781548110610ea957610ea9612216565b90600052602060002090600c020190506000816004015411610f335760405162461bcd60e51b815260206004820152603160248201527f566f74696e673a20566f7465206861736e2774206265656e2063616c6c65642060448201527f666f7220746869732070726f706f73616c0000000000000000000000000000006064820152608401610576565b6202a3008160040154610f469190612242565b4211610fba5760405162461bcd60e51b815260206004820152602360248201527f566f74696e673a20566f74696e672077696e646f77206861736e2774206f706560448201527f6e656400000000000000000000000000000000000000000000000000000000006064820152608401610576565b841561104c57620697808160040154610fd39190612242565b42106110475760405162461bcd60e51b815260206004820152602c60248201527f566f74696e673a2041666669726d617469766520766f74696e672077696e646f60448201527f772068617320636c6f73656400000000000000000000000000000000000000006064820152608401610576565b6110d3565b6207e900816004015461105f9190612242565b42106110d35760405162461bcd60e51b815260206004820152602960248201527f566f74696e673a204e6567617469766520766f74696e672077696e646f77206860448201527f617320636c6f73656400000000000000000000000000000000000000000000006064820152608401610576565b600b81015460405163046a32f360e11b81526001600160a01b0386811660048301526024820192909252604481018590526000917f000000000000000000000000000000000000000000000000000000000000000016906308d465e6906064016040805180830381865afa15801561114f573d6000803e3d6000fd5b505050506040513d601f19601f8201168201806040525081019061117391906123ba565b9050806020015187836007016000886001600160a01b03166001600160a01b03168152602001908152602001600020546111ad9190612242565b11156112215760405162461bcd60e51b815260206004820152603160248201527f566f74696e673a204e6f7420656e6f75676820766f74696e6720706f7765722060448201527f746f2063617374207468697320766f74650000000000000000000000000000006064820152608401610576565b6001600160a01b03851660009081526007830160205260408120805489929061124b908490612242565b90915550508515611275578682600801600082825461126a9190612242565b9091555061128f9050565b868260090160008282546112899190612242565b90915550505b604080518715158152602081018990526001600160a01b038716918a917fcbdf6214089cba887ecbf35a0b6a734589959c9763342c756bb2a80ca2bc9f6e910160405180910390a35050505050505050565b6000816113305760405162461bcd60e51b815260206004820152601c60248201527f566f74696e673a204e6f20616374696f6e7320737065636966696564000000006044820152606401610576565b600080546001810182559080527f290decd9548b62a8d60345a988386fc84ba6bc95484008f6362f93160ef3e563600c820290810180547fffffffffffffffffffffff0000000000000000000000000000000000000000ff163361010002178155906113bf907f290decd9548b62a8d60345a988386fc84ba6bc95484008f6362f93160ef3e564018888611d67565b504281600301819055507f00000000000000000000000000000000000000000000000000000000000000006001600160a01b031663363487bc6040518163ffffffff1660e01b8152600401602060405180830381865afa158015611427573d6000803e3d6000fd5b505050506040513d601f19601f8201168201806040525081019061144b91906123a1565b600a82015560005b848110156115cd5781600201604051806060016040528088888581811061147c5761147c612216565b905060200281019061148e9190612409565b61149c906020810190611f6b565b6001600160a01b031681526020018888858181106114bc576114bc612216565b90506020028101906114ce9190612409565b6114dc906020810190612429565b8080601f01602080910402602001604051908101604052809392919081815260200183838082843760009201919091525050509082525060200188888581811061152857611528612216565b905060200281019061153a9190612409565b6040013590528154600180820184556000938452602093849020835160039093020180547fffffffffffffffffffffffff0000000000000000000000000000000000000000166001600160a01b03909316929092178255828401518051939492936115ad93928501929190910190611deb565b5060408201518160020155505080806115c590612386565b915050611453565b50604051339083907fe56c149b9a4f632c327b2bc8b691ebfd0d76354ca7b678faea1effbd9d9f223d90600090a35095945050505050565b600080838154811061161957611619612216565b600091825260208083206001600160a01b03861684526007600c909302019190910190526040902054905092915050565b81336001600160a01b038216148061167b57506001600160a01b038181166000908152600160205260409020541633145b6116c75760405162461bcd60e51b815260206004820152601d60248201527f566f74696e673a2043616c6c6572206e6f7420617574686f72697a65640000006044820152606401610576565b6001600160a01b03831660009081526002602052604090206001015485148061171557506001600160a01b03831660009081526002602052604090205462093a80906117139042612477565b115b6117875760405162461bcd60e51b815260206004820152602e60248201527f566f74696e673a2043616e206f6e6c792073706f6e736f72206f6e652070726f60448201527f706f73616c20706572207765656b0000000000000000000000000000000000006064820152608401610576565b600080868154811061179b5761179b612216565b90600052602060002090600c0201905080600401546000146117ff5760405162461bcd60e51b815260206004820152601460248201527f566f74696e673a20476f6e6520746f20766f74650000000000000000000000006044820152606401610576565b62278d0081600301546118129190612242565b42106118605760405162461bcd60e51b815260206004820181905260248201527f566f74696e673a2053706f6e736f72696e672077696e646f77207061737365646044820152606401610576565b6001600160a01b03848116600081815260026020526040808220600181018b9055429055600a850154905163046a32f360e11b81526004810193909352602483015260448201869052917f000000000000000000000000000000000000000000000000000000000000000016906308d465e6906064016040805180830381865afa1580156118f2573d6000803e3d6000fd5b505050506040513d601f19601f8201168201806040525081019061191691906123ba565b9050806020015186836006016000886001600160a01b03166001600160a01b03168152602001908152602001600020546119509190612242565b111561199e5760405162461bcd60e51b815260206004820152601f60248201527f566f74696e673a204e6f7420656e6f75676820766f74696e6720706f776572006044820152606401610576565b6001600160a01b0385166000908152600683016020526040812080548892906119c8908490612242565b92505081905550858260050160008282546119e39190612242565b90915550506040518681526001600160a01b0386169088907fdf1985be8066f51229b5a3b530a445682812a3c384a5414ed34fa80e892dce0a9060200160405180910390a350505050505050565b3360008181526001602090815260409182902080547fffffffffffffffffffffffff0000000000000000000000000000000000000000166001600160a01b03861690811790915591519182527f54b2c36a8e9609b3856a128ab0d9388e54e17d25dcc1b3332e9a2bfe8a5a12e2910160405180910390a250565b6000808381548110611abf57611abf612216565b600091825260208083206001600160a01b03861684526006600c909302019190910190526040902054905092915050565b80336001600160a01b0382161480611b2157506001600160a01b038181166000908152600160205260409020541633145b611b6d5760405162461bcd60e51b815260206004820152601d60248201527f566f74696e673a2043616c6c6572206e6f7420617574686f72697a65640000006044820152606401610576565b6000808581548110611b8157611b81612216565b90600052602060002090600c020190508060040154600014611be55760405162461bcd60e51b815260206004820152601460248201527f566f74696e673a20476f6e6520746f20766f74650000000000000000000000006044820152606401610576565b62278d008160030154611bf89190612242565b4210611c465760405162461bcd60e51b815260206004820181905260248201527f566f74696e673a2053706f6e736f72696e672077696e646f77207061737365646044820152606401610576565b6001600160a01b0383166000908152600682016020526040902054841115611cd65760405162461bcd60e51b815260206004820152602560248201527f566f74696e673a20416d6f756e742067726561746572207468616e2073706f6e60448201527f736f7265640000000000000000000000000000000000000000000000000000006064820152608401610576565b6001600160a01b038316600090815260068201602052604081208054869290611d00908490612477565b9250508190555083816005016000828254611d1b9190612477565b90915550506040518481526001600160a01b0384169086907feabb9788e481eecaf208dddaa2d5c986c2177e4b1b00cc25792b87c9ed00e7b29060200160405180910390a35050505050565b828054611d73906121db565b90600052602060002090601f016020900481019282611d955760008555611ddb565b82601f10611dae5782800160ff19823516178555611ddb565b82800160010185558215611ddb579182015b82811115611ddb578235825591602001919060010190611dc0565b50611de7929150611e5f565b5090565b828054611df7906121db565b90600052602060002090601f016020900481019282611e195760008555611ddb565b82601f10611e3257805160ff1916838001178555611ddb565b82800160010185558215611ddb579182015b82811115611ddb578251825591602001919060010190611e44565b5b80821115611de75760008155600101611e60565b600060208284031215611e8657600080fd5b5035919050565b60005b83811015611ea8578181015183820152602001611e90565b83811115611eb7576000848401525b50505050565b60008151808452611ed5816020860160208601611e8d565b601f01601f19169290920160200192915050565b60006101408c151583526001600160a01b038c166020840152806040840152611f148184018c611ebd565b606084019a909a525050608081019690965260a086019490945260c085019290925260e0840152610100830152610120909101529392505050565b80356001600160a01b0381168114611f6657600080fd5b919050565b600060208284031215611f7d57600080fd5b611f8682611f4f565b9392505050565b60006020808301818452808551808352604092508286019150828160051b87010184880160005b8381101561200957603f19898403018552815160606001600160a01b03825116855288820151818a870152611feb82870182611ebd565b92890151958901959095525094870194925090860190600101611fb4565b509098975050505050505050565b801515811461202557600080fd5b50565b600080600080600060a0868803121561204057600080fd5b8535945060208601359350604086013561205981612017565b925061206760608701611f4f565b949793965091946080013592915050565b6000806000806040858703121561208e57600080fd5b843567ffffffffffffffff808211156120a657600080fd5b818701915087601f8301126120ba57600080fd5b8135818111156120c957600080fd5b8860208285010111156120db57600080fd5b6020928301965094509086013590808211156120f657600080fd5b818701915087601f83011261210a57600080fd5b81358181111561211957600080fd5b8860208260051b850101111561212e57600080fd5b95989497505060200194505050565b6000806040838503121561215057600080fd5b8235915061216060208401611f4f565b90509250929050565b6000806000806080858703121561217f57600080fd5b843593506020850135925061219660408601611f4f565b9396929550929360600135925050565b6000806000606084860312156121bb57600080fd5b83359250602084013591506121d260408501611f4f565b90509250925092565b600181811c908216806121ef57607f821691505b6020821081141561221057634e487b7160e01b600052602260045260246000fd5b50919050565b634e487b7160e01b600052603260045260246000fd5b634e487b7160e01b600052601160045260246000fd5b600082198211156122555761225561222c565b500190565b6001600160a01b038416815260606020820152600061227c6060830185611ebd565b9050826040830152949350505050565b634e487b7160e01b600052604160045260246000fd5b600080604083850312156122b557600080fd5b82516122c081612017565b602084015190925067ffffffffffffffff808211156122de57600080fd5b818501915085601f8301126122f257600080fd5b8151818111156123045761230461228c565b604051601f8201601f19908116603f0116810190838211818310171561232c5761232c61228c565b8160405282815288602084870101111561234557600080fd5b612356836020830160208801611e8d565b80955050505050509250929050565b82815260406020820152600061237e6040830184611ebd565b949350505050565b600060001982141561239a5761239a61222c565b5060010190565b6000602082840312156123b357600080fd5b5051919050565b6000604082840312156123cc57600080fd5b6040516040810181811067ffffffffffffffff821117156123ef576123ef61228c565b604052825181526020928301519281019290925250919050565b60008235605e1983360301811261241f57600080fd5b9190910192915050565b6000808335601e1984360301811261244057600080fd5b83018035915067ffffffffffffffff82111561245b57600080fd5b60200191503681900382131561247057600080fd5b9250929050565b6000828210156124895761248961222c565b50039056fea264697066735822122024d50fbf566ba73d45624e5af784075353bdcc2a6d1e84c9c4d9a36ab30b138d64736f6c634300080c00330000000000000000000000009ac2ba4bf7facb0bbb33447e5ff8f8d63b71ddc10000000000000000000000005f67441090fbdf57f1d9f28dd65a29b0bb3e83a7
Deployed Bytecode
0x608060405234801561001057600080fd5b50600436106101a35760003560e01c806351ec8bd7116100ee5780639f31490d11610097578063da19ddfb11610071578063da19ddfb1461039a578063de3479df146103c1578063df4c119a14610354578063f4f624e5146103d457600080fd5b80639f31490d14610354578063b84882b31461035e578063b84bddf41461037157600080fd5b80636b10b120116100c85780636b10b120146103265780636d0799321461033957806389508d151461034357600080fd5b806351ec8bd7146102c157806359383bf5146102d4578063681973601461031357600080fd5b80632e80d9b61161015057806344c7c8671161012a57806344c7c8671461029357806349fe1f6d1461029b5780634f526875146102ae57600080fd5b80632e80d9b614610257578063328dd9821461026957806342f87abf1461028957600080fd5b80630fe0554d116101815780630fe0554d14610207578063191d79a7146102115780631af74bc31461024d57600080fd5b8063013cf08b146101a857806305fe07fb146101da5780630d61b519146101f2575b600080fd5b6101bb6101b6366004611e74565b6103e7565b6040516101d19a99989796959493929190611ee9565b60405180910390f35b6101e46212750081565b6040519081526020016101d1565b610205610200366004611e74565b6104dc565b005b6101e462278d0081565b61023861021f366004611f6b565b6002602052600090815260409020805460019091015482565b604080519283526020830191909152016101d1565b6101e46206978081565b6101e46a01a784379d99db4200000081565b61027c610277366004611e74565b610aa3565b6040516101d19190611f8d565b6101e46207e90081565b6000546101e4565b6102056102a9366004611e74565b610bd8565b6102056102bc366004612028565b610e18565b6101e46102cf366004612078565b6112e1565b6102fb7f0000000000000000000000005f67441090fbdf57f1d9f28dd65a29b0bb3e83a781565b6040516001600160a01b0390911681526020016101d1565b6101e461032136600461213d565b611605565b610205610334366004612169565b61164a565b6101e46202a30081565b6101e46969e10de76676d080000081565b6101e462093a8081565b61020561036c366004611f6b565b611a31565b6102fb61037f366004611f6b565b6001602052600090815260409020546001600160a01b031681565b6102fb7f0000000000000000000000009ac2ba4bf7facb0bbb33447e5ff8f8d63b71ddc181565b6101e46103cf36600461213d565b611aab565b6102056103e23660046121a6565b611af0565b600081815481106103f757600080fd5b60009182526020909120600c90910201805460018201805460ff831694506101009092046001600160a01b0316929161042f906121db565b80601f016020809104026020016040519081016040528092919081815260200182805461045b906121db565b80156104a85780601f1061047d576101008083540402835291602001916104a8565b820191906000526020600020905b81548152906001019060200180831161048b57829003601f168201915b50505050509080600301549080600401549080600501549080600801549080600901549080600a01549080600b015490508a565b60008082815481106104f0576104f0612216565b90600052602060002090600c02019050600081600401541161057f5760405162461bcd60e51b815260206004820152603160248201527f566f74696e673a20566f7465206861736e2774206265656e2063616c6c65642060448201527f666f7220746869732070726f706f73616c00000000000000000000000000000060648201526084015b60405180910390fd5b6a01a784379d99db42000000816008015410156106045760405162461bcd60e51b815260206004820152602260248201527f566f74696e673a2051756f72756d206861736e2774206265656e20726561636860448201527f65640000000000000000000000000000000000000000000000000000000000006064820152608401610576565b80600901548160080154116106815760405162461bcd60e51b815260206004820152602360248201527f566f74696e673a2050726f706f73616c206861736e277420706173736564207660448201527f6f746500000000000000000000000000000000000000000000000000000000006064820152608401610576565b62093a8081600401546106949190612242565b42116107085760405162461bcd60e51b815260206004820152602660248201527f566f74696e673a20457865637574696f6e2077696e646f77206861736e27742060448201527f6f70656e656400000000000000000000000000000000000000000000000000006064820152608401610576565b62127500816004015461071b9190612242565b421061078f5760405162461bcd60e51b815260206004820152602360248201527f566f74696e673a20457865637574696f6e2077696e646f772068617320636c6f60448201527f73656400000000000000000000000000000000000000000000000000000000006064820152608401610576565b805460ff16156108075760405162461bcd60e51b815260206004820152602a60248201527f566f74696e673a2050726f706f73616c2068617320616c72656164792062656560448201527f6e206578656375746564000000000000000000000000000000000000000000006064820152608401610576565b805460ff191660011781556002810180546040805160208084028201810190925282815260009390929091849084015b8282101561091b57600084815260209081902060408051606081019091526003850290910180546001600160a01b031682526001810180549293919291840191610880906121db565b80601f01602080910402602001604051908101604052809291908181526020018280546108ac906121db565b80156108f95780601f106108ce576101008083540402835291602001916108f9565b820191906000526020600020905b8154815290600101906020018083116108dc57829003601f168201915b5050505050815260200160028201548152505081526020019060010190610837565b50505050905060005b8151811015610a72576000807f0000000000000000000000005f67441090fbdf57f1d9f28dd65a29b0bb3e83a76001600160a01b031663c6b295c185858151811061097157610971612216565b60200260200101516000015186868151811061098f5761098f612216565b6020026020010151602001518787815181106109ad576109ad612216565b6020026020010151604001516040518463ffffffff1660e01b81526004016109d79392919061225a565b6000604051808303816000875af11580156109f6573d6000803e3d6000fd5b505050506040513d6000823e601f3d908101601f19168201604052610a1e91908101906122a2565b9150915081610a5d5782816040517f2c4029e9000000000000000000000000000000000000000000000000000000008152600401610576929190612365565b50508080610a6a90612386565b915050610924565b5060405183907f33e13ecb54c3076d8e8bb8c2881800a4d972b792045ffae98fdf46df365fed7590600090a2505050565b606060008281548110610ab857610ab8612216565b90600052602060002090600c0201600201805480602002602001604051908101604052809291908181526020016000905b82821015610bcd57600084815260209081902060408051606081019091526003850290910180546001600160a01b031682526001810180549293919291840191610b32906121db565b80601f0160208091040260200160405190810160405280929190818152602001828054610b5e906121db565b8015610bab5780601f10610b8057610100808354040283529160200191610bab565b820191906000526020600020905b815481529060010190602001808311610b8e57829003601f168201915b5050505050815260200160028201548152505081526020019060010190610ae9565b505050509050919050565b6000808281548110610bec57610bec612216565b90600052602060002090600c0201905062278d008160030154610c0f9190612242565b4210610c5d5760405162461bcd60e51b815260206004820181905260248201527f566f74696e673a2053706f6e736f72696e672077696e646f77207061737365646044820152606401610576565b600481015415610cd55760405162461bcd60e51b815260206004820152602560248201527f566f74696e673a2050726f706f73616c20616c726561647920676f6e6520746f60448201527f20766f74650000000000000000000000000000000000000000000000000000006064820152608401610576565b6969e10de76676d080000081600501541015610d595760405162461bcd60e51b815260206004820152602160248201527f566f74696e673a2053706f6e736f72207468726573686f6c64206e6f74206d6560448201527f74000000000000000000000000000000000000000000000000000000000000006064820152608401610576565b4281600401819055507f0000000000000000000000009ac2ba4bf7facb0bbb33447e5ff8f8d63b71ddc16001600160a01b031663363487bc6040518163ffffffff1660e01b8152600401602060405180830381865afa158015610dc0573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610de491906123a1565b600b82015560405182907f4bebca4d6e9291871f8e494321fae9eba02c07d0844ff387b8efd20c7a41de3290600090a25050565b81336001600160a01b0382161480610e4957506001600160a01b038181166000908152600160205260409020541633145b610e955760405162461bcd60e51b815260206004820152601d60248201527f566f74696e673a2043616c6c6572206e6f7420617574686f72697a65640000006044820152606401610576565b6000808781548110610ea957610ea9612216565b90600052602060002090600c020190506000816004015411610f335760405162461bcd60e51b815260206004820152603160248201527f566f74696e673a20566f7465206861736e2774206265656e2063616c6c65642060448201527f666f7220746869732070726f706f73616c0000000000000000000000000000006064820152608401610576565b6202a3008160040154610f469190612242565b4211610fba5760405162461bcd60e51b815260206004820152602360248201527f566f74696e673a20566f74696e672077696e646f77206861736e2774206f706560448201527f6e656400000000000000000000000000000000000000000000000000000000006064820152608401610576565b841561104c57620697808160040154610fd39190612242565b42106110475760405162461bcd60e51b815260206004820152602c60248201527f566f74696e673a2041666669726d617469766520766f74696e672077696e646f60448201527f772068617320636c6f73656400000000000000000000000000000000000000006064820152608401610576565b6110d3565b6207e900816004015461105f9190612242565b42106110d35760405162461bcd60e51b815260206004820152602960248201527f566f74696e673a204e6567617469766520766f74696e672077696e646f77206860448201527f617320636c6f73656400000000000000000000000000000000000000000000006064820152608401610576565b600b81015460405163046a32f360e11b81526001600160a01b0386811660048301526024820192909252604481018590526000917f0000000000000000000000009ac2ba4bf7facb0bbb33447e5ff8f8d63b71ddc116906308d465e6906064016040805180830381865afa15801561114f573d6000803e3d6000fd5b505050506040513d601f19601f8201168201806040525081019061117391906123ba565b9050806020015187836007016000886001600160a01b03166001600160a01b03168152602001908152602001600020546111ad9190612242565b11156112215760405162461bcd60e51b815260206004820152603160248201527f566f74696e673a204e6f7420656e6f75676820766f74696e6720706f7765722060448201527f746f2063617374207468697320766f74650000000000000000000000000000006064820152608401610576565b6001600160a01b03851660009081526007830160205260408120805489929061124b908490612242565b90915550508515611275578682600801600082825461126a9190612242565b9091555061128f9050565b868260090160008282546112899190612242565b90915550505b604080518715158152602081018990526001600160a01b038716918a917fcbdf6214089cba887ecbf35a0b6a734589959c9763342c756bb2a80ca2bc9f6e910160405180910390a35050505050505050565b6000816113305760405162461bcd60e51b815260206004820152601c60248201527f566f74696e673a204e6f20616374696f6e7320737065636966696564000000006044820152606401610576565b600080546001810182559080527f290decd9548b62a8d60345a988386fc84ba6bc95484008f6362f93160ef3e563600c820290810180547fffffffffffffffffffffff0000000000000000000000000000000000000000ff163361010002178155906113bf907f290decd9548b62a8d60345a988386fc84ba6bc95484008f6362f93160ef3e564018888611d67565b504281600301819055507f0000000000000000000000009ac2ba4bf7facb0bbb33447e5ff8f8d63b71ddc16001600160a01b031663363487bc6040518163ffffffff1660e01b8152600401602060405180830381865afa158015611427573d6000803e3d6000fd5b505050506040513d601f19601f8201168201806040525081019061144b91906123a1565b600a82015560005b848110156115cd5781600201604051806060016040528088888581811061147c5761147c612216565b905060200281019061148e9190612409565b61149c906020810190611f6b565b6001600160a01b031681526020018888858181106114bc576114bc612216565b90506020028101906114ce9190612409565b6114dc906020810190612429565b8080601f01602080910402602001604051908101604052809392919081815260200183838082843760009201919091525050509082525060200188888581811061152857611528612216565b905060200281019061153a9190612409565b6040013590528154600180820184556000938452602093849020835160039093020180547fffffffffffffffffffffffff0000000000000000000000000000000000000000166001600160a01b03909316929092178255828401518051939492936115ad93928501929190910190611deb565b5060408201518160020155505080806115c590612386565b915050611453565b50604051339083907fe56c149b9a4f632c327b2bc8b691ebfd0d76354ca7b678faea1effbd9d9f223d90600090a35095945050505050565b600080838154811061161957611619612216565b600091825260208083206001600160a01b03861684526007600c909302019190910190526040902054905092915050565b81336001600160a01b038216148061167b57506001600160a01b038181166000908152600160205260409020541633145b6116c75760405162461bcd60e51b815260206004820152601d60248201527f566f74696e673a2043616c6c6572206e6f7420617574686f72697a65640000006044820152606401610576565b6001600160a01b03831660009081526002602052604090206001015485148061171557506001600160a01b03831660009081526002602052604090205462093a80906117139042612477565b115b6117875760405162461bcd60e51b815260206004820152602e60248201527f566f74696e673a2043616e206f6e6c792073706f6e736f72206f6e652070726f60448201527f706f73616c20706572207765656b0000000000000000000000000000000000006064820152608401610576565b600080868154811061179b5761179b612216565b90600052602060002090600c0201905080600401546000146117ff5760405162461bcd60e51b815260206004820152601460248201527f566f74696e673a20476f6e6520746f20766f74650000000000000000000000006044820152606401610576565b62278d0081600301546118129190612242565b42106118605760405162461bcd60e51b815260206004820181905260248201527f566f74696e673a2053706f6e736f72696e672077696e646f77207061737365646044820152606401610576565b6001600160a01b03848116600081815260026020526040808220600181018b9055429055600a850154905163046a32f360e11b81526004810193909352602483015260448201869052917f0000000000000000000000009ac2ba4bf7facb0bbb33447e5ff8f8d63b71ddc116906308d465e6906064016040805180830381865afa1580156118f2573d6000803e3d6000fd5b505050506040513d601f19601f8201168201806040525081019061191691906123ba565b9050806020015186836006016000886001600160a01b03166001600160a01b03168152602001908152602001600020546119509190612242565b111561199e5760405162461bcd60e51b815260206004820152601f60248201527f566f74696e673a204e6f7420656e6f75676820766f74696e6720706f776572006044820152606401610576565b6001600160a01b0385166000908152600683016020526040812080548892906119c8908490612242565b92505081905550858260050160008282546119e39190612242565b90915550506040518681526001600160a01b0386169088907fdf1985be8066f51229b5a3b530a445682812a3c384a5414ed34fa80e892dce0a9060200160405180910390a350505050505050565b3360008181526001602090815260409182902080547fffffffffffffffffffffffff0000000000000000000000000000000000000000166001600160a01b03861690811790915591519182527f54b2c36a8e9609b3856a128ab0d9388e54e17d25dcc1b3332e9a2bfe8a5a12e2910160405180910390a250565b6000808381548110611abf57611abf612216565b600091825260208083206001600160a01b03861684526006600c909302019190910190526040902054905092915050565b80336001600160a01b0382161480611b2157506001600160a01b038181166000908152600160205260409020541633145b611b6d5760405162461bcd60e51b815260206004820152601d60248201527f566f74696e673a2043616c6c6572206e6f7420617574686f72697a65640000006044820152606401610576565b6000808581548110611b8157611b81612216565b90600052602060002090600c020190508060040154600014611be55760405162461bcd60e51b815260206004820152601460248201527f566f74696e673a20476f6e6520746f20766f74650000000000000000000000006044820152606401610576565b62278d008160030154611bf89190612242565b4210611c465760405162461bcd60e51b815260206004820181905260248201527f566f74696e673a2053706f6e736f72696e672077696e646f77207061737365646044820152606401610576565b6001600160a01b0383166000908152600682016020526040902054841115611cd65760405162461bcd60e51b815260206004820152602560248201527f566f74696e673a20416d6f756e742067726561746572207468616e2073706f6e60448201527f736f7265640000000000000000000000000000000000000000000000000000006064820152608401610576565b6001600160a01b038316600090815260068201602052604081208054869290611d00908490612477565b9250508190555083816005016000828254611d1b9190612477565b90915550506040518481526001600160a01b0384169086907feabb9788e481eecaf208dddaa2d5c986c2177e4b1b00cc25792b87c9ed00e7b29060200160405180910390a35050505050565b828054611d73906121db565b90600052602060002090601f016020900481019282611d955760008555611ddb565b82601f10611dae5782800160ff19823516178555611ddb565b82800160010185558215611ddb579182015b82811115611ddb578235825591602001919060010190611dc0565b50611de7929150611e5f565b5090565b828054611df7906121db565b90600052602060002090601f016020900481019282611e195760008555611ddb565b82601f10611e3257805160ff1916838001178555611ddb565b82800160010185558215611ddb579182015b82811115611ddb578251825591602001919060010190611e44565b5b80821115611de75760008155600101611e60565b600060208284031215611e8657600080fd5b5035919050565b60005b83811015611ea8578181015183820152602001611e90565b83811115611eb7576000848401525b50505050565b60008151808452611ed5816020860160208601611e8d565b601f01601f19169290920160200192915050565b60006101408c151583526001600160a01b038c166020840152806040840152611f148184018c611ebd565b606084019a909a525050608081019690965260a086019490945260c085019290925260e0840152610100830152610120909101529392505050565b80356001600160a01b0381168114611f6657600080fd5b919050565b600060208284031215611f7d57600080fd5b611f8682611f4f565b9392505050565b60006020808301818452808551808352604092508286019150828160051b87010184880160005b8381101561200957603f19898403018552815160606001600160a01b03825116855288820151818a870152611feb82870182611ebd565b92890151958901959095525094870194925090860190600101611fb4565b509098975050505050505050565b801515811461202557600080fd5b50565b600080600080600060a0868803121561204057600080fd5b8535945060208601359350604086013561205981612017565b925061206760608701611f4f565b949793965091946080013592915050565b6000806000806040858703121561208e57600080fd5b843567ffffffffffffffff808211156120a657600080fd5b818701915087601f8301126120ba57600080fd5b8135818111156120c957600080fd5b8860208285010111156120db57600080fd5b6020928301965094509086013590808211156120f657600080fd5b818701915087601f83011261210a57600080fd5b81358181111561211957600080fd5b8860208260051b850101111561212e57600080fd5b95989497505060200194505050565b6000806040838503121561215057600080fd5b8235915061216060208401611f4f565b90509250929050565b6000806000806080858703121561217f57600080fd5b843593506020850135925061219660408601611f4f565b9396929550929360600135925050565b6000806000606084860312156121bb57600080fd5b83359250602084013591506121d260408501611f4f565b90509250925092565b600181811c908216806121ef57607f821691505b6020821081141561221057634e487b7160e01b600052602260045260246000fd5b50919050565b634e487b7160e01b600052603260045260246000fd5b634e487b7160e01b600052601160045260246000fd5b600082198211156122555761225561222c565b500190565b6001600160a01b038416815260606020820152600061227c6060830185611ebd565b9050826040830152949350505050565b634e487b7160e01b600052604160045260246000fd5b600080604083850312156122b557600080fd5b82516122c081612017565b602084015190925067ffffffffffffffff808211156122de57600080fd5b818501915085601f8301126122f257600080fd5b8151818111156123045761230461228c565b604051601f8201601f19908116603f0116810190838211818310171561232c5761232c61228c565b8160405282815288602084870101111561234557600080fd5b612356836020830160208801611e8d565b80955050505050509250929050565b82815260406020820152600061237e6040830184611ebd565b949350505050565b600060001982141561239a5761239a61222c565b5060010190565b6000602082840312156123b357600080fd5b5051919050565b6000604082840312156123cc57600080fd5b6040516040810181811067ffffffffffffffff821117156123ef576123ef61228c565b604052825181526020928301519281019290925250919050565b60008235605e1983360301811261241f57600080fd5b9190910192915050565b6000808335601e1984360301811261244057600080fd5b83018035915067ffffffffffffffff82111561245b57600080fd5b60200191503681900382131561247057600080fd5b9250929050565b6000828210156124895761248961222c565b50039056fea264697066735822122024d50fbf566ba73d45624e5af784075353bdcc2a6d1e84c9c4d9a36ab30b138d64736f6c634300080c0033
Constructor Arguments (ABI-Encoded and is the last bytes of the Contract Creation Code above)
0000000000000000000000009ac2ba4bf7facb0bbb33447e5ff8f8d63b71ddc10000000000000000000000005f67441090fbdf57f1d9f28dd65a29b0bb3e83a7
-----Decoded View---------------
Arg [0] : _stakingContract (address): 0x9AC2bA4bf7FaCB0bbB33447e5fF8f8D63B71dDC1
Arg [1] : _delegator (address): 0x5f67441090FbDf57F1D9f28dd65a29b0bB3E83a7
-----Encoded View---------------
2 Constructor Arguments found :
Arg [0] : 0000000000000000000000009ac2ba4bf7facb0bbb33447e5ff8f8d63b71ddc1
Arg [1] : 0000000000000000000000005f67441090fbdf57f1d9f28dd65a29b0bb3e83a7
Loading...
Loading
Loading...
Loading
Multichain Portfolio | 29 Chains
Chain | Token | Portfolio % | Price | Amount | Value |
---|
[ 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.