Obyte: GBYTE Token
Token Contract
Overview
ETH Balance
0 ETH
Eth Value
$0.00Token Holdings
Latest 25 from a total of 545 transactions
Transaction Hash |
Method
|
Block
|
From
|
To
|
|||||
---|---|---|---|---|---|---|---|---|---|
Approve | 21561363 | 1 hr ago | IN | 0 ETH | 0.00058989 | ||||
Approve | 21539662 | 3 days ago | IN | 0 ETH | 0.00045956 | ||||
Withdraw | 21446701 | 16 days ago | IN | 0 ETH | 0.00086323 | ||||
Withdraw | 21436251 | 17 days ago | IN | 0 ETH | 0.00129119 | ||||
Claim | 21425110 | 19 days ago | IN | 0.4 ETH | 0.0066566 | ||||
Withdraw | 21421343 | 19 days ago | IN | 0 ETH | 0.00079156 | ||||
Approve | 21415064 | 20 days ago | IN | 0 ETH | 0.00066527 | ||||
Approve | 21414765 | 20 days ago | IN | 0 ETH | 0.00077674 | ||||
Claim | 21414757 | 20 days ago | IN | 0.51465 ETH | 0.00609029 | ||||
Approve | 21408067 | 21 days ago | IN | 0 ETH | 0.00036646 | ||||
Transfer | 21403859 | 22 days ago | IN | 0 ETH | 0.00056513 | ||||
Claim | 21399761 | 22 days ago | IN | 0.025 ETH | 0.00400048 | ||||
Approve | 21397775 | 22 days ago | IN | 0 ETH | 0.00028931 | ||||
Approve | 21397770 | 22 days ago | IN | 0 ETH | 0.00025939 | ||||
Withdraw | 21348380 | 29 days ago | IN | 0 ETH | 0.00091508 | ||||
Transfer | 21340354 | 30 days ago | IN | 0 ETH | 0.00108252 | ||||
Transfer | 21340315 | 30 days ago | IN | 0 ETH | 0.00115684 | ||||
Transfer | 21340062 | 30 days ago | IN | 0 ETH | 0.00118468 | ||||
Approve | 21333210 | 31 days ago | IN | 0 ETH | 0.00084431 | ||||
Claim | 21326744 | 32 days ago | IN | 0.2432165 ETH | 0.0065835 | ||||
Transfer | 21253304 | 43 days ago | IN | 0 ETH | 0.00066062 | ||||
Transfer | 21253296 | 43 days ago | IN | 0 ETH | 0.00064658 | ||||
Withdraw | 21232083 | 46 days ago | IN | 0 ETH | 0.00069562 | ||||
Transfer From | 21226442 | 46 days ago | IN | 0 ETH | 0.00045295 | ||||
Approve | 21226425 | 46 days ago | IN | 0 ETH | 0.00051035 |
Latest 25 internal transactions (View All)
Advanced mode:
Parent Transaction Hash | Block |
From
|
To
|
|||
---|---|---|---|---|---|---|
21446701 | 16 days ago | 0.4 ETH | ||||
21436251 | 17 days ago | 0.51465 ETH | ||||
21421343 | 19 days ago | 0.025 ETH | ||||
21348380 | 29 days ago | 0.2432165 ETH | ||||
21232083 | 46 days ago | 0.325 ETH | ||||
21169959 | 54 days ago | 0.29525 ETH | ||||
21070591 | 68 days ago | 0.945 ETH | ||||
20957212 | 84 days ago | 0.25 ETH | ||||
20711312 | 118 days ago | 0.24 ETH | ||||
20625381 | 130 days ago | 0.15625 ETH | ||||
20598394 | 134 days ago | 0.52125 ETH | ||||
20463225 | 153 days ago | 0.25 ETH | ||||
20390240 | 163 days ago | 0.162 ETH | ||||
20335167 | 171 days ago | 0.0325 ETH | ||||
20311732 | 174 days ago | 0.11275 ETH | ||||
20170875 | 194 days ago | 0.2075 ETH | ||||
20052253 | 210 days ago | 1.30325 ETH | ||||
20011516 | 216 days ago | 0.188 ETH | ||||
20004056 | 217 days ago | 0.1545 ETH | ||||
19961282 | 223 days ago | 0.11575 ETH | ||||
19926038 | 228 days ago | 0.10325 ETH | ||||
19728014 | 256 days ago | 0.389 ETH | ||||
19674117 | 263 days ago | 0.24 ETH | ||||
19616566 | 271 days ago | 0.4075 ETH | ||||
19616268 | 271 days ago | 0.2932125 ETH |
Loading...
Loading
Minimal Proxy Contract for 0x66e474364b4d3d6b64a388d990231854ac2cf3a1
Contract Name:
Import
Compiler Version
v0.8.6+commit.11564f7e
Contract Source Code (Solidity)
/** *Submitted for verification at Etherscan.io on 2021-06-28 */ // SPDX-License-Identifier: MIT pragma solidity ^0.8.3; /** * @dev Standard math utilities missing in the Solidity language. */ library Math { /** * @dev Returns the largest of two numbers. */ function max(uint256 a, uint256 b) internal pure returns (uint256) { return a >= b ? a : b; } /** * @dev Returns the smallest of two numbers. */ function min(uint256 a, uint256 b) internal pure returns (uint256) { return a < b ? a : b; } /** * @dev Returns the average of two numbers. The result is rounded towards * zero. */ function average(uint256 a, uint256 b) internal pure returns (uint256) { // (a + b) / 2 can overflow, so we distribute return (a / 2) + (b / 2) + ((a % 2 + b % 2) / 2); } } /* * @dev Provides information about the current execution context, including the * sender of the transaction and its data. While these are generally available * via msg.sender and msg.data, they should not be accessed in such a direct * manner, since when dealing with meta-transactions the account sending and * paying for execution may not be the actual sender (as far as an application * is concerned). * * This contract is only required for intermediate, library-like contracts. */ abstract contract Context { function _msgSender() internal view virtual returns (address) { return msg.sender; } function _msgData() internal view virtual returns (bytes calldata) { this; // silence state mutability warning without generating bytecode - see https://github.com/ethereum/solidity/issues/2691 return msg.data; } } /** * @dev https://eips.ethereum.org/EIPS/eip-1167[EIP 1167] is a standard for * deploying minimal proxy contracts, also known as "clones". * * > To simply and cheaply clone contract functionality in an immutable way, this standard specifies * > a minimal bytecode implementation that delegates all calls to a known, fixed address. * * The library includes functions to deploy a proxy using either `create` (traditional deployment) or `create2` * (salted deterministic deployment). It also includes functions to predict the addresses of clones deployed using the * deterministic method. * * _Available since v3.4._ */ library Clones { /** * @dev Deploys and returns the address of a clone that mimics the behaviour of `implementation`. * * This function uses the create opcode, which should never revert. */ function clone(address implementation) internal returns (address instance) { // solhint-disable-next-line no-inline-assembly assembly { let ptr := mload(0x40) mstore(ptr, 0x3d602d80600a3d3981f3363d3d373d3d3d363d73000000000000000000000000) mstore(add(ptr, 0x14), shl(0x60, implementation)) mstore(add(ptr, 0x28), 0x5af43d82803e903d91602b57fd5bf30000000000000000000000000000000000) instance := create(0, ptr, 0x37) } require(instance != address(0), "ERC1167: create failed"); } /** * @dev Deploys and returns the address of a clone that mimics the behaviour of `implementation`. * * This function uses the create2 opcode and a `salt` to deterministically deploy * the clone. Using the same `implementation` and `salt` multiple time will revert, since * the clones cannot be deployed twice at the same address. */ function cloneDeterministic(address implementation, bytes32 salt) internal returns (address instance) { // solhint-disable-next-line no-inline-assembly assembly { let ptr := mload(0x40) mstore(ptr, 0x3d602d80600a3d3981f3363d3d373d3d3d363d73000000000000000000000000) mstore(add(ptr, 0x14), shl(0x60, implementation)) mstore(add(ptr, 0x28), 0x5af43d82803e903d91602b57fd5bf30000000000000000000000000000000000) instance := create2(0, ptr, 0x37, salt) } require(instance != address(0), "ERC1167: create2 failed"); } /** * @dev Computes the address of a clone deployed using {Clones-cloneDeterministic}. */ function predictDeterministicAddress(address implementation, bytes32 salt, address deployer) internal pure returns (address predicted) { // solhint-disable-next-line no-inline-assembly assembly { let ptr := mload(0x40) mstore(ptr, 0x3d602d80600a3d3981f3363d3d373d3d3d363d73000000000000000000000000) mstore(add(ptr, 0x14), shl(0x60, implementation)) mstore(add(ptr, 0x28), 0x5af43d82803e903d91602b57fd5bf3ff00000000000000000000000000000000) mstore(add(ptr, 0x38), shl(0x60, deployer)) mstore(add(ptr, 0x4c), salt) mstore(add(ptr, 0x6c), keccak256(ptr, 0x37)) predicted := keccak256(add(ptr, 0x37), 0x55) } } /** * @dev Computes the address of a clone deployed using {Clones-cloneDeterministic}. */ function predictDeterministicAddress(address implementation, bytes32 salt) internal view returns (address predicted) { return predictDeterministicAddress(implementation, salt, address(this)); } } /** * @dev Interface of the ERC20 standard as defined in the EIP. */ interface IERC20 { /** * @dev Returns the amount of tokens in existence. */ function totalSupply() external view returns (uint256); /** * @dev Returns the amount of tokens owned by `account`. */ function balanceOf(address account) external view returns (uint256); /** * @dev Moves `amount` tokens from the caller's account to `recipient`. * * Returns a boolean value indicating whether the operation succeeded. * * Emits a {Transfer} event. */ function transfer(address recipient, uint256 amount) external returns (bool); /** * @dev Returns the remaining number of tokens that `spender` will be * allowed to spend on behalf of `owner` through {transferFrom}. This is * zero by default. * * This value changes when {approve} or {transferFrom} are called. */ function allowance(address owner, address spender) external view returns (uint256); /** * @dev Sets `amount` as the allowance of `spender` over the caller's tokens. * * Returns a boolean value indicating whether the operation succeeded. * * IMPORTANT: Beware that changing an allowance with this method brings the risk * that someone may use both the old and the new allowance by unfortunate * transaction ordering. One possible solution to mitigate this race * condition is to first reduce the spender's allowance to 0 and set the * desired value afterwards: * https://github.com/ethereum/EIPs/issues/20#issuecomment-263524729 * * Emits an {Approval} event. */ function approve(address spender, uint256 amount) external returns (bool); /** * @dev Moves `amount` tokens from `sender` to `recipient` using the * allowance mechanism. `amount` is then deducted from the caller's * allowance. * * Returns a boolean value indicating whether the operation succeeded. * * Emits a {Transfer} event. */ function transferFrom(address sender, address recipient, uint256 amount) external returns (bool); /** * @dev Emitted when `value` tokens are moved from one account (`from`) to * another (`to`). * * Note that `value` may be zero. */ event Transfer(address indexed from, address indexed to, uint256 value); /** * @dev Emitted when the allowance of a `spender` for an `owner` is set by * a call to {approve}. `value` is the new allowance. */ event Approval(address indexed owner, address indexed spender, uint256 value); } /** * @dev Contract module that helps prevent reentrant calls to a function. * * Inheriting from `ReentrancyGuard` will make the {nonReentrant} modifier * available, which can be applied to functions to make sure there are no nested * (reentrant) calls to them. * * Note that because there is a single `nonReentrant` guard, functions marked as * `nonReentrant` may not call one another. This can be worked around by making * those functions `private`, and then adding `external` `nonReentrant` entry * points to them. * * TIP: If you would like to learn more about reentrancy and alternative ways * to protect against it, check out our blog post * https://blog.openzeppelin.com/reentrancy-after-istanbul/[Reentrancy After Istanbul]. */ abstract contract ReentrancyGuard { // Booleans are more expensive than uint256 or any type that takes up a full // word because each write operation emits an extra SLOAD to first read the // slot's contents, replace the bits taken up by the boolean, and then write // back. This is the compiler's defense against contract upgrades and // pointer aliasing, and it cannot be disabled. // The values being non-zero value makes deployment a bit more expensive, // but in exchange the refund on every call to nonReentrant will be lower in // amount. Since refunds are capped to a percentage of the total // transaction's gas, it is best to keep them low in cases like this one, to // increase the likelihood of the full refund coming into effect. uint256 private constant _NOT_ENTERED = 1; uint256 private constant _ENTERED = 2; uint256 private _status; constructor () { _status = _NOT_ENTERED; } /** * @dev Prevents a contract from calling itself, directly or indirectly. * Calling a `nonReentrant` function from another `nonReentrant` * function is not supported. It is possible to prevent this from happening * by making the `nonReentrant` function external, and make it call a * `private` function that does the actual work. */ modifier nonReentrant() { // On the first call to nonReentrant, _notEntered will be true require(_status != _ENTERED, "ReentrancyGuard: reentrant call"); // Any calls to nonReentrant after this point will fail _status = _ENTERED; _; // By storing the original value once again, a refund is triggered (see // https://eips.ethereum.org/EIPS/eip-2200) _status = _NOT_ENTERED; } } contract Governance is ReentrancyGuard { uint constant public governance_challenging_period = 10 days; uint constant public governance_freeze_period = 30 days; address public votingTokenAddress; address public governedContractAddress; mapping(address => uint) public balances; VotedValue[] public votedValues; mapping(string => VotedValue) public votedValuesMap; constructor(address _governedContractAddress, address _votingTokenAddress){ init(_governedContractAddress, _votingTokenAddress); } function init(address _governedContractAddress, address _votingTokenAddress) public { require(governedContractAddress == address(0), "governance already initialized"); governedContractAddress = _governedContractAddress; votingTokenAddress = _votingTokenAddress; } function addressBelongsToGovernance(address addr) public view returns (bool) { for (uint i = 0; i < votedValues.length; i++) if (address(votedValues[i]) == addr) return true; return false; } function isUntiedFromAllVotes(address addr) public view returns (bool) { for (uint i = 0; i < votedValues.length; i++) if (votedValues[i].hasVote(addr)) return false; return true; } function addVotedValue(string memory name, VotedValue votedValue) external { require(msg.sender == governedContractAddress, "not authorized"); votedValues.push(votedValue); votedValuesMap[name] = votedValue; } // deposit function deposit(uint amount) payable external { deposit(msg.sender, amount); } function deposit(address from, uint amount) nonReentrant payable public { require(from == msg.sender || addressBelongsToGovernance(msg.sender), "not allowed"); if (votingTokenAddress == address(0)) require(msg.value == amount, "wrong amount received"); else { require(msg.value == 0, "don't send ETH"); require(IERC20(votingTokenAddress).transferFrom(from, address(this), amount), "failed to pull gov deposit"); } balances[from] += amount; } // withdrawal functions function withdraw() external { withdraw(balances[msg.sender]); } function withdraw(uint amount) nonReentrant public { require(amount > 0, "zero withdrawal requested"); require(amount <= balances[msg.sender], "not enough balance"); require(isUntiedFromAllVotes(msg.sender), "some votes not removed yet"); balances[msg.sender] -= amount; if (votingTokenAddress == address(0)) payable(msg.sender).transfer(amount); else require(IERC20(votingTokenAddress).transfer(msg.sender, amount), "failed to withdraw gov deposit"); } } abstract contract VotedValue is ReentrancyGuard { Governance public governance; uint public challenging_period_start_ts; mapping(address => bool) public hasVote; constructor(Governance _governance){ governance = _governance; } function checkVoteChangeLock() view public { require(challenging_period_start_ts + governance.governance_challenging_period() + governance.governance_freeze_period() < block.timestamp, "you cannot change your vote yet"); } function checkChallengingPeriodExpiry() view public { require(block.timestamp > challenging_period_start_ts + governance.governance_challenging_period(), "challenging period not expired yet"); } } contract VotedValueUint is VotedValue { function(uint) external validationCallback; function(uint) external commitCallback; uint public leader; uint public current_value; mapping(address => uint) public choices; mapping(uint => uint) public votesByValue; mapping(uint => mapping(address => uint)) public votesByValueAddress; constructor() VotedValue(Governance(address(0))) {} // constructor(Governance _governance, uint initial_value, function(uint) external _validationCallback, function(uint) external _commitCallback) VotedValue(_governance) { // leader = initial_value; // current_value = initial_value; // validationCallback = _validationCallback; // commitCallback = _commitCallback; // } function init(Governance _governance, uint initial_value, function(uint) external _validationCallback, function(uint) external _commitCallback) external { require(address(governance) == address(0), "already initialized"); governance = _governance; leader = initial_value; current_value = initial_value; validationCallback = _validationCallback; commitCallback = _commitCallback; } function vote(uint value) nonReentrant external { _vote(value); } function voteAndDeposit(uint value, uint amount) nonReentrant payable external { governance.deposit{value: msg.value}(msg.sender, amount); _vote(value); } function _vote(uint value) private { validationCallback(value); uint prev_choice = choices[msg.sender]; bool hadVote = hasVote[msg.sender]; if (prev_choice == leader) checkVoteChangeLock(); // first, remove votes from the previous choice if (hadVote) removeVote(prev_choice); // then, add them to the new choice uint balance = governance.balances(msg.sender); require(balance > 0, "no balance"); votesByValue[value] += balance; votesByValueAddress[value][msg.sender] = balance; choices[msg.sender] = value; hasVote[msg.sender] = true; // check if the leader has just changed if (votesByValue[value] > votesByValue[leader]){ leader = value; challenging_period_start_ts = block.timestamp; } } function unvote() external { if (!hasVote[msg.sender]) return; uint prev_choice = choices[msg.sender]; if (prev_choice == leader) checkVoteChangeLock(); removeVote(prev_choice); delete choices[msg.sender]; delete hasVote[msg.sender]; } function removeVote(uint value) internal { votesByValue[value] -= votesByValueAddress[value][msg.sender]; votesByValueAddress[value][msg.sender] = 0; } function commit() nonReentrant external { require(leader != current_value, "already equal to leader"); checkChallengingPeriodExpiry(); current_value = leader; commitCallback(leader); } } contract VotedValueUintArray is VotedValue { function(uint[] memory) external validationCallback; function(uint[] memory) external commitCallback; uint[] public leader; uint[] public current_value; mapping(address => uint[]) public choices; mapping(bytes32 => uint) public votesByValue; mapping(bytes32 => mapping(address => uint)) public votesByValueAddress; constructor() VotedValue(Governance(address(0))) {} // constructor(Governance _governance, uint[] memory initial_value, function(uint[] memory) external _validationCallback, function(uint[] memory) external _commitCallback) VotedValue(_governance) { // leader = initial_value; // current_value = initial_value; // validationCallback = _validationCallback; // commitCallback = _commitCallback; // } function init(Governance _governance, uint[] memory initial_value, function(uint[] memory) external _validationCallback, function(uint[] memory) external _commitCallback) external { require(address(governance) == address(0), "already initialized"); governance = _governance; leader = initial_value; current_value = initial_value; validationCallback = _validationCallback; commitCallback = _commitCallback; } function equal(uint[] memory a1, uint[] memory a2) public pure returns (bool) { if (a1.length != a2.length) return false; for (uint i = 0; i < a1.length; i++) if (a1[i] != a2[i]) return false; return true; } function getKey(uint[] memory a) public pure returns (bytes32){ return keccak256(abi.encodePacked(a)); } function vote(uint[] memory value) nonReentrant external { _vote(value); } function voteAndDeposit(uint[] memory value, uint amount) nonReentrant payable external { governance.deposit{value: msg.value}(msg.sender, amount); _vote(value); } function _vote(uint[] memory value) private { validationCallback(value); uint[] storage prev_choice = choices[msg.sender]; bool hadVote = hasVote[msg.sender]; if (equal(prev_choice, leader)) checkVoteChangeLock(); // remove one's vote from the previous choice first if (hadVote) removeVote(prev_choice); // then, add it to the new choice, if any bytes32 key = getKey(value); uint balance = governance.balances(msg.sender); require(balance > 0, "no balance"); votesByValue[key] += balance; votesByValueAddress[key][msg.sender] = balance; choices[msg.sender] = value; hasVote[msg.sender] = true; // check if the leader has just changed if (votesByValue[key] > votesByValue[getKey(leader)]){ leader = value; challenging_period_start_ts = block.timestamp; } } function unvote() external { if (!hasVote[msg.sender]) return; uint[] storage prev_choice = choices[msg.sender]; if (equal(prev_choice, leader)) checkVoteChangeLock(); removeVote(prev_choice); delete choices[msg.sender]; delete hasVote[msg.sender]; } function removeVote(uint[] memory value) internal { bytes32 key = getKey(value); votesByValue[key] -= votesByValueAddress[key][msg.sender]; votesByValueAddress[key][msg.sender] = 0; } function commit() nonReentrant external { require(!equal(leader, current_value), "already equal to leader"); checkChallengingPeriodExpiry(); current_value = leader; commitCallback(leader); } } contract VotedValueAddress is VotedValue { function(address) external validationCallback; function(address) external commitCallback; address public leader; address public current_value; // mapping(who => value) mapping(address => address) public choices; // mapping(value => votes) mapping(address => uint) public votesByValue; // mapping(value => mapping(who => votes)) mapping(address => mapping(address => uint)) public votesByValueAddress; constructor() VotedValue(Governance(address(0))) {} // constructor(Governance _governance, address initial_value, function(address) external _validationCallback, function(address) external _commitCallback) VotedValue(_governance) { // leader = initial_value; // current_value = initial_value; // validationCallback = _validationCallback; // commitCallback = _commitCallback; // } function init(Governance _governance, address initial_value, function(address) external _validationCallback, function(address) external _commitCallback) external { require(address(governance) == address(0), "already initialized"); governance = _governance; leader = initial_value; current_value = initial_value; validationCallback = _validationCallback; commitCallback = _commitCallback; } function vote(address value) nonReentrant external { _vote(value); } function voteAndDeposit(address value, uint amount) nonReentrant payable external { governance.deposit{value: msg.value}(msg.sender, amount); _vote(value); } function _vote(address value) private { validationCallback(value); address prev_choice = choices[msg.sender]; bool hadVote = hasVote[msg.sender]; if (prev_choice == leader) checkVoteChangeLock(); // first, remove votes from the previous choice if (hadVote) removeVote(prev_choice); // then, add them to the new choice uint balance = governance.balances(msg.sender); require(balance > 0, "no balance"); votesByValue[value] += balance; votesByValueAddress[value][msg.sender] = balance; choices[msg.sender] = value; hasVote[msg.sender] = true; // check if the leader has just changed if (votesByValue[value] > votesByValue[leader]){ leader = value; challenging_period_start_ts = block.timestamp; } } function unvote() external { if (!hasVote[msg.sender]) return; address prev_choice = choices[msg.sender]; if (prev_choice == leader) checkVoteChangeLock(); removeVote(prev_choice); delete choices[msg.sender]; delete hasVote[msg.sender]; } function removeVote(address value) internal { votesByValue[value] -= votesByValueAddress[value][msg.sender]; votesByValueAddress[value][msg.sender] = 0; } function commit() nonReentrant external { require(leader != current_value, "already equal to leader"); checkChallengingPeriodExpiry(); current_value = leader; commitCallback(leader); } } contract VotedValueFactory { address public votedValueUintMaster; address public votedValueUintArrayMaster; address public votedValueAddressMaster; constructor(address _votedValueUintMaster, address _votedValueUintArrayMaster, address _votedValueAddressMaster) { votedValueUintMaster = _votedValueUintMaster; votedValueUintArrayMaster = _votedValueUintArrayMaster; votedValueAddressMaster = _votedValueAddressMaster; } function createVotedValueUint(Governance governance, uint initial_value, function(uint) external validationCallback, function(uint) external commitCallback) external returns (VotedValueUint) { VotedValueUint vv = VotedValueUint(Clones.clone(votedValueUintMaster)); vv.init(governance, initial_value, validationCallback, commitCallback); return vv; } function createVotedValueUintArray(Governance governance, uint[] memory initial_value, function(uint[] memory) external validationCallback, function(uint[] memory) external commitCallback) external returns (VotedValueUintArray) { VotedValueUintArray vv = VotedValueUintArray(Clones.clone(votedValueUintArrayMaster)); vv.init(governance, initial_value, validationCallback, commitCallback); return vv; } function createVotedValueAddress(Governance governance, address initial_value, function(address) external validationCallback, function(address) external commitCallback) external returns (VotedValueAddress) { VotedValueAddress vv = VotedValueAddress(Clones.clone(votedValueAddressMaster)); vv.init(governance, initial_value, validationCallback, commitCallback); return vv; } } contract GovernanceFactory { address public governanceMaster; constructor(address _governanceMaster) { governanceMaster = _governanceMaster; } function createGovernance(address governedContractAddress, address votingTokenAddress) external returns (Governance) { Governance governance = Governance(Clones.clone(governanceMaster)); governance.init(governedContractAddress, votingTokenAddress); return governance; } } // The purpose of the library is to separate some of the code out of the Export/Import contracts and keep their sizes under the 24KiB limit library CounterstakeLibrary { enum Side {no, yes} // small values (bool, uint32, ...) are grouped together in order to be packed efficiently struct Claim { uint amount; // int reward; address payable recipient_address; // 20 bytes, 12 bytes left uint32 txts; uint32 ts; address payable claimant_address; uint32 expiry_ts; uint16 period_number; Side current_outcome; bool is_large; bool withdrawn; bool finished; string sender_address; // string txid; string data; uint yes_stake; uint no_stake; // uint challenging_target; } struct Settings { address tokenAddress; uint16 ratio100;// = 100; uint16 counterstake_coef100;// = 150; uint32 min_tx_age; uint min_stake; uint[] challenging_periods;// = [12 hours, 3 days, 1 weeks, 30 days]; uint[] large_challenging_periods;// = [3 days, 1 weeks, 30 days]; uint large_threshold; } event NewClaim(uint indexed claim_num, address author_address, string sender_address, address recipient_address, string txid, uint32 txts, uint amount, int reward, uint stake, string data, uint32 expiry_ts); event NewChallenge(uint indexed claim_num, address author_address, uint stake, Side outcome, Side current_outcome, uint yes_stake, uint no_stake, uint32 expiry_ts, uint challenging_target); event FinishedClaim(uint indexed claim_num, Side outcome); struct ClaimRequest { string txid; uint32 txts; uint amount; int reward; uint stake; uint required_stake; address payable recipient_address; string sender_address; string data; } function claim( Settings storage settings, mapping(string => uint) storage claim_nums, mapping(uint => Claim) storage claims, mapping(uint => mapping(Side => mapping(address => uint))) storage stakes, uint claim_num, ClaimRequest memory req ) external { require(req.amount > 0, "0 claim"); require(req.stake >= req.required_stake, "the stake is too small"); require(block.timestamp >= req.txts + settings.min_tx_age, "too early"); if (req.recipient_address == address(0)) req.recipient_address = payable(msg.sender); if (req.reward < 0) require(req.recipient_address == payable(msg.sender), "the sender disallowed third-party claiming by setting a negative reward"); string memory claim_id = getClaimId(req.sender_address, req.recipient_address, req.txid, req.txts, req.amount, req.reward, req.data); require(claim_nums[claim_id] == 0, "this transfer has already been claimed"); bool is_large = (settings.large_threshold > 0 && req.stake >= settings.large_threshold); uint32 expiry_ts = uint32(block.timestamp + getChallengingPeriod(settings, 0, is_large)); // might wrap claim_nums[claim_id] = claim_num; // uint challenging_target = req.stake * settings.counterstake_coef100/100; claims[claim_num] = Claim({ amount: req.amount, // reward: req.reward, recipient_address: req.recipient_address, claimant_address: payable(msg.sender), sender_address: req.sender_address, // txid: req.txid, data: req.data, yes_stake: req.stake, no_stake: 0, current_outcome: Side.yes, is_large: is_large, period_number: 0, txts: req.txts, ts: uint32(block.timestamp), expiry_ts: expiry_ts, // challenging_target: req.stake * settings.counterstake_coef100/100, withdrawn: false, finished: false }); stakes[claim_num][Side.yes][msg.sender] = req.stake; emit NewClaim(claim_num, msg.sender, req.sender_address, req.recipient_address, req.txid, req.txts, req.amount, req.reward, req.stake, req.data, expiry_ts); // return claim_id; } function challenge( Settings storage settings, Claim storage c, mapping(uint => mapping(Side => mapping(address => uint))) storage stakes, uint claim_num, Side stake_on, uint stake ) external { require(block.timestamp < c.expiry_ts, "the challenging period has expired"); require(stake_on != c.current_outcome, "this outcome is already current"); uint excess; uint challenging_target = (c.current_outcome == Side.yes ? c.yes_stake : c.no_stake) * settings.counterstake_coef100/100; { // circumvent stack too deep uint stake_on_proposed_outcome = (stake_on == Side.yes ? c.yes_stake : c.no_stake) + stake; bool would_override_current_outcome = stake_on_proposed_outcome >= challenging_target; excess = would_override_current_outcome ? stake_on_proposed_outcome - challenging_target : 0; uint accepted_stake = stake - excess; if (stake_on == Side.yes) c.yes_stake += accepted_stake; else c.no_stake += accepted_stake; if (would_override_current_outcome){ c.period_number++; c.current_outcome = stake_on; c.expiry_ts = uint32(block.timestamp + getChallengingPeriod(settings, c.period_number, c.is_large)); challenging_target = challenging_target * settings.counterstake_coef100/100; } stakes[claim_num][stake_on][msg.sender] += accepted_stake; } emit NewChallenge(claim_num, msg.sender, stake, stake_on, c.current_outcome, c.yes_stake, c.no_stake, c.expiry_ts, challenging_target); if (excess > 0){ if (settings.tokenAddress == address(0)) payable(msg.sender).transfer(excess); else require(IERC20(settings.tokenAddress).transfer(msg.sender, excess), "failed to transfer the token"); } } function finish( Claim storage c, mapping(uint => mapping(Side => mapping(address => uint))) storage stakes, uint claim_num, address payable to_address ) external returns (bool, bool, uint) { require(block.timestamp > c.expiry_ts, "challenging period is still ongoing"); if (to_address == address(0)) to_address = payable(msg.sender); bool is_winning_claimant = (to_address == c.claimant_address && c.current_outcome == Side.yes); require(!(is_winning_claimant && c.withdrawn), "already withdrawn"); uint won_stake; { // circumvent stack too deep uint my_stake = stakes[claim_num][c.current_outcome][to_address]; require(my_stake > 0 || is_winning_claimant, "you are not the recipient and you didn't stake on the winning outcome or you have already withdrawn"); uint winning_stake = c.current_outcome == Side.yes ? c.yes_stake : c.no_stake; if (my_stake > 0) won_stake = (c.yes_stake + c.no_stake) * my_stake / winning_stake; } if (is_winning_claimant) c.withdrawn = true; bool finished; if (!c.finished){ finished = true; c.finished = true; // Side losing_outcome = outcome == Side.yes ? Side.no : Side.yes; // delete stakes[claim_id][losing_outcome]; // can't purge the stakes that will never be claimed emit FinishedClaim(claim_num, c.current_outcome); } delete stakes[claim_num][c.current_outcome][to_address]; return (finished, is_winning_claimant, won_stake); } function getChallengingPeriod(Settings storage settings, uint16 period_number, bool bLarge) public view returns (uint) { uint[] storage periods = bLarge ? settings.large_challenging_periods : settings.challenging_periods; if (period_number > periods.length - 1) period_number = uint16(periods.length - 1); return periods[period_number]; } function validateChallengingPeriods(uint[] memory periods) pure external { require(periods.length > 0, "empty periods"); uint prev_period = 0; for (uint i = 0; i < periods.length; i++) { require(periods[i] < 3 * 365 days, "some periods are longer than 3 years"); require(periods[i] >= prev_period, "subsequent periods cannot get shorter"); prev_period = periods[i]; } } function getClaimId(string memory sender_address, address recipient_address, string memory txid, uint32 txts, uint amount, int reward, string memory data) public pure returns (string memory){ return string(abi.encodePacked(sender_address, '_', toAsciiString(recipient_address), '_', txid, '_', uint2str(txts), '_', uint2str(amount), '_', int2str(reward), '_', data)); } function uint2str(uint256 _i) private pure returns (string memory) { if (_i == 0) return "0"; uint256 j = _i; uint256 length; while (j != 0) { length++; j /= 10; } bytes memory bstr = new bytes(length); uint256 k = length; j = _i; while (j != 0) { bstr[--k] = bytes1(uint8(48 + j % 10)); j /= 10; } return string(bstr); } function int2str(int256 _i) private pure returns (string memory) { require(_i < type(int).max, "int too large"); return _i >= 0 ? uint2str(uint(_i)) : string(abi.encodePacked('-', uint2str(uint(-_i)))); } function toAsciiString(address x) private pure returns (string memory) { bytes memory s = new bytes(40); for (uint i = 0; i < 20; i++) { bytes1 b = bytes1(uint8(uint(uint160(x)) / (2**(8*(19 - i))))); bytes1 hi = bytes1(uint8(b) / 16); bytes1 lo = bytes1(uint8(b) - 16 * uint8(hi)); s[2*i] = char(hi); s[2*i+1] = char(lo); } return string(s); } function char(bytes1 b) private pure returns (bytes1 c) { if (uint8(b) < 10) return bytes1(uint8(b) + 0x30); else return bytes1(uint8(b) + 0x57); } function isContract(address _addr) public view returns (bool){ uint32 size; assembly { size := extcodesize(_addr) } return (size > 0); } } interface CounterstakeReceiver { function onReceivedFromClaim(uint claim_num, uint net_claimed_amount, uint won_stake, string memory sender_address, address claim_recipient_address, string memory data) external; } abstract contract Counterstake is ReentrancyGuard { event NewClaim(uint indexed claim_num, address author_address, string sender_address, address recipient_address, string txid, uint32 txts, uint amount, int reward, uint stake, string data, uint32 expiry_ts); event NewChallenge(uint indexed claim_num, address author_address, uint stake, CounterstakeLibrary.Side outcome, CounterstakeLibrary.Side current_outcome, uint yes_stake, uint no_stake, uint32 expiry_ts, uint challenging_target); event FinishedClaim(uint indexed claim_num, CounterstakeLibrary.Side outcome); Governance public governance; CounterstakeLibrary.Settings public settings; uint64 public last_claim_num; uint64[] public ongoing_claim_nums; mapping(uint => uint) public num2index; mapping(string => uint) public claim_nums; mapping(uint => CounterstakeLibrary.Claim) private claims; mapping(uint => mapping(CounterstakeLibrary.Side => mapping(address => uint))) public stakes; function getClaim(uint claim_num) external view returns (CounterstakeLibrary.Claim memory) { return claims[claim_num]; } function getClaim(string memory claim_id) external view returns (CounterstakeLibrary.Claim memory) { return claims[claim_nums[claim_id]]; } function getOngoingClaimNums() external view returns (uint64[] memory) { return ongoing_claim_nums; } constructor (address _tokenAddr, uint16 _counterstake_coef100, uint16 _ratio100, uint _large_threshold, uint[] memory _challenging_periods, uint[] memory _large_challenging_periods) { initCounterstake(_tokenAddr, _counterstake_coef100, _ratio100, _large_threshold, _challenging_periods, _large_challenging_periods); } function initCounterstake(address _tokenAddr, uint16 _counterstake_coef100, uint16 _ratio100, uint _large_threshold, uint[] memory _challenging_periods, uint[] memory _large_challenging_periods) public { require(address(governance) == address(0), "already initialized"); settings = CounterstakeLibrary.Settings({ tokenAddress: _tokenAddr, counterstake_coef100: _counterstake_coef100 > 100 ? _counterstake_coef100 : 150, ratio100: _ratio100 > 0 ? _ratio100 : 100, min_stake: 0, min_tx_age: 0, challenging_periods: _challenging_periods, large_challenging_periods: _large_challenging_periods, large_threshold: _large_threshold }); } /* modifier onlyETH(){ require(settings.tokenAddress == address(0), "ETH only"); _; } modifier onlyERC20(){ require(settings.tokenAddress != address(0), "ERC20 only"); _; }*/ modifier onlyVotedValueContract(){ require(governance.addressBelongsToGovernance(msg.sender), "not from voted value contract"); _; } // would be happy to call this from the constructor but unfortunately `this` is not set at that time yet function setupGovernance(GovernanceFactory governanceFactory, VotedValueFactory votedValueFactory) virtual public { require(address(governance) == address(0), "already initialized"); governance = governanceFactory.createGovernance(address(this), settings.tokenAddress); governance.addVotedValue("ratio100", votedValueFactory.createVotedValueUint(governance, settings.ratio100, this.validateRatio, this.setRatio)); governance.addVotedValue("counterstake_coef100", votedValueFactory.createVotedValueUint(governance, settings.counterstake_coef100, this.validateCounterstakeCoef, this.setCounterstakeCoef)); governance.addVotedValue("min_stake", votedValueFactory.createVotedValueUint(governance, settings.min_stake, this.validateMinStake, this.setMinStake)); governance.addVotedValue("min_tx_age", votedValueFactory.createVotedValueUint(governance, settings.min_tx_age, this.validateMinTxAge, this.setMinTxAge)); governance.addVotedValue("large_threshold", votedValueFactory.createVotedValueUint(governance, settings.large_threshold, this.validateLargeThreshold, this.setLargeThreshold)); governance.addVotedValue("challenging_periods", votedValueFactory.createVotedValueUintArray(governance, settings.challenging_periods, this.validateChallengingPeriods, this.setChallengingPeriods)); governance.addVotedValue("large_challenging_periods", votedValueFactory.createVotedValueUintArray(governance, settings.large_challenging_periods, this.validateChallengingPeriods, this.setLargeChallengingPeriods)); } function validateRatio(uint _ratio100) pure external { require(_ratio100 > 0 && _ratio100 < 64000, "bad ratio"); } function setRatio(uint _ratio100) onlyVotedValueContract external { settings.ratio100 = uint16(_ratio100); } function validateCounterstakeCoef(uint _counterstake_coef100) pure external { require(_counterstake_coef100 > 100 && _counterstake_coef100 < 64000, "bad counterstake coef"); } function setCounterstakeCoef(uint _counterstake_coef100) onlyVotedValueContract external { settings.counterstake_coef100 = uint16(_counterstake_coef100); } function validateMinStake(uint _min_stake) pure external { // anything goes } function setMinStake(uint _min_stake) onlyVotedValueContract external { settings.min_stake = _min_stake; } function validateMinTxAge(uint _min_tx_age) pure external { require(_min_tx_age < 4 weeks, "min tx age too large"); } function setMinTxAge(uint _min_tx_age) onlyVotedValueContract external { settings.min_tx_age = uint32(_min_tx_age); } function validateLargeThreshold(uint _large_threshold) pure external { // anything goes } function setLargeThreshold(uint _large_threshold) onlyVotedValueContract external { settings.large_threshold = _large_threshold; } function validateChallengingPeriods(uint[] memory periods) pure external { CounterstakeLibrary.validateChallengingPeriods(periods); } function setChallengingPeriods(uint[] memory _challenging_periods) onlyVotedValueContract external { settings.challenging_periods = _challenging_periods; } function setLargeChallengingPeriods(uint[] memory _large_challenging_periods) onlyVotedValueContract external { settings.large_challenging_periods = _large_challenging_periods; } function getChallengingPeriod(uint16 period_number, bool bLarge) external view returns (uint) { return CounterstakeLibrary.getChallengingPeriod(settings, period_number, bLarge); } function getRequiredStake(uint amount) public view virtual returns (uint); function getMissingStake(uint claim_num, CounterstakeLibrary.Side stake_on) external view returns (uint) { CounterstakeLibrary.Claim storage c = claims[claim_num]; require(c.yes_stake > 0, "no such claim"); uint current_stake = (stake_on == CounterstakeLibrary.Side.yes) ? c.yes_stake : c.no_stake; return (c.current_outcome == CounterstakeLibrary.Side.yes ? c.yes_stake : c.no_stake) * settings.counterstake_coef100/100 - current_stake; } function claim(string memory txid, uint32 txts, uint amount, int reward, uint stake, string memory sender_address, address payable recipient_address, string memory data) nonReentrant payable external { if (recipient_address == address(0)) recipient_address = payable(msg.sender); bool bThirdPartyClaiming = (recipient_address != payable(msg.sender) && reward >= 0); uint paid_amount; if (bThirdPartyClaiming) { require(amount > uint(reward), "reward too large"); paid_amount = amount - uint(reward); } receiveMoneyInClaim(stake, paid_amount); uint required_stake = getRequiredStake(amount); CounterstakeLibrary.ClaimRequest memory req = CounterstakeLibrary.ClaimRequest({ txid: txid, txts: txts, amount: amount, reward: reward, stake: stake, required_stake: required_stake, recipient_address: recipient_address, sender_address: sender_address, data: data }); last_claim_num++; ongoing_claim_nums.push(last_claim_num); num2index[last_claim_num] = ongoing_claim_nums.length - 1; CounterstakeLibrary.claim(settings, claim_nums, claims, stakes, last_claim_num, req); if (bThirdPartyClaiming){ sendToClaimRecipient(recipient_address, paid_amount); notifyPaymentRecipient(recipient_address, paid_amount, 0, last_claim_num); } } function challenge(string calldata claim_id, CounterstakeLibrary.Side stake_on, uint stake) payable external { challenge(claim_nums[claim_id], stake_on, stake); } function challenge(uint claim_num, CounterstakeLibrary.Side stake_on, uint stake) nonReentrant payable public { receiveStakeAsset(stake); CounterstakeLibrary.Claim storage c = claims[claim_num]; require(c.amount > 0, "no such claim"); CounterstakeLibrary.challenge(settings, c, stakes, claim_num, stake_on, stake); } function withdraw(string memory claim_id) external { withdraw(claim_nums[claim_id], payable(0)); } function withdraw(uint claim_num) external { withdraw(claim_num, payable(0)); } function withdraw(string memory claim_id, address payable to_address) external { withdraw(claim_nums[claim_id], to_address); } function withdraw(uint claim_num, address payable to_address) nonReentrant public { if (to_address == address(0)) to_address = payable(msg.sender); require(claim_num > 0, "no such claim num"); CounterstakeLibrary.Claim storage c = claims[claim_num]; require(c.amount > 0, "no such claim"); (bool finished, bool is_winning_claimant, uint won_stake) = CounterstakeLibrary.finish(c, stakes, claim_num, to_address); if (finished){ uint index = num2index[claim_num]; uint last_index = ongoing_claim_nums.length - 1; if (index != last_index){ // move the last element in place of our removed element require(index < last_index, "BUG index after last"); uint64 claim_num_of_last_element = ongoing_claim_nums[last_index]; num2index[claim_num_of_last_element] = index; ongoing_claim_nums[index] = claim_num_of_last_element; } ongoing_claim_nums.pop(); delete num2index[claim_num]; } uint claimed_amount_to_be_paid = is_winning_claimant ? c.amount : 0; sendWithdrawals(to_address, claimed_amount_to_be_paid, won_stake); notifyPaymentRecipient(to_address, claimed_amount_to_be_paid, won_stake, claim_num); } function notifyPaymentRecipient(address payable payment_recipient_address, uint net_claimed_amount, uint won_stake, uint claim_num) private { if (CounterstakeLibrary.isContract(payment_recipient_address)){ CounterstakeLibrary.Claim storage c = claims[claim_num]; // CounterstakeReceiver(payment_recipient_address).onReceivedFromClaim(claim_num, is_winning_claimant ? claimed_amount : 0, won_stake); (bool res, ) = payment_recipient_address.call(abi.encodeWithSignature("onReceivedFromClaim(uint256,uint256,uint256,string,address,string)", claim_num, net_claimed_amount, won_stake, c.sender_address, c.recipient_address, c.data)); if (!res){ // ignore } } } function receiveStakeAsset(uint stake_asset_amount) internal { if (settings.tokenAddress == address(0)) require(msg.value == stake_asset_amount, "wrong amount received"); else { require(msg.value == 0, "don't send ETH"); require(IERC20(settings.tokenAddress).transferFrom(msg.sender, address(this), stake_asset_amount), "failed to pull the token"); } } function sendWithdrawals(address payable to_address, uint claimed_amount_to_be_paid, uint won_stake) internal virtual; function sendToClaimRecipient(address payable to_address, uint paid_amount) internal virtual; function receiveMoneyInClaim(uint stake, uint paid_amount) internal virtual; } interface IOracle { // returns a fraction num/den function getPrice(string memory base, string memory quote) external view returns (uint num, uint den); } /** * @dev Implementation of the {IERC20} interface. * * This implementation is agnostic to the way tokens are created. This means * that a supply mechanism has to be added in a derived contract using {_mint}. * For a generic mechanism see {ERC20PresetMinterPauser}. * * TIP: For a detailed writeup see our guide * https://forum.zeppelin.solutions/t/how-to-implement-erc20-supply-mechanisms/226[How * to implement supply mechanisms]. * * We have followed general OpenZeppelin guidelines: functions revert instead * of returning `false` on failure. This behavior is nonetheless conventional * and does not conflict with the expectations of ERC20 applications. * * Additionally, an {Approval} event is emitted on calls to {transferFrom}. * This allows applications to reconstruct the allowance for all accounts just * by listening to said events. Other implementations of the EIP may not emit * these events, as it isn't required by the specification. * * Finally, the non-standard {decreaseAllowance} and {increaseAllowance} * functions have been added to mitigate the well-known issues around setting * allowances. See {IERC20-approve}. */ contract ERC20 is Context, IERC20 { mapping (address => uint256) private _balances; mapping (address => mapping (address => uint256)) private _allowances; uint256 private _totalSupply; string public name; string public symbol; /** * @dev Sets the values for {name} and {symbol}. * * The defaut value of {decimals} is 18. To select a different value for * {decimals} you should overload it. * * All three of these values are immutable: they can only be set once during * construction. */ constructor (string memory name_, string memory symbol_) { name = name_; symbol = symbol_; } /** * @dev Returns the number of decimals used to get its user representation. * For example, if `decimals` equals `2`, a balance of `505` tokens should * be displayed to a user as `5,05` (`505 / 10 ** 2`). * * Tokens usually opt for a value of 18, imitating the relationship between * Ether and Wei. This is the value {ERC20} uses, unless this function is * overloaded; * * NOTE: This information is only used for _display_ purposes: it in * no way affects any of the arithmetic of the contract, including * {IERC20-balanceOf} and {IERC20-transfer}. */ function decimals() public view virtual returns (uint8) { return 18; } /** * @dev See {IERC20-totalSupply}. */ function totalSupply() public view virtual override returns (uint256) { return _totalSupply; } /** * @dev See {IERC20-balanceOf}. */ function balanceOf(address account) public view virtual override returns (uint256) { return _balances[account]; } /** * @dev See {IERC20-transfer}. * * Requirements: * * - `recipient` cannot be the zero address. * - the caller must have a balance of at least `amount`. */ function transfer(address recipient, uint256 amount) public virtual override returns (bool) { _transfer(_msgSender(), recipient, amount); return true; } /** * @dev See {IERC20-allowance}. */ function allowance(address owner, address spender) public view virtual override returns (uint256) { return _allowances[owner][spender]; } /** * @dev See {IERC20-approve}. * * Requirements: * * - `spender` cannot be the zero address. */ function approve(address spender, uint256 amount) public virtual override returns (bool) { _approve(_msgSender(), spender, amount); return true; } /** * @dev See {IERC20-transferFrom}. * * Emits an {Approval} event indicating the updated allowance. This is not * required by the EIP. See the note at the beginning of {ERC20}. * * Requirements: * * - `sender` and `recipient` cannot be the zero address. * - `sender` must have a balance of at least `amount`. * - the caller must have allowance for ``sender``'s tokens of at least * `amount`. */ function transferFrom(address sender, address recipient, uint256 amount) public virtual override returns (bool) { _transfer(sender, recipient, amount); uint256 currentAllowance = _allowances[sender][_msgSender()]; require(currentAllowance >= amount, "ERC20: transfer amount exceeds allowance"); _approve(sender, _msgSender(), currentAllowance - amount); return true; } /** * @dev Atomically increases the allowance granted to `spender` by the caller. * * This is an alternative to {approve} that can be used as a mitigation for * problems described in {IERC20-approve}. * * Emits an {Approval} event indicating the updated allowance. * * Requirements: * * - `spender` cannot be the zero address. */ function increaseAllowance(address spender, uint256 addedValue) public virtual returns (bool) { _approve(_msgSender(), spender, _allowances[_msgSender()][spender] + addedValue); return true; } /** * @dev Atomically decreases the allowance granted to `spender` by the caller. * * This is an alternative to {approve} that can be used as a mitigation for * problems described in {IERC20-approve}. * * Emits an {Approval} event indicating the updated allowance. * * Requirements: * * - `spender` cannot be the zero address. * - `spender` must have allowance for the caller of at least * `subtractedValue`. */ function decreaseAllowance(address spender, uint256 subtractedValue) public virtual returns (bool) { uint256 currentAllowance = _allowances[_msgSender()][spender]; require(currentAllowance >= subtractedValue, "ERC20: decreased allowance below zero"); _approve(_msgSender(), spender, currentAllowance - subtractedValue); return true; } /** * @dev Moves tokens `amount` from `sender` to `recipient`. * * This is internal function is equivalent to {transfer}, and can be used to * e.g. implement automatic token fees, slashing mechanisms, etc. * * Emits a {Transfer} event. * * Requirements: * * - `sender` cannot be the zero address. * - `recipient` cannot be the zero address. * - `sender` must have a balance of at least `amount`. */ function _transfer(address sender, address recipient, uint256 amount) internal virtual { require(sender != address(0), "ERC20: transfer from the zero address"); require(recipient != address(0), "ERC20: transfer to the zero address"); uint256 senderBalance = _balances[sender]; require(senderBalance >= amount, "ERC20: transfer amount exceeds balance"); _balances[sender] = senderBalance - amount; _balances[recipient] += amount; emit Transfer(sender, recipient, amount); } /** @dev Creates `amount` tokens and assigns them to `account`, increasing * the total supply. * * Emits a {Transfer} event with `from` set to the zero address. * * Requirements: * * - `to` cannot be the zero address. */ function _mint(address account, uint256 amount) internal virtual { require(account != address(0), "ERC20: mint to the zero address"); _totalSupply += amount; _balances[account] += amount; emit Transfer(address(0), account, amount); } /** * @dev Destroys `amount` tokens from `account`, reducing the * total supply. * * Emits a {Transfer} event with `to` set to the zero address. * * Requirements: * * - `account` cannot be the zero address. * - `account` must have at least `amount` tokens. */ function _burn(address account, uint256 amount) internal virtual { require(account != address(0), "ERC20: burn from the zero address"); uint256 accountBalance = _balances[account]; require(accountBalance >= amount, "ERC20: burn amount exceeds balance"); _balances[account] = accountBalance - amount; _totalSupply -= amount; emit Transfer(account, address(0), amount); } /** * @dev Sets `amount` as the allowance of `spender` over the `owner` s tokens. * * This internal function is equivalent to `approve`, and can be used to * e.g. set automatic allowances for certain subsystems, etc. * * Emits an {Approval} event. * * Requirements: * * - `owner` cannot be the zero address. * - `spender` cannot be the zero address. */ function _approve(address owner, address spender, uint256 amount) internal virtual { require(owner != address(0), "ERC20: approve from the zero address"); require(spender != address(0), "ERC20: approve to the zero address"); _allowances[owner][spender] = amount; emit Approval(owner, spender, amount); } } interface IERC20WithSymbol is IERC20 { function symbol() external view returns (string memory); } contract Import is ERC20, Counterstake { event NewRepatriation(address sender_address, uint amount, uint reward, string home_address, string data); address public oracleAddress; // min price of imported asset in terms of stake asset, to protect against malicious oracles // The price is multiplied by 1e20 uint public min_price20; string public home_network; string public home_asset; bytes32 private constant base_hash = keccak256(abi.encodePacked("base")); bytes32 private constant zx_hash = keccak256(abi.encodePacked("0x0000000000000000000000000000000000000000")); constructor (string memory _home_network, string memory _home_asset, string memory __name, string memory __symbol, address stakeTokenAddr, address oracleAddr, uint16 _counterstake_coef100, uint16 _ratio100, uint _large_threshold, uint[] memory _challenging_periods, uint[] memory _large_challenging_periods) Counterstake(stakeTokenAddr, _counterstake_coef100, _ratio100, _large_threshold, _challenging_periods, _large_challenging_periods) ERC20(__name, __symbol) { initImport(_home_network, _home_asset, __name, __symbol, oracleAddr); } function initImport(string memory _home_network, string memory _home_asset, string memory __name, string memory __symbol, address oracleAddr) public { require(address(governance) == address(0), "already initialized"); oracleAddress = oracleAddr; home_network = _home_network; home_asset = _home_asset; name = __name; symbol = __symbol; } function setupGovernance(GovernanceFactory governanceFactory, VotedValueFactory votedValueFactory) override virtual public { super.setupGovernance(governanceFactory, votedValueFactory); governance.addVotedValue("oracleAddress", votedValueFactory.createVotedValueAddress(governance, oracleAddress, this.validateOracle, this.setOracle)); governance.addVotedValue("min_price20", votedValueFactory.createVotedValueUint(governance, min_price20, this.validateMinPrice, this.setMinPrice)); } function getOraclePrice(address oracleAddr) view private returns (uint, uint) { bytes32 home_asset_hash = keccak256(abi.encodePacked(home_asset)); return IOracle(oracleAddr).getPrice( home_asset_hash == base_hash || home_asset_hash == zx_hash ? home_network : home_asset, settings.tokenAddress == address(0) ? "_NATIVE_" : IERC20WithSymbol(settings.tokenAddress).symbol() ); } function validateOracle(address oracleAddr) view external { require(CounterstakeLibrary.isContract(oracleAddr), "bad oracle"); (uint num, uint den) = getOraclePrice(oracleAddr); require(num > 0 || den > 0, "no price from oracle"); } function setOracle(address oracleAddr) onlyVotedValueContract external { oracleAddress = oracleAddr; } function validateMinPrice(uint _min_price20) pure external { // anything goes } function setMinPrice(uint _min_price20) onlyVotedValueContract external { min_price20 = _min_price20; } // repatriate function transferToHomeChain(string memory home_address, string memory data, uint amount, uint reward) external { _burn(msg.sender, amount); emit NewRepatriation(msg.sender, amount, reward, home_address, data); } function getRequiredStake(uint amount) public view override returns (uint) { (uint num, uint den) = getOraclePrice(oracleAddress); require(num > 0, "price num must be positive"); require(den > 0, "price den must be positive"); uint stake_in_image_asset = amount * settings.ratio100 / 100; return Math.max(Math.max(stake_in_image_asset * num / den, stake_in_image_asset * min_price20 / 1e20), settings.min_stake); } function sendWithdrawals(address payable to_address, uint paid_claimed_amount, uint won_stake) internal override { if (paid_claimed_amount > 0){ _mint(to_address, paid_claimed_amount); } if (settings.tokenAddress == address(0)) to_address.transfer(won_stake); else require(IERC20(settings.tokenAddress).transfer(to_address, won_stake), "failed to send the won stake"); } function receiveMoneyInClaim(uint stake, uint paid_amount) internal override { if (paid_amount > 0) _burn(msg.sender, paid_amount); receiveStakeAsset(stake); } function sendToClaimRecipient(address payable to_address, uint paid_amount) internal override { _mint(to_address, paid_amount); } }
[{"inputs":[{"internalType":"string","name":"_home_network","type":"string"},{"internalType":"string","name":"_home_asset","type":"string"},{"internalType":"string","name":"__name","type":"string"},{"internalType":"string","name":"__symbol","type":"string"},{"internalType":"address","name":"stakeTokenAddr","type":"address"},{"internalType":"address","name":"oracleAddr","type":"address"},{"internalType":"uint16","name":"_counterstake_coef100","type":"uint16"},{"internalType":"uint16","name":"_ratio100","type":"uint16"},{"internalType":"uint256","name":"_large_threshold","type":"uint256"},{"internalType":"uint256[]","name":"_challenging_periods","type":"uint256[]"},{"internalType":"uint256[]","name":"_large_challenging_periods","type":"uint256[]"}],"stateMutability":"nonpayable","type":"constructor"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"owner","type":"address"},{"indexed":true,"internalType":"address","name":"spender","type":"address"},{"indexed":false,"internalType":"uint256","name":"value","type":"uint256"}],"name":"Approval","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"uint256","name":"claim_num","type":"uint256"},{"indexed":false,"internalType":"enum CounterstakeLibrary.Side","name":"outcome","type":"uint8"}],"name":"FinishedClaim","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"uint256","name":"claim_num","type":"uint256"},{"indexed":false,"internalType":"address","name":"author_address","type":"address"},{"indexed":false,"internalType":"uint256","name":"stake","type":"uint256"},{"indexed":false,"internalType":"enum CounterstakeLibrary.Side","name":"outcome","type":"uint8"},{"indexed":false,"internalType":"enum CounterstakeLibrary.Side","name":"current_outcome","type":"uint8"},{"indexed":false,"internalType":"uint256","name":"yes_stake","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"no_stake","type":"uint256"},{"indexed":false,"internalType":"uint32","name":"expiry_ts","type":"uint32"},{"indexed":false,"internalType":"uint256","name":"challenging_target","type":"uint256"}],"name":"NewChallenge","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"uint256","name":"claim_num","type":"uint256"},{"indexed":false,"internalType":"address","name":"author_address","type":"address"},{"indexed":false,"internalType":"string","name":"sender_address","type":"string"},{"indexed":false,"internalType":"address","name":"recipient_address","type":"address"},{"indexed":false,"internalType":"string","name":"txid","type":"string"},{"indexed":false,"internalType":"uint32","name":"txts","type":"uint32"},{"indexed":false,"internalType":"uint256","name":"amount","type":"uint256"},{"indexed":false,"internalType":"int256","name":"reward","type":"int256"},{"indexed":false,"internalType":"uint256","name":"stake","type":"uint256"},{"indexed":false,"internalType":"string","name":"data","type":"string"},{"indexed":false,"internalType":"uint32","name":"expiry_ts","type":"uint32"}],"name":"NewClaim","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"sender_address","type":"address"},{"indexed":false,"internalType":"uint256","name":"amount","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"reward","type":"uint256"},{"indexed":false,"internalType":"string","name":"home_address","type":"string"},{"indexed":false,"internalType":"string","name":"data","type":"string"}],"name":"NewRepatriation","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"from","type":"address"},{"indexed":true,"internalType":"address","name":"to","type":"address"},{"indexed":false,"internalType":"uint256","name":"value","type":"uint256"}],"name":"Transfer","type":"event"},{"inputs":[{"internalType":"address","name":"owner","type":"address"},{"internalType":"address","name":"spender","type":"address"}],"name":"allowance","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"spender","type":"address"},{"internalType":"uint256","name":"amount","type":"uint256"}],"name":"approve","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"account","type":"address"}],"name":"balanceOf","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"claim_num","type":"uint256"},{"internalType":"enum CounterstakeLibrary.Side","name":"stake_on","type":"uint8"},{"internalType":"uint256","name":"stake","type":"uint256"}],"name":"challenge","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[{"internalType":"string","name":"claim_id","type":"string"},{"internalType":"enum CounterstakeLibrary.Side","name":"stake_on","type":"uint8"},{"internalType":"uint256","name":"stake","type":"uint256"}],"name":"challenge","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[{"internalType":"string","name":"txid","type":"string"},{"internalType":"uint32","name":"txts","type":"uint32"},{"internalType":"uint256","name":"amount","type":"uint256"},{"internalType":"int256","name":"reward","type":"int256"},{"internalType":"uint256","name":"stake","type":"uint256"},{"internalType":"string","name":"sender_address","type":"string"},{"internalType":"address payable","name":"recipient_address","type":"address"},{"internalType":"string","name":"data","type":"string"}],"name":"claim","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[{"internalType":"string","name":"","type":"string"}],"name":"claim_nums","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"decimals","outputs":[{"internalType":"uint8","name":"","type":"uint8"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"spender","type":"address"},{"internalType":"uint256","name":"subtractedValue","type":"uint256"}],"name":"decreaseAllowance","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint16","name":"period_number","type":"uint16"},{"internalType":"bool","name":"bLarge","type":"bool"}],"name":"getChallengingPeriod","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"string","name":"claim_id","type":"string"}],"name":"getClaim","outputs":[{"components":[{"internalType":"uint256","name":"amount","type":"uint256"},{"internalType":"address payable","name":"recipient_address","type":"address"},{"internalType":"uint32","name":"txts","type":"uint32"},{"internalType":"uint32","name":"ts","type":"uint32"},{"internalType":"address payable","name":"claimant_address","type":"address"},{"internalType":"uint32","name":"expiry_ts","type":"uint32"},{"internalType":"uint16","name":"period_number","type":"uint16"},{"internalType":"enum CounterstakeLibrary.Side","name":"current_outcome","type":"uint8"},{"internalType":"bool","name":"is_large","type":"bool"},{"internalType":"bool","name":"withdrawn","type":"bool"},{"internalType":"bool","name":"finished","type":"bool"},{"internalType":"string","name":"sender_address","type":"string"},{"internalType":"string","name":"data","type":"string"},{"internalType":"uint256","name":"yes_stake","type":"uint256"},{"internalType":"uint256","name":"no_stake","type":"uint256"}],"internalType":"struct CounterstakeLibrary.Claim","name":"","type":"tuple"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"claim_num","type":"uint256"}],"name":"getClaim","outputs":[{"components":[{"internalType":"uint256","name":"amount","type":"uint256"},{"internalType":"address payable","name":"recipient_address","type":"address"},{"internalType":"uint32","name":"txts","type":"uint32"},{"internalType":"uint32","name":"ts","type":"uint32"},{"internalType":"address payable","name":"claimant_address","type":"address"},{"internalType":"uint32","name":"expiry_ts","type":"uint32"},{"internalType":"uint16","name":"period_number","type":"uint16"},{"internalType":"enum CounterstakeLibrary.Side","name":"current_outcome","type":"uint8"},{"internalType":"bool","name":"is_large","type":"bool"},{"internalType":"bool","name":"withdrawn","type":"bool"},{"internalType":"bool","name":"finished","type":"bool"},{"internalType":"string","name":"sender_address","type":"string"},{"internalType":"string","name":"data","type":"string"},{"internalType":"uint256","name":"yes_stake","type":"uint256"},{"internalType":"uint256","name":"no_stake","type":"uint256"}],"internalType":"struct CounterstakeLibrary.Claim","name":"","type":"tuple"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"claim_num","type":"uint256"},{"internalType":"enum CounterstakeLibrary.Side","name":"stake_on","type":"uint8"}],"name":"getMissingStake","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"getOngoingClaimNums","outputs":[{"internalType":"uint64[]","name":"","type":"uint64[]"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"amount","type":"uint256"}],"name":"getRequiredStake","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"governance","outputs":[{"internalType":"contract Governance","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"home_asset","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"home_network","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"spender","type":"address"},{"internalType":"uint256","name":"addedValue","type":"uint256"}],"name":"increaseAllowance","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_tokenAddr","type":"address"},{"internalType":"uint16","name":"_counterstake_coef100","type":"uint16"},{"internalType":"uint16","name":"_ratio100","type":"uint16"},{"internalType":"uint256","name":"_large_threshold","type":"uint256"},{"internalType":"uint256[]","name":"_challenging_periods","type":"uint256[]"},{"internalType":"uint256[]","name":"_large_challenging_periods","type":"uint256[]"}],"name":"initCounterstake","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"string","name":"_home_network","type":"string"},{"internalType":"string","name":"_home_asset","type":"string"},{"internalType":"string","name":"__name","type":"string"},{"internalType":"string","name":"__symbol","type":"string"},{"internalType":"address","name":"oracleAddr","type":"address"}],"name":"initImport","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"last_claim_num","outputs":[{"internalType":"uint64","name":"","type":"uint64"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"min_price20","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"name","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"}],"name":"num2index","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"}],"name":"ongoing_claim_nums","outputs":[{"internalType":"uint64","name":"","type":"uint64"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"oracleAddress","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256[]","name":"_challenging_periods","type":"uint256[]"}],"name":"setChallengingPeriods","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_counterstake_coef100","type":"uint256"}],"name":"setCounterstakeCoef","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256[]","name":"_large_challenging_periods","type":"uint256[]"}],"name":"setLargeChallengingPeriods","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_large_threshold","type":"uint256"}],"name":"setLargeThreshold","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_min_price20","type":"uint256"}],"name":"setMinPrice","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_min_stake","type":"uint256"}],"name":"setMinStake","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_min_tx_age","type":"uint256"}],"name":"setMinTxAge","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"oracleAddr","type":"address"}],"name":"setOracle","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_ratio100","type":"uint256"}],"name":"setRatio","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"settings","outputs":[{"internalType":"address","name":"tokenAddress","type":"address"},{"internalType":"uint16","name":"ratio100","type":"uint16"},{"internalType":"uint16","name":"counterstake_coef100","type":"uint16"},{"internalType":"uint32","name":"min_tx_age","type":"uint32"},{"internalType":"uint256","name":"min_stake","type":"uint256"},{"internalType":"uint256","name":"large_threshold","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"contract GovernanceFactory","name":"governanceFactory","type":"address"},{"internalType":"contract VotedValueFactory","name":"votedValueFactory","type":"address"}],"name":"setupGovernance","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"},{"internalType":"enum CounterstakeLibrary.Side","name":"","type":"uint8"},{"internalType":"address","name":"","type":"address"}],"name":"stakes","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"symbol","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"totalSupply","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"recipient","type":"address"},{"internalType":"uint256","name":"amount","type":"uint256"}],"name":"transfer","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"sender","type":"address"},{"internalType":"address","name":"recipient","type":"address"},{"internalType":"uint256","name":"amount","type":"uint256"}],"name":"transferFrom","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"string","name":"home_address","type":"string"},{"internalType":"string","name":"data","type":"string"},{"internalType":"uint256","name":"amount","type":"uint256"},{"internalType":"uint256","name":"reward","type":"uint256"}],"name":"transferToHomeChain","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256[]","name":"periods","type":"uint256[]"}],"name":"validateChallengingPeriods","outputs":[],"stateMutability":"pure","type":"function"},{"inputs":[{"internalType":"uint256","name":"_counterstake_coef100","type":"uint256"}],"name":"validateCounterstakeCoef","outputs":[],"stateMutability":"pure","type":"function"},{"inputs":[{"internalType":"uint256","name":"_large_threshold","type":"uint256"}],"name":"validateLargeThreshold","outputs":[],"stateMutability":"pure","type":"function"},{"inputs":[{"internalType":"uint256","name":"_min_price20","type":"uint256"}],"name":"validateMinPrice","outputs":[],"stateMutability":"pure","type":"function"},{"inputs":[{"internalType":"uint256","name":"_min_stake","type":"uint256"}],"name":"validateMinStake","outputs":[],"stateMutability":"pure","type":"function"},{"inputs":[{"internalType":"uint256","name":"_min_tx_age","type":"uint256"}],"name":"validateMinTxAge","outputs":[],"stateMutability":"pure","type":"function"},{"inputs":[{"internalType":"address","name":"oracleAddr","type":"address"}],"name":"validateOracle","outputs":[],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"_ratio100","type":"uint256"}],"name":"validateRatio","outputs":[],"stateMutability":"pure","type":"function"},{"inputs":[{"internalType":"uint256","name":"claim_num","type":"uint256"},{"internalType":"address payable","name":"to_address","type":"address"}],"name":"withdraw","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"claim_num","type":"uint256"}],"name":"withdraw","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"string","name":"claim_id","type":"string"}],"name":"withdraw","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"string","name":"claim_id","type":"string"},{"internalType":"address payable","name":"to_address","type":"address"}],"name":"withdraw","outputs":[],"stateMutability":"nonpayable","type":"function"}]
Loading...
Loading
Loading...
Loading
OVERVIEW
Obyte is a distributed ledger based on directed acyclic graph (DAG). Unlike centralized ledgers and blockchains, access to Obyte ledger is decentralized, disintermediated, free (as in freedom), equal, and open.Multichain Portfolio | 30 Chains
Chain | Token | Portfolio % | Price | Amount | Value |
---|---|---|---|---|---|
ETH | 100.00% | $7.05 | 4.8557 | $34.23 |
Loading...
Loading
[ Download: CSV Export ]
[ 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.