Polygon Sponsored slots available. Book your slot here!
Source Code
Latest 25 from a total of 74 transactions
| Transaction Hash |
|
Block
|
From
|
To
|
|||||
|---|---|---|---|---|---|---|---|---|---|
| Execute Proposal | 74519326 | 172 days ago | IN | 0 POL | 0.0043313 | ||||
| Vote | 74402235 | 175 days ago | IN | 0 POL | 0.00232692 | ||||
| Vote | 74365801 | 176 days ago | IN | 0 POL | 0.00223329 | ||||
| Vote | 74359018 | 176 days ago | IN | 0 POL | 0.00217329 | ||||
| Vote | 74350906 | 176 days ago | IN | 0 POL | 0.00255486 | ||||
| Call Vote | 74234851 | 179 days ago | IN | 0 POL | 0.00230178 | ||||
| Sponsor Proposal | 74234830 | 179 days ago | IN | 0 POL | 0.0029022 | ||||
| Create Proposal | 74233764 | 179 days ago | IN | 0 POL | 0.01488538 | ||||
| Execute Proposal | 74103915 | 182 days ago | IN | 0 POL | 0.01930575 | ||||
| Call Vote | 73994567 | 185 days ago | IN | 0 POL | 0.00230178 | ||||
| Sponsor Proposal | 73994552 | 185 days ago | IN | 0 POL | 0.00303363 | ||||
| Create Proposal | 73991328 | 185 days ago | IN | 0 POL | 0.02977068 | ||||
| Vote | 73838220 | 189 days ago | IN | 0 POL | 0.00268629 | ||||
| Call Vote | 73757269 | 192 days ago | IN | 0 POL | 0.00230178 | ||||
| Sponsor Proposal | 73610036 | 195 days ago | IN | 0 POL | 0.00405963 | ||||
| Create Proposal | 73447682 | 199 days ago | IN | 0 POL | 0.04594302 | ||||
| Create Proposal | 67665076 | 343 days ago | IN | 0 POL | 0.02071356 | ||||
| Vote | 63641601 | 444 days ago | IN | 0 POL | 0.00339577 | ||||
| Vote | 63641544 | 444 days ago | IN | 0 POL | 0.00487446 | ||||
| Call Vote | 63555685 | 446 days ago | IN | 0 POL | 0.00230178 | ||||
| Sponsor Proposal | 63555648 | 446 days ago | IN | 0 POL | 0.00405927 | ||||
| Create Proposal | 63153553 | 456 days ago | IN | 0 POL | 0.06939296 | ||||
| Execute Proposal | 40143539 | 1044 days ago | IN | 0 POL | 0.01978139 | ||||
| Vote | 39941046 | 1050 days ago | IN | 0 POL | 0.00684816 | ||||
| Vote | 39926401 | 1050 days ago | IN | 0 POL | 0.00637027 |
Cross-Chain Transactions
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
Contract ABI
API[{"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
Loading...
Loading
Net Worth in USD
$0.00
Net Worth in POL
Multichain Portfolio | 35 Chains
| Chain | Token | Portfolio % | Price | Amount | Value |
|---|
Loading...
Loading
Loading...
Loading
Loading...
Loading
[ Download: CSV Export ]
A contract address hosts a smart contract, which is a set of code stored on the blockchain that runs when predetermined conditions are met. Learn more about addresses in our Knowledge Base.