Source Code
Overview
S Balance
More Info
ContractCreator
Transaction Hash |
Method
|
Block
|
From
|
To
|
|||||
---|---|---|---|---|---|---|---|---|---|
Loading...
Loading
Contract Name:
RingADuck
Compiler Version
v0.8.19+commit.7dd6d404
Optimization Enabled:
Yes with 200 runs
Other Settings:
default evmVersion
Contract Source Code (Solidity Standard Json-Input format)
// SPDX-License-Identifier: MIT pragma solidity ^0.8.19; import "../interfaces/IGame.sol"; import "../interfaces/IUniversalCashier.sol"; import "@openzeppelin/contracts-upgradeable/security/PausableUpgradeable.sol"; import "@openzeppelin/contracts-upgradeable/access/Ownable2StepUpgradeable.sol"; import "@openzeppelin/contracts-upgradeable/security/ReentrancyGuardUpgradeable.sol"; contract RingADuck is IGame, PausableUpgradeable, Ownable2StepUpgradeable, ReentrancyGuardUpgradeable { // References address public proxyManager; IUniversalCashier public cashier; // Game configuration uint256 public constant PLAY_COST = 1 ether; // 1 S per game uint256 public constant RINGS_PER_GAME = 3; // 3 rings per game // MEGA DUCK configuration uint256 public constant MEGA_MULTIPLIER = 100; // 100x multiplier uint256 public constant MEGA_DURATION = 10 minutes; uint256 public constant MEGA_CHANCE = 1; // 0.01% chance (1 in 10000) uint256 public lastMegaDuckTime; bool public isMegaDuckActive; // Duck prize configuration struct DuckPrize { uint256 value; // Prize value in wei uint256 probability; // Probability in basis points (100 = 1%) } // Duck types and their probabilities DuckPrize[] public duckPrizes; // Bonus features uint256 public constant BONUS_ROUND_CHANCE = 500; // 5% chance uint256 public constant BONUS_MULTIPLIER = 2; // 2x multiplier uint256 public constant TRIPLE_MATCH_BONUS = 5000; // 50% bonus for matching 3 // Game state struct Game { address player; bool completed; uint256 timestamp; uint256[] ringValues; // Values of ducks hit bool bonusRound; // Whether this was a bonus round uint256 totalPayout; } mapping(uint256 => Game) public games; uint256 public currentGameId; // House stats uint256 public totalGamesPlayed; uint256 public totalPaid; uint256 public houseProfits; // Add prepaid plays tracking mapping(address => uint256) public prepaidPlays; // Bulk purchase tiers struct DiscountTier { uint256 minPlays; uint256 discount; // In basis points (100 = 1%) } DiscountTier[] public discountTiers; // Constants uint256 public constant MIN_PREPAID = 1; // Minimum 1 play uint256 public constant MAX_PREPAID = 100; // Maximum 100 plays // Events event GameStarted(uint256 indexed gameId, address indexed player, bool isBonusRound); event RingThrown(uint256 indexed gameId, uint256 ringNumber, uint256 value); event GameCompleted( uint256 indexed gameId, address indexed player, uint256[] values, bool bonusRound, uint256 totalPayout, uint256 houseFee ); event MegaDuckActivated(uint256 timestamp, uint256 duration); event MegaDuckDeactivated(uint256 timestamp); event MegaDuckCaught(uint256 indexed gameId, address indexed player, uint256 amount); event PrepaidPlaysAdded(address indexed player, uint256 plays, uint256 cost); event PrepaidPlayUsed(address indexed player, uint256 remainingPlays); function initialize(address _proxyManager, address _cashier) public initializer { __Pausable_init(); __Ownable2Step_init(); __ReentrancyGuard_init(); proxyManager = _proxyManager; cashier = IUniversalCashier(_cashier); // Initialize duck prize configuration // Common ducks (75% chance total) duckPrizes.push(DuckPrize({value: 0.2 ether, probability: 3500})); // 35% - 0.2 S duckPrizes.push(DuckPrize({value: 0.3 ether, probability: 2000})); // 20% - 0.3 S duckPrizes.push(DuckPrize({value: 0.5 ether, probability: 2000})); // 20% - 0.5 S // Uncommon ducks (20% chance total) duckPrizes.push(DuckPrize({value: 0.8 ether, probability: 1200})); // 12% - 0.8 S duckPrizes.push(DuckPrize({value: 1.2 ether, probability: 800})); // 8% - 1.2 S // Rare ducks (5% chance total) duckPrizes.push(DuckPrize({value: 2 ether, probability: 300})); // 3% - 2.0 S duckPrizes.push(DuckPrize({value: 5 ether, probability: 180})); // 1.8% - 5.0 S duckPrizes.push(DuckPrize({value: 10 ether, probability: 20})); // 0.2% - 10.0 S // Setup new discount tiers discountTiers.push(DiscountTier({minPlays: 5, discount: 300})); // 3% off for 5+ plays discountTiers.push(DiscountTier({minPlays: 20, discount: 500})); // 5% off for 20+ plays discountTiers.push(DiscountTier({minPlays: 50, discount: 700})); // 7% off for 50+ plays discountTiers.push(DiscountTier({minPlays: 100, discount: 1000})); // 10% off for 100 plays } function getDiscountForPlays(uint256 plays) public view returns (uint256) { uint256 discount = 0; for (uint256 i = 0; i < discountTiers.length; i++) { if (plays >= discountTiers[i].minPlays) { discount = discountTiers[i].discount; } } return discount; } function buyPrepaidPlays(uint256 numberOfPlays) external { require(numberOfPlays >= MIN_PREPAID, "Min 1 play"); require(numberOfPlays <= MAX_PREPAID, "Max 100 plays"); uint256 totalCost = PLAY_COST * numberOfPlays; uint256 discount = getDiscountForPlays(numberOfPlays); // Apply discount if (discount > 0) { totalCost = totalCost * (10000 - discount) / 10000; } // Debit through cashier require(cashier.debitPlayer(msg.sender, totalCost), "Payment failed"); prepaidPlays[msg.sender] += numberOfPlays; emit PrepaidPlaysAdded(msg.sender, numberOfPlays, totalCost); } function play(address player, uint256 amount) external payable override { require(msg.sender == proxyManager, "Only proxy manager"); // Check if using prepaid plays if (amount == 0 && prepaidPlays[player] > 0) { prepaidPlays[player]--; emit PrepaidPlayUsed(player, prepaidPlays[player]); } else { require(amount == PLAY_COST, "Invalid play cost"); } // Check and update MEGA DUCK status _updateMegaDuckStatus(); // Determine if this is a bonus round (5% chance) bool isBonusRound = _random(currentGameId, "BONUS") <= BONUS_ROUND_CHANCE; currentGameId++; games[currentGameId] = Game({ player: player, completed: false, timestamp: block.timestamp, ringValues: new uint256[](RINGS_PER_GAME), bonusRound: isBonusRound, totalPayout: 0 }); totalGamesPlayed++; emit GameStarted(currentGameId, player, isBonusRound); } function _updateMegaDuckStatus() internal { // Check if MEGA DUCK should be activated if (!isMegaDuckActive && block.timestamp >= lastMegaDuckTime + 1 days && _random(block.timestamp, "MEGA_TRIGGER") <= MEGA_CHANCE) { isMegaDuckActive = true; lastMegaDuckTime = block.timestamp; emit MegaDuckActivated(block.timestamp, MEGA_DURATION); } // Check if MEGA DUCK should be deactivated if (isMegaDuckActive && block.timestamp >= lastMegaDuckTime + MEGA_DURATION) { isMegaDuckActive = false; emit MegaDuckDeactivated(block.timestamp); } } function throwRings(uint256 gameId, uint256[] calldata duckIndices) external onlyOwner { Game storage game = games[gameId]; require(!game.completed, "Game already completed"); require(duckIndices.length == RINGS_PER_GAME, "Invalid number of rings"); uint256 totalWin = 0; bool caughtMegaDuck = false; // Process each ring throw for (uint256 i = 0; i < RINGS_PER_GAME; i++) { require(duckIndices[i] < duckPrizes.length, "Invalid duck index"); uint256 value = duckPrizes[duckIndices[i]].value; // Check for MEGA DUCK catch (0.01% chance when active) if (isMegaDuckActive && _random(block.timestamp + i, "MEGA_CATCH") <= MEGA_CHANCE) { value *= MEGA_MULTIPLIER; caughtMegaDuck = true; } game.ringValues[i] = value; totalWin += value; emit RingThrown(gameId, i, value); } // Check for triple match bonus if (_checkTripleMatch(game.ringValues)) { totalWin += (totalWin * TRIPLE_MATCH_BONUS) / 10000; } // Apply bonus round multiplier if applicable if (game.bonusRound) { totalWin *= BONUS_MULTIPLIER; } game.totalPayout = totalWin; game.completed = true; // Update stats totalPaid += totalWin; houseProfits += (PLAY_COST - totalWin); // Send payout if won if (totalWin > 0) { cashier.creditPlayer{value: totalWin}(game.player, totalWin); } if (caughtMegaDuck) { emit MegaDuckCaught(gameId, game.player, totalWin); } emit GameCompleted( gameId, game.player, game.ringValues, game.bonusRound, totalWin, 0 // No house fee ); } function _checkTripleMatch(uint256[] memory values) internal pure returns (bool) { return values[0] == values[1] && values[1] == values[2]; } function _random(uint256 seed, string memory tag) internal view returns (uint256) { return uint256(keccak256(abi.encodePacked( block.timestamp, block.prevrandao, seed, tag ))) % 10000; // Returns 0-9999 for probability calculations } // View functions function getGameDetails(uint256 gameId) external view returns ( address player, bool completed, uint256 timestamp, uint256[] memory ringValues, bool bonusRound, uint256 totalPayout ) { Game storage game = games[gameId]; return ( game.player, game.completed, game.timestamp, game.ringValues, game.bonusRound, game.totalPayout ); } function getHouseStats() external view returns ( uint256 _totalGames, uint256 _totalPaid, uint256 _houseProfits ) { return (totalGamesPlayed, totalPaid, houseProfits); } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.9.0) (access/Ownable2Step.sol) pragma solidity ^0.8.0; import "./OwnableUpgradeable.sol"; import {Initializable} from "../proxy/utils/Initializable.sol"; /** * @dev Contract module which provides 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} and {acceptOwnership}. * * This module is used through inheritance. It will make available all functions * from parent (Ownable). */ abstract contract Ownable2StepUpgradeable is Initializable, OwnableUpgradeable { address private _pendingOwner; event OwnershipTransferStarted(address indexed previousOwner, address indexed newOwner); function __Ownable2Step_init() internal onlyInitializing { __Ownable_init_unchained(); } function __Ownable2Step_init_unchained() internal onlyInitializing { } /** * @dev Returns the address of the pending owner. */ function pendingOwner() public view virtual returns (address) { return _pendingOwner; } /** * @dev Starts the ownership transfer of the contract to a new account. Replaces the pending transfer if there is one. * Can only be called by the current owner. */ function transferOwnership(address newOwner) public virtual override onlyOwner { _pendingOwner = newOwner; emit OwnershipTransferStarted(owner(), newOwner); } /** * @dev Transfers ownership of the contract to a new account (`newOwner`) and deletes any pending owner. * Internal function without access restriction. */ function _transferOwnership(address newOwner) internal virtual override { delete _pendingOwner; super._transferOwnership(newOwner); } /** * @dev The new owner accepts the ownership transfer. */ function acceptOwnership() public virtual { address sender = _msgSender(); require(pendingOwner() == sender, "Ownable2Step: caller is not the new owner"); _transferOwnership(sender); } /** * @dev This empty reserved space is put in place to allow future versions to add new * variables without shifting down storage in the inheritance chain. * See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps */ uint256[49] private __gap; }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.9.0) (access/Ownable.sol) pragma solidity ^0.8.0; import "../utils/ContextUpgradeable.sol"; import {Initializable} from "../proxy/utils/Initializable.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 OwnableUpgradeable is Initializable, ContextUpgradeable { address private _owner; event OwnershipTransferred(address indexed previousOwner, address indexed newOwner); /** * @dev Initializes the contract setting the deployer as the initial owner. */ function __Ownable_init() internal onlyInitializing { __Ownable_init_unchained(); } function __Ownable_init_unchained() internal onlyInitializing { _transferOwnership(_msgSender()); } /** * @dev Throws if called by any account other than the owner. */ modifier onlyOwner() { _checkOwner(); _; } /** * @dev Returns the address of the current owner. */ function owner() public view virtual returns (address) { return _owner; } /** * @dev Throws if the sender is not the owner. */ function _checkOwner() internal view virtual { require(owner() == _msgSender(), "Ownable: caller is not the owner"); } /** * @dev Leaves the contract without owner. It will not be possible to call * `onlyOwner` functions. Can only be called by the current owner. * * NOTE: Renouncing ownership will leave the contract without an owner, * thereby disabling 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); } /** * @dev This empty reserved space is put in place to allow future versions to add new * variables without shifting down storage in the inheritance chain. * See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps */ uint256[49] private __gap; }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.9.0) (proxy/utils/Initializable.sol) pragma solidity ^0.8.2; import "../../utils/AddressUpgradeable.sol"; /** * @dev This is a base contract to aid in writing upgradeable contracts, or any kind of contract that will be deployed * behind a proxy. Since proxied contracts do not make use of a constructor, it's common to move constructor logic to an * external initializer function, usually called `initialize`. It then becomes necessary to protect this initializer * function so it can only be called once. The {initializer} modifier provided by this contract will have this effect. * * The initialization functions use a version number. Once a version number is used, it is consumed and cannot be * reused. This mechanism prevents re-execution of each "step" but allows the creation of new initialization steps in * case an upgrade adds a module that needs to be initialized. * * For example: * * [.hljs-theme-light.nopadding] * ```solidity * contract MyToken is ERC20Upgradeable { * function initialize() initializer public { * __ERC20_init("MyToken", "MTK"); * } * } * * contract MyTokenV2 is MyToken, ERC20PermitUpgradeable { * function initializeV2() reinitializer(2) public { * __ERC20Permit_init("MyToken"); * } * } * ``` * * TIP: To avoid leaving the proxy in an uninitialized state, the initializer function should be called as early as * possible by providing the encoded function call as the `_data` argument to {ERC1967Proxy-constructor}. * * CAUTION: When used with inheritance, manual care must be taken to not invoke a parent initializer twice, or to ensure * that all initializers are idempotent. This is not verified automatically as constructors are by Solidity. * * [CAUTION] * ==== * Avoid leaving a contract uninitialized. * * An uninitialized contract can be taken over by an attacker. This applies to both a proxy and its implementation * contract, which may impact the proxy. To prevent the implementation contract from being used, you should invoke * the {_disableInitializers} function in the constructor to automatically lock it when it is deployed: * * [.hljs-theme-light.nopadding] * ``` * /// @custom:oz-upgrades-unsafe-allow constructor * constructor() { * _disableInitializers(); * } * ``` * ==== */ abstract contract Initializable { /** * @dev Indicates that the contract has been initialized. * @custom:oz-retyped-from bool */ uint8 private _initialized; /** * @dev Indicates that the contract is in the process of being initialized. */ bool private _initializing; /** * @dev Triggered when the contract has been initialized or reinitialized. */ event Initialized(uint8 version); /** * @dev A modifier that defines a protected initializer function that can be invoked at most once. In its scope, * `onlyInitializing` functions can be used to initialize parent contracts. * * Similar to `reinitializer(1)`, except that functions marked with `initializer` can be nested in the context of a * constructor. * * Emits an {Initialized} event. */ modifier initializer() { bool isTopLevelCall = !_initializing; require( (isTopLevelCall && _initialized < 1) || (!AddressUpgradeable.isContract(address(this)) && _initialized == 1), "Initializable: contract is already initialized" ); _initialized = 1; if (isTopLevelCall) { _initializing = true; } _; if (isTopLevelCall) { _initializing = false; emit Initialized(1); } } /** * @dev A modifier that defines a protected reinitializer function that can be invoked at most once, and only if the * contract hasn't been initialized to a greater version before. In its scope, `onlyInitializing` functions can be * used to initialize parent contracts. * * A reinitializer may be used after the original initialization step. This is essential to configure modules that * are added through upgrades and that require initialization. * * When `version` is 1, this modifier is similar to `initializer`, except that functions marked with `reinitializer` * cannot be nested. If one is invoked in the context of another, execution will revert. * * Note that versions can jump in increments greater than 1; this implies that if multiple reinitializers coexist in * a contract, executing them in the right order is up to the developer or operator. * * WARNING: setting the version to 255 will prevent any future reinitialization. * * Emits an {Initialized} event. */ modifier reinitializer(uint8 version) { require(!_initializing && _initialized < version, "Initializable: contract is already initialized"); _initialized = version; _initializing = true; _; _initializing = false; emit Initialized(version); } /** * @dev Modifier to protect an initialization function so that it can only be invoked by functions with the * {initializer} and {reinitializer} modifiers, directly or indirectly. */ modifier onlyInitializing() { require(_initializing, "Initializable: contract is not initializing"); _; } /** * @dev Locks the contract, preventing any future reinitialization. This cannot be part of an initializer call. * Calling this in the constructor of a contract will prevent that contract from being initialized or reinitialized * to any version. It is recommended to use this to lock implementation contracts that are designed to be called * through proxies. * * Emits an {Initialized} event the first time it is successfully executed. */ function _disableInitializers() internal virtual { require(!_initializing, "Initializable: contract is initializing"); if (_initialized != type(uint8).max) { _initialized = type(uint8).max; emit Initialized(type(uint8).max); } } /** * @dev Returns the highest version that has been initialized. See {reinitializer}. */ function _getInitializedVersion() internal view returns (uint8) { return _initialized; } /** * @dev Returns `true` if the contract is currently initializing. See {onlyInitializing}. */ function _isInitializing() internal view returns (bool) { return _initializing; } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.7.0) (security/Pausable.sol) pragma solidity ^0.8.0; import "../utils/ContextUpgradeable.sol"; import {Initializable} from "../proxy/utils/Initializable.sol"; /** * @dev Contract module which allows children to implement an emergency stop * mechanism that can be triggered by an authorized account. * * This module is used through inheritance. It will make available the * modifiers `whenNotPaused` and `whenPaused`, which can be applied to * the functions of your contract. Note that they will not be pausable by * simply including this module, only once the modifiers are put in place. */ abstract contract PausableUpgradeable is Initializable, ContextUpgradeable { /** * @dev Emitted when the pause is triggered by `account`. */ event Paused(address account); /** * @dev Emitted when the pause is lifted by `account`. */ event Unpaused(address account); bool private _paused; /** * @dev Initializes the contract in unpaused state. */ function __Pausable_init() internal onlyInitializing { __Pausable_init_unchained(); } function __Pausable_init_unchained() internal onlyInitializing { _paused = false; } /** * @dev Modifier to make a function callable only when the contract is not paused. * * Requirements: * * - The contract must not be paused. */ modifier whenNotPaused() { _requireNotPaused(); _; } /** * @dev Modifier to make a function callable only when the contract is paused. * * Requirements: * * - The contract must be paused. */ modifier whenPaused() { _requirePaused(); _; } /** * @dev Returns true if the contract is paused, and false otherwise. */ function paused() public view virtual returns (bool) { return _paused; } /** * @dev Throws if the contract is paused. */ function _requireNotPaused() internal view virtual { require(!paused(), "Pausable: paused"); } /** * @dev Throws if the contract is not paused. */ function _requirePaused() internal view virtual { require(paused(), "Pausable: not paused"); } /** * @dev Triggers stopped state. * * Requirements: * * - The contract must not be paused. */ function _pause() internal virtual whenNotPaused { _paused = true; emit Paused(_msgSender()); } /** * @dev Returns to normal state. * * Requirements: * * - The contract must be paused. */ function _unpause() internal virtual whenPaused { _paused = false; emit Unpaused(_msgSender()); } /** * @dev This empty reserved space is put in place to allow future versions to add new * variables without shifting down storage in the inheritance chain. * See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps */ uint256[49] private __gap; }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.9.0) (security/ReentrancyGuard.sol) pragma solidity ^0.8.0; import {Initializable} from "../proxy/utils/Initializable.sol"; /** * @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 ReentrancyGuardUpgradeable is Initializable { // 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; function __ReentrancyGuard_init() internal onlyInitializing { __ReentrancyGuard_init_unchained(); } function __ReentrancyGuard_init_unchained() internal onlyInitializing { _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 making it call a * `private` function that does the actual work. */ modifier nonReentrant() { _nonReentrantBefore(); _; _nonReentrantAfter(); } function _nonReentrantBefore() private { // On the first call to nonReentrant, _status will be _NOT_ENTERED require(_status != _ENTERED, "ReentrancyGuard: reentrant call"); // Any calls to nonReentrant after this point will fail _status = _ENTERED; } function _nonReentrantAfter() private { // By storing the original value once again, a refund is triggered (see // https://eips.ethereum.org/EIPS/eip-2200) _status = _NOT_ENTERED; } /** * @dev Returns true if the reentrancy guard is currently set to "entered", which indicates there is a * `nonReentrant` function in the call stack. */ function _reentrancyGuardEntered() internal view returns (bool) { return _status == _ENTERED; } /** * @dev This empty reserved space is put in place to allow future versions to add new * variables without shifting down storage in the inheritance chain. * See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps */ uint256[49] private __gap; }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.9.0) (utils/Address.sol) pragma solidity ^0.8.1; /** * @dev Collection of functions related to the address type */ library AddressUpgradeable { /** * @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 * * Furthermore, `isContract` will also return true if the target contract within * the same transaction is already scheduled for destruction by `SELFDESTRUCT`, * which only has an effect at the end of a transaction. * ==== * * [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://consensys.net/diligence/blog/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.8.0/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 functionCallWithValue(target, data, 0, "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"); (bool success, bytes memory returndata) = target.call{value: value}(data); return verifyCallResultFromTarget(target, 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) { (bool success, bytes memory returndata) = target.staticcall(data); return verifyCallResultFromTarget(target, 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) { (bool success, bytes memory returndata) = target.delegatecall(data); return verifyCallResultFromTarget(target, success, returndata, errorMessage); } /** * @dev Tool to verify that a low level call to smart-contract was successful, and revert (either by bubbling * the revert reason or using the provided one) in case of unsuccessful call or if target was not a contract. * * _Available since v4.8._ */ function verifyCallResultFromTarget( address target, bool success, bytes memory returndata, string memory errorMessage ) internal view returns (bytes memory) { if (success) { if (returndata.length == 0) { // only check isContract if the call was successful and the return data is empty // otherwise we already know that it was a contract require(isContract(target), "Address: call to non-contract"); } return returndata; } else { _revert(returndata, errorMessage); } } /** * @dev Tool to verify that a low level call was successful, and revert if it wasn't, either by bubbling the * revert reason or 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 { _revert(returndata, errorMessage); } } function _revert(bytes memory returndata, string memory errorMessage) private pure { // 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 /// @solidity memory-safe-assembly assembly { let returndata_size := mload(returndata) revert(add(32, returndata), returndata_size) } } else { revert(errorMessage); } } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.9.4) (utils/Context.sol) pragma solidity ^0.8.0; import {Initializable} from "../proxy/utils/Initializable.sol"; /** * @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 ContextUpgradeable is Initializable { function __Context_init() internal onlyInitializing { } function __Context_init_unchained() internal onlyInitializing { } function _msgSender() internal view virtual returns (address) { return msg.sender; } function _msgData() internal view virtual returns (bytes calldata) { return msg.data; } function _contextSuffixLength() internal view virtual returns (uint256) { return 0; } /** * @dev This empty reserved space is put in place to allow future versions to add new * variables without shifting down storage in the inheritance chain. * See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps */ uint256[50] private __gap; }
// SPDX-License-Identifier: MIT pragma solidity ^0.8.19; interface IGame { function play(address player, uint256 amount) external payable; }
// SPDX-License-Identifier: MIT pragma solidity ^0.8.19; interface IUniversalCashier { function deposit() external payable; function withdraw(uint256 amount) external; function debitPlayer(address player, uint256 amount) external returns (bool); function creditPlayer(address player, uint256 amount) external payable; function getBalance(address player) external view returns (uint256); function authorizeGame(address game, bool status) external; function setTreasury(address _treasury) external; }
{ "optimizer": { "enabled": true, "runs": 200 }, "outputSelection": { "*": { "*": [ "evm.bytecode", "evm.deployedBytecode", "devdoc", "userdoc", "metadata", "abi" ] } }, "libraries": {} }
[{"anonymous":false,"inputs":[{"indexed":true,"internalType":"uint256","name":"gameId","type":"uint256"},{"indexed":true,"internalType":"address","name":"player","type":"address"},{"indexed":false,"internalType":"uint256[]","name":"values","type":"uint256[]"},{"indexed":false,"internalType":"bool","name":"bonusRound","type":"bool"},{"indexed":false,"internalType":"uint256","name":"totalPayout","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"houseFee","type":"uint256"}],"name":"GameCompleted","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"uint256","name":"gameId","type":"uint256"},{"indexed":true,"internalType":"address","name":"player","type":"address"},{"indexed":false,"internalType":"bool","name":"isBonusRound","type":"bool"}],"name":"GameStarted","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint8","name":"version","type":"uint8"}],"name":"Initialized","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"timestamp","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"duration","type":"uint256"}],"name":"MegaDuckActivated","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"uint256","name":"gameId","type":"uint256"},{"indexed":true,"internalType":"address","name":"player","type":"address"},{"indexed":false,"internalType":"uint256","name":"amount","type":"uint256"}],"name":"MegaDuckCaught","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"timestamp","type":"uint256"}],"name":"MegaDuckDeactivated","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"previousOwner","type":"address"},{"indexed":true,"internalType":"address","name":"newOwner","type":"address"}],"name":"OwnershipTransferStarted","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"previousOwner","type":"address"},{"indexed":true,"internalType":"address","name":"newOwner","type":"address"}],"name":"OwnershipTransferred","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"account","type":"address"}],"name":"Paused","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"player","type":"address"},{"indexed":false,"internalType":"uint256","name":"remainingPlays","type":"uint256"}],"name":"PrepaidPlayUsed","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"player","type":"address"},{"indexed":false,"internalType":"uint256","name":"plays","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"cost","type":"uint256"}],"name":"PrepaidPlaysAdded","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"uint256","name":"gameId","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"ringNumber","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"value","type":"uint256"}],"name":"RingThrown","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"account","type":"address"}],"name":"Unpaused","type":"event"},{"inputs":[],"name":"BONUS_MULTIPLIER","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"BONUS_ROUND_CHANCE","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"MAX_PREPAID","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"MEGA_CHANCE","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"MEGA_DURATION","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"MEGA_MULTIPLIER","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"MIN_PREPAID","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"PLAY_COST","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"RINGS_PER_GAME","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"TRIPLE_MATCH_BONUS","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"acceptOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"numberOfPlays","type":"uint256"}],"name":"buyPrepaidPlays","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"cashier","outputs":[{"internalType":"contract IUniversalCashier","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"currentGameId","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"}],"name":"discountTiers","outputs":[{"internalType":"uint256","name":"minPlays","type":"uint256"},{"internalType":"uint256","name":"discount","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"}],"name":"duckPrizes","outputs":[{"internalType":"uint256","name":"value","type":"uint256"},{"internalType":"uint256","name":"probability","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"}],"name":"games","outputs":[{"internalType":"address","name":"player","type":"address"},{"internalType":"bool","name":"completed","type":"bool"},{"internalType":"uint256","name":"timestamp","type":"uint256"},{"internalType":"bool","name":"bonusRound","type":"bool"},{"internalType":"uint256","name":"totalPayout","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"plays","type":"uint256"}],"name":"getDiscountForPlays","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"gameId","type":"uint256"}],"name":"getGameDetails","outputs":[{"internalType":"address","name":"player","type":"address"},{"internalType":"bool","name":"completed","type":"bool"},{"internalType":"uint256","name":"timestamp","type":"uint256"},{"internalType":"uint256[]","name":"ringValues","type":"uint256[]"},{"internalType":"bool","name":"bonusRound","type":"bool"},{"internalType":"uint256","name":"totalPayout","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"getHouseStats","outputs":[{"internalType":"uint256","name":"_totalGames","type":"uint256"},{"internalType":"uint256","name":"_totalPaid","type":"uint256"},{"internalType":"uint256","name":"_houseProfits","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"houseProfits","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_proxyManager","type":"address"},{"internalType":"address","name":"_cashier","type":"address"}],"name":"initialize","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"isMegaDuckActive","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"lastMegaDuckTime","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"owner","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"paused","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"pendingOwner","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"player","type":"address"},{"internalType":"uint256","name":"amount","type":"uint256"}],"name":"play","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"prepaidPlays","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"proxyManager","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"renounceOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"gameId","type":"uint256"},{"internalType":"uint256[]","name":"duckIndices","type":"uint256[]"}],"name":"throwRings","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"totalGamesPlayed","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"totalPaid","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"newOwner","type":"address"}],"name":"transferOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"}]
Contract Creation Code
608060405234801561001057600080fd5b50611e32806100206000396000f3fe6080604052600436106102045760003560e01c806379ba509711610118578063dc8f84e2116100a0578063ed740e971161006f578063ed740e97146105f4578063f0e10c0d14610614578063f2fde38b14610627578063fa84dc2f14610647578063fda5eff5146105c857600080fd5b8063dc8f84e214610593578063e30c3978146105aa578063e366c058146105c8578063e7b0f666146105dd57600080fd5b80639a46f89f116100e75780639a46f89f14610349578063b216a6c61461052c578063d053a25714610548578063d09bd9721461055e578063d4cffc4d1461057357600080fd5b806379ba5097146104b05780638aa28550146104c55780638da5cb5b146104da57806396b4ca701461050c57600080fd5b8063406c7b8d1161019b5780635c4921291161016a5780635c492129146104215780635c975abb146104385780636003bdaf146104505780636e032e3d14610485578063715018a61461049b57600080fd5b8063406c7b8d14610396578063485cc955146103b6578063496621d7146103d6578063536a3ddc1461040a57600080fd5b80631b31abda116101d75780631b31abda146103175780632731f9ef146103495780632dd713411461035e5780632e9830ce1461037457600080fd5b8063036ebf5d1461020957806304336f8f146102385780630fc345c91461025c578063117a5b901461028a575b600080fd5b34801561021557600080fd5b5060fe546102239060ff1681565b60405190151581526020015b60405180910390f35b34801561024457600080fd5b5061024e61138881565b60405190815260200161022f565b34801561026857600080fd5b5061024e610277366004611a89565b6101056020526000908152604090205481565b34801561029657600080fd5b506102e36102a5366004611aa4565b6101006020526000908152604090208054600182015460038301546004909301546001600160a01b0383169360ff600160a01b909404841693169085565b604080516001600160a01b03909616865293151560208601529284019190915215156060830152608082015260a00161022f565b34801561032357600080fd5b50610337610332366004611aa4565b610667565b60405161022f96959493929190611abd565b34801561035557600080fd5b5061024e600181565b34801561036a57600080fd5b5061024e60fd5481565b34801561038057600080fd5b5061039461038f366004611aa4565b61071c565b005b3480156103a257600080fd5b506103946103b1366004611b2d565b610904565b3480156103c257600080fd5b506103946103d1366004611bac565b610d62565b3480156103e257600080fd5b506101025461010354610104546040805193845260208401929092529082015260600161022f565b34801561041657600080fd5b5061024e6101015481565b34801561042d57600080fd5b5061024e6101025481565b34801561044457600080fd5b5060335460ff16610223565b34801561045c57600080fd5b5061047061046b366004611aa4565b6111bb565b6040805192835260208301919091520161022f565b34801561049157600080fd5b5061024e61025881565b3480156104a757600080fd5b506103946111e9565b3480156104bc57600080fd5b506103946111fd565b3480156104d157600080fd5b5061024e600281565b3480156104e657600080fd5b506065546001600160a01b03165b6040516001600160a01b03909116815260200161022f565b34801561051857600080fd5b5060fb546104f4906001600160a01b031681565b34801561053857600080fd5b5061024e670de0b6b3a764000081565b34801561055457600080fd5b5061024e6101f481565b34801561056a57600080fd5b5061024e600381565b34801561057f57600080fd5b5061024e61058e366004611aa4565b611277565b34801561059f57600080fd5b5061024e6101045481565b3480156105b657600080fd5b506097546001600160a01b03166104f4565b3480156105d457600080fd5b5061024e606481565b3480156105e957600080fd5b5061024e6101035481565b34801561060057600080fd5b5060fc546104f4906001600160a01b031681565b610394610622366004611bdf565b6112f5565b34801561063357600080fd5b50610394610642366004611a89565b6115cf565b34801561065357600080fd5b50610470610662366004611aa4565b611640565b60008181526101006020908152604080832080546001820154600383015460048401546002850180548751818a0281018a01909852808852899889986060988a98899891976001600160a01b03821697600160a01b90920460ff9081169793969395911693919285918301828280156106ff57602002820191906000526020600020905b8154815260200190600101908083116106eb575b505050505092509650965096509650965096505091939550919395565b600181101561075f5760405162461bcd60e51b815260206004820152600a6024820152694d696e203120706c617960b01b60448201526064015b60405180910390fd5b60648111156107a05760405162461bcd60e51b815260206004820152600d60248201526c4d61782031303020706c61797360981b6044820152606401610756565b60006107b482670de0b6b3a7640000611c1f565b905060006107c183611277565b905080156107ed576127106107d68282611c36565b6107e09084611c1f565b6107ea9190611c5f565b91505b60fc5460405163026ff28360e61b8152336004820152602481018490526001600160a01b0390911690639bfca0c0906044016020604051808303816000875af115801561083e573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906108629190611c73565b61089f5760405162461bcd60e51b815260206004820152600e60248201526d14185e5b595b9d0819985a5b195960921b6044820152606401610756565b3360009081526101056020526040812080548592906108bf908490611c9c565b9091555050604080518481526020810184905233917f9732e8185758002b9bc38f23a7b9154c71de4b606eac0a37a41a43418e8ae552910160405180910390a2505050565b61090c611651565b6000838152610100602052604090208054600160a01b900460ff161561096d5760405162461bcd60e51b815260206004820152601660248201527511d85b5948185b1c9958591e4818dbdb5c1b195d195960521b6044820152606401610756565b600382146109bd5760405162461bcd60e51b815260206004820152601760248201527f496e76616c6964206e756d626572206f662072696e67730000000000000000006044820152606401610756565b60008060005b6003811015610b465760ff548686838181106109e1576109e1611caf565b9050602002013510610a2a5760405162461bcd60e51b8152602060048201526012602482015271092dcecc2d8d2c840c8eac6d640d2dcc8caf60731b6044820152606401610756565b600060ff878784818110610a4057610a40611caf565b9050602002013581548110610a5757610a57611caf565b600091825260209091206002909102015460fe5490915060ff168015610ab257506001610aaf610a878442611c9c565b6040518060400160405280600a81526020016909a8a8e82be8682a886960b31b8152506116ab565b11155b15610ac957610ac2606482611c1f565b9050600192505b80856002018381548110610adf57610adf611caf565b600091825260209091200155610af58185611c9c565b604080518481526020810184905291955089917f073b239ab660d21537a3acad9c340b6192a7705e84918ff1524a40181ddecb0b910160405180910390a25080610b3e81611cc5565b9150506109c3565b50610ba283600201805480602002602001604051908101604052809291908181526020018280548015610b9857602002820191906000526020600020905b815481526020019060010190808311610b84575b50505050506116f3565b15610bcd57612710610bb661138884611c1f565b610bc09190611c5f565b610bca9083611c9c565b91505b600383015460ff1615610be857610be5600283611c1f565b91505b60048301829055825460ff60a01b1916600160a01b1783556101038054839190600090610c16908490611c9c565b90915550610c2e905082670de0b6b3a7640000611c36565b6101046000828254610c409190611c9c565b90915550508115610cb55760fc5483546040516339c255ad60e11b81526001600160a01b03918216600482015260248101859052911690637384ab5a9084906044016000604051808303818588803b158015610c9b57600080fd5b505af1158015610caf573d6000803e3d6000fd5b50505050505b8015610cff5782546040518381526001600160a01b039091169087907f7ef80aec4a36549407e99f8e8023b32234edbc3e942e2a78cef9cb641b6dd2259060200160405180910390a35b825460038401546040516001600160a01b039092169188917f295a47f98aa27dff43fc584330c1cc5e5a35401a0bff554c9f9b605aababfa4291610d5291600289019160ff909116908890600090611cde565b60405180910390a3505050505050565b600054610100900460ff1615808015610d825750600054600160ff909116105b80610d9c5750303b158015610d9c575060005460ff166001145b610dff5760405162461bcd60e51b815260206004820152602e60248201527f496e697469616c697a61626c653a20636f6e747261637420697320616c72656160448201526d191e481a5b9a5d1a585b1a5e995960921b6064820152608401610756565b6000805460ff191660011790558015610e22576000805461ff0019166101001790555b610e2a611770565b610e3261179f565b610e3a6117ce565b60fb80546001600160a01b038086166001600160a01b03199283161790925560fc8054928516929091169190911790556040805180820182526702c68af0bb1400008152610dac602080830191825260ff8054600181810183556000838152955160029283027fe08ec2af2cfc251225e1968fd6ca21e4044f129bffa95bac3503be8bdb30a3678181019290925595517fe08ec2af2cfc251225e1968fd6ca21e4044f129bffa95bac3503be8bdb30a3689687015587518089018952670429d069189e000081526107d081870181815286548086018855878b5292519286028085019390935551918801919091558851808a018a526706f05b59d3b20000815280870191825285548085018755868a5290519085028084019190915590519087015587518089018952670b1a2bc2ec50000081526104b081870190815285548085018755868a529151918502808401929092555190870155875180890189526710a741a462780000815261032081870190815285548085018755868a52915191850280840192909255519087015587518089018952671bc16d674ec80000815261012c81870181815286548086018855878b5292519286028085019390935551918801919091558851808a018a52674563918244f40000815260b481880190815286548086018855878b5291519186028085019290925551908801558851808a018a52678ac7230489e800008152601481880181815287548087018955978b5291519686029384019690965551919096015586518088018852600581528085019586526101068054808401825581895291519184027fc9ef9fceea91e87b2c84ea400a44fde78842aae8aa24cd4b502ce5fb4d91e63b8181019390935596517fc9ef9fceea91e87b2c84ea400a44fde78842aae8aa24cd4b502ce5fb4d91e63c978801558851808a018a529485526101f485870190815281548085018355828a52955195850280840196909655519487019490945587518089018952603281526102bc81870190815285548085018755868a5291519185028084019290925551908701558751808901909852606488526103e8948801948552835491820184559290955294519390940293840192909255905191015580156111b6576000805461ff0019169055604051600181527f7f26b83ff96e1f2b6a682f133852f6798a09c465da95921460cefb38474024989060200160405180910390a15b505050565b60ff81815481106111cb57600080fd5b60009182526020909120600290910201805460019091015490915082565b6111f1611651565b6111fb60006117fd565b565b60975433906001600160a01b0316811461126b5760405162461bcd60e51b815260206004820152602960248201527f4f776e61626c6532537465703a2063616c6c6572206973206e6f7420746865206044820152683732bb9037bbb732b960b91b6064820152608401610756565b611274816117fd565b50565b600080805b610106548110156112ee57610106818154811061129b5761129b611caf565b90600052602060002090600202016000015484106112dc5761010681815481106112c7576112c7611caf565b90600052602060002090600202016001015491505b806112e681611cc5565b91505061127c565b5092915050565b60fb546001600160a01b031633146113445760405162461bcd60e51b815260206004820152601260248201527127b7363c90383937bc3c9036b0b730b3b2b960711b6044820152606401610756565b8015801561136a57506001600160a01b0382166000908152610105602052604090205415155b156113ec576001600160a01b03821660009081526101056020526040812080549161139483611d3d565b90915550506001600160a01b038216600081815261010560209081526040918290205491519182527fad33e07839b794fbe43ff46168b0df91fc810a1dff35da022bf8a2925f51f357910160405180910390a2611437565b670de0b6b3a764000081146114375760405162461bcd60e51b8152602060048201526011602482015270125b9d985b1a59081c1b185e4818dbdcdd607a1b6044820152606401610756565b61143f611816565b60006101f461146e6101015460405180604001604052806005815260200164424f4e555360d81b8152506116ab565b610101805492909111159250600061148583611cc5565b90915550506040805160c0810182526001600160a01b03851681526000602082015242818301528151600380825260808201909352909160608301919081602001602082028036833701905050815282151560208083019190915260006040928301819052610101548152610100825282902083518154858401511515600160a01b026001600160a81b03199091166001600160a01b0390921691909117178155918301516001830155606083015180516115469260028501920190611a0d565b50608082015160038201805460ff191691151591909117905560a090910151600490910155610102805490600061157c83611cc5565b9190505550826001600160a01b0316610101547fbc0699d39a841f5917765a96bad29ca3fc6af279e1a5d2b2dd6ac382cc11aaf0836040516115c2911515815260200190565b60405180910390a3505050565b6115d7611651565b609780546001600160a01b0383166001600160a01b031990911681179091556116086065546001600160a01b031690565b6001600160a01b03167f38d16b8cac22d99fc7c124b9cd0de2d3fa1faef420bfe791d8c362d765e2270060405160405180910390a350565b61010681815481106111cb57600080fd5b6065546001600160a01b031633146111fb5760405162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e65726044820152606401610756565b6000612710424485856040516020016116c79493929190611d54565b6040516020818303038152906040528051906020012060001c6116ea9190611d9d565b90505b92915050565b60008160018151811061170857611708611caf565b60200260200101518260008151811061172357611723611caf565b60200260200101511480156116ed57508160028151811061174657611746611caf565b60200260200101518260018151811061176157611761611caf565b60200260200101511492915050565b600054610100900460ff166117975760405162461bcd60e51b815260040161075690611db1565b6111fb61192a565b600054610100900460ff166117c65760405162461bcd60e51b815260040161075690611db1565b6111fb61195d565b600054610100900460ff166117f55760405162461bcd60e51b815260040161075690611db1565b6111fb61198d565b609780546001600160a01b0319169055611274816119bb565b60fe5460ff16158015611838575060fd546118349062015180611c9c565b4210155b80156118725750600161186f426040518060400160405280600c81526020016b26a2a3a0afaa2924a3a3a2a960a11b8152506116ab565b11155b156118c55760fe805460ff191660011790554260fd8190556040805191825261025860208301527fa85dc7964b8b25b9951840e354cc63e587d589e5b132d44eeba47fd2eb749d0d910160405180910390a15b60fe5460ff1680156118e6575061025860fd546118e29190611c9c565b4210155b156111fb5760fe805460ff191690556040514281527f6c4a9d3b0ebd1b87178cad599c022a7da0376a3ae0719e3a9876c8690ca6d8089060200160405180910390a1565b600054610100900460ff166119515760405162461bcd60e51b815260040161075690611db1565b6033805460ff19169055565b600054610100900460ff166119845760405162461bcd60e51b815260040161075690611db1565b6111fb336117fd565b600054610100900460ff166119b45760405162461bcd60e51b815260040161075690611db1565b600160c955565b606580546001600160a01b038381166001600160a01b0319831681179093556040519116919082907f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e090600090a35050565b828054828255906000526020600020908101928215611a48579160200282015b82811115611a48578251825591602001919060010190611a2d565b50611a54929150611a58565b5090565b5b80821115611a545760008155600101611a59565b80356001600160a01b0381168114611a8457600080fd5b919050565b600060208284031215611a9b57600080fd5b6116ea82611a6d565b600060208284031215611ab657600080fd5b5035919050565b600060c0820160018060a01b038916835260208815158185015287604085015260c0606085015281875180845260e086019150828901935060005b81811015611b1457845183529383019391830191600101611af8565b5050951515608085015250505060a00152949350505050565b600080600060408486031215611b4257600080fd5b83359250602084013567ffffffffffffffff80821115611b6157600080fd5b818601915086601f830112611b7557600080fd5b813581811115611b8457600080fd5b8760208260051b8501011115611b9957600080fd5b6020830194508093505050509250925092565b60008060408385031215611bbf57600080fd5b611bc883611a6d565b9150611bd660208401611a6d565b90509250929050565b60008060408385031215611bf257600080fd5b611bfb83611a6d565b946020939093013593505050565b634e487b7160e01b600052601160045260246000fd5b80820281158282048414176116ed576116ed611c09565b818103818111156116ed576116ed611c09565b634e487b7160e01b600052601260045260246000fd5b600082611c6e57611c6e611c49565b500490565b600060208284031215611c8557600080fd5b81518015158114611c9557600080fd5b9392505050565b808201808211156116ed576116ed611c09565b634e487b7160e01b600052603260045260246000fd5b600060018201611cd757611cd7611c09565b5060010190565b6000608082016080835280875480835260a08501915088600052602092508260002060005b82811015611d1f57815484529284019260019182019101611d03565b50505095151590830152506040810192909252606090910152919050565b600081611d4c57611d4c611c09565b506000190190565b848152600060208581840152846040840152835160005b81811015611d8757858101830151858201606001528201611d6b565b5060009301606001928352509095945050505050565b600082611dac57611dac611c49565b500690565b6020808252602b908201527f496e697469616c697a61626c653a20636f6e7472616374206973206e6f74206960408201526a6e697469616c697a696e6760a81b60608201526080019056fea26469706673582212204c950c3678846aa58e17b3a1719d78a70f8357c843fa6f8d1cce8c0e474b5abe64736f6c63430008130033
Deployed Bytecode
0x6080604052600436106102045760003560e01c806379ba509711610118578063dc8f84e2116100a0578063ed740e971161006f578063ed740e97146105f4578063f0e10c0d14610614578063f2fde38b14610627578063fa84dc2f14610647578063fda5eff5146105c857600080fd5b8063dc8f84e214610593578063e30c3978146105aa578063e366c058146105c8578063e7b0f666146105dd57600080fd5b80639a46f89f116100e75780639a46f89f14610349578063b216a6c61461052c578063d053a25714610548578063d09bd9721461055e578063d4cffc4d1461057357600080fd5b806379ba5097146104b05780638aa28550146104c55780638da5cb5b146104da57806396b4ca701461050c57600080fd5b8063406c7b8d1161019b5780635c4921291161016a5780635c492129146104215780635c975abb146104385780636003bdaf146104505780636e032e3d14610485578063715018a61461049b57600080fd5b8063406c7b8d14610396578063485cc955146103b6578063496621d7146103d6578063536a3ddc1461040a57600080fd5b80631b31abda116101d75780631b31abda146103175780632731f9ef146103495780632dd713411461035e5780632e9830ce1461037457600080fd5b8063036ebf5d1461020957806304336f8f146102385780630fc345c91461025c578063117a5b901461028a575b600080fd5b34801561021557600080fd5b5060fe546102239060ff1681565b60405190151581526020015b60405180910390f35b34801561024457600080fd5b5061024e61138881565b60405190815260200161022f565b34801561026857600080fd5b5061024e610277366004611a89565b6101056020526000908152604090205481565b34801561029657600080fd5b506102e36102a5366004611aa4565b6101006020526000908152604090208054600182015460038301546004909301546001600160a01b0383169360ff600160a01b909404841693169085565b604080516001600160a01b03909616865293151560208601529284019190915215156060830152608082015260a00161022f565b34801561032357600080fd5b50610337610332366004611aa4565b610667565b60405161022f96959493929190611abd565b34801561035557600080fd5b5061024e600181565b34801561036a57600080fd5b5061024e60fd5481565b34801561038057600080fd5b5061039461038f366004611aa4565b61071c565b005b3480156103a257600080fd5b506103946103b1366004611b2d565b610904565b3480156103c257600080fd5b506103946103d1366004611bac565b610d62565b3480156103e257600080fd5b506101025461010354610104546040805193845260208401929092529082015260600161022f565b34801561041657600080fd5b5061024e6101015481565b34801561042d57600080fd5b5061024e6101025481565b34801561044457600080fd5b5060335460ff16610223565b34801561045c57600080fd5b5061047061046b366004611aa4565b6111bb565b6040805192835260208301919091520161022f565b34801561049157600080fd5b5061024e61025881565b3480156104a757600080fd5b506103946111e9565b3480156104bc57600080fd5b506103946111fd565b3480156104d157600080fd5b5061024e600281565b3480156104e657600080fd5b506065546001600160a01b03165b6040516001600160a01b03909116815260200161022f565b34801561051857600080fd5b5060fb546104f4906001600160a01b031681565b34801561053857600080fd5b5061024e670de0b6b3a764000081565b34801561055457600080fd5b5061024e6101f481565b34801561056a57600080fd5b5061024e600381565b34801561057f57600080fd5b5061024e61058e366004611aa4565b611277565b34801561059f57600080fd5b5061024e6101045481565b3480156105b657600080fd5b506097546001600160a01b03166104f4565b3480156105d457600080fd5b5061024e606481565b3480156105e957600080fd5b5061024e6101035481565b34801561060057600080fd5b5060fc546104f4906001600160a01b031681565b610394610622366004611bdf565b6112f5565b34801561063357600080fd5b50610394610642366004611a89565b6115cf565b34801561065357600080fd5b50610470610662366004611aa4565b611640565b60008181526101006020908152604080832080546001820154600383015460048401546002850180548751818a0281018a01909852808852899889986060988a98899891976001600160a01b03821697600160a01b90920460ff9081169793969395911693919285918301828280156106ff57602002820191906000526020600020905b8154815260200190600101908083116106eb575b505050505092509650965096509650965096505091939550919395565b600181101561075f5760405162461bcd60e51b815260206004820152600a6024820152694d696e203120706c617960b01b60448201526064015b60405180910390fd5b60648111156107a05760405162461bcd60e51b815260206004820152600d60248201526c4d61782031303020706c61797360981b6044820152606401610756565b60006107b482670de0b6b3a7640000611c1f565b905060006107c183611277565b905080156107ed576127106107d68282611c36565b6107e09084611c1f565b6107ea9190611c5f565b91505b60fc5460405163026ff28360e61b8152336004820152602481018490526001600160a01b0390911690639bfca0c0906044016020604051808303816000875af115801561083e573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906108629190611c73565b61089f5760405162461bcd60e51b815260206004820152600e60248201526d14185e5b595b9d0819985a5b195960921b6044820152606401610756565b3360009081526101056020526040812080548592906108bf908490611c9c565b9091555050604080518481526020810184905233917f9732e8185758002b9bc38f23a7b9154c71de4b606eac0a37a41a43418e8ae552910160405180910390a2505050565b61090c611651565b6000838152610100602052604090208054600160a01b900460ff161561096d5760405162461bcd60e51b815260206004820152601660248201527511d85b5948185b1c9958591e4818dbdb5c1b195d195960521b6044820152606401610756565b600382146109bd5760405162461bcd60e51b815260206004820152601760248201527f496e76616c6964206e756d626572206f662072696e67730000000000000000006044820152606401610756565b60008060005b6003811015610b465760ff548686838181106109e1576109e1611caf565b9050602002013510610a2a5760405162461bcd60e51b8152602060048201526012602482015271092dcecc2d8d2c840c8eac6d640d2dcc8caf60731b6044820152606401610756565b600060ff878784818110610a4057610a40611caf565b9050602002013581548110610a5757610a57611caf565b600091825260209091206002909102015460fe5490915060ff168015610ab257506001610aaf610a878442611c9c565b6040518060400160405280600a81526020016909a8a8e82be8682a886960b31b8152506116ab565b11155b15610ac957610ac2606482611c1f565b9050600192505b80856002018381548110610adf57610adf611caf565b600091825260209091200155610af58185611c9c565b604080518481526020810184905291955089917f073b239ab660d21537a3acad9c340b6192a7705e84918ff1524a40181ddecb0b910160405180910390a25080610b3e81611cc5565b9150506109c3565b50610ba283600201805480602002602001604051908101604052809291908181526020018280548015610b9857602002820191906000526020600020905b815481526020019060010190808311610b84575b50505050506116f3565b15610bcd57612710610bb661138884611c1f565b610bc09190611c5f565b610bca9083611c9c565b91505b600383015460ff1615610be857610be5600283611c1f565b91505b60048301829055825460ff60a01b1916600160a01b1783556101038054839190600090610c16908490611c9c565b90915550610c2e905082670de0b6b3a7640000611c36565b6101046000828254610c409190611c9c565b90915550508115610cb55760fc5483546040516339c255ad60e11b81526001600160a01b03918216600482015260248101859052911690637384ab5a9084906044016000604051808303818588803b158015610c9b57600080fd5b505af1158015610caf573d6000803e3d6000fd5b50505050505b8015610cff5782546040518381526001600160a01b039091169087907f7ef80aec4a36549407e99f8e8023b32234edbc3e942e2a78cef9cb641b6dd2259060200160405180910390a35b825460038401546040516001600160a01b039092169188917f295a47f98aa27dff43fc584330c1cc5e5a35401a0bff554c9f9b605aababfa4291610d5291600289019160ff909116908890600090611cde565b60405180910390a3505050505050565b600054610100900460ff1615808015610d825750600054600160ff909116105b80610d9c5750303b158015610d9c575060005460ff166001145b610dff5760405162461bcd60e51b815260206004820152602e60248201527f496e697469616c697a61626c653a20636f6e747261637420697320616c72656160448201526d191e481a5b9a5d1a585b1a5e995960921b6064820152608401610756565b6000805460ff191660011790558015610e22576000805461ff0019166101001790555b610e2a611770565b610e3261179f565b610e3a6117ce565b60fb80546001600160a01b038086166001600160a01b03199283161790925560fc8054928516929091169190911790556040805180820182526702c68af0bb1400008152610dac602080830191825260ff8054600181810183556000838152955160029283027fe08ec2af2cfc251225e1968fd6ca21e4044f129bffa95bac3503be8bdb30a3678181019290925595517fe08ec2af2cfc251225e1968fd6ca21e4044f129bffa95bac3503be8bdb30a3689687015587518089018952670429d069189e000081526107d081870181815286548086018855878b5292519286028085019390935551918801919091558851808a018a526706f05b59d3b20000815280870191825285548085018755868a5290519085028084019190915590519087015587518089018952670b1a2bc2ec50000081526104b081870190815285548085018755868a529151918502808401929092555190870155875180890189526710a741a462780000815261032081870190815285548085018755868a52915191850280840192909255519087015587518089018952671bc16d674ec80000815261012c81870181815286548086018855878b5292519286028085019390935551918801919091558851808a018a52674563918244f40000815260b481880190815286548086018855878b5291519186028085019290925551908801558851808a018a52678ac7230489e800008152601481880181815287548087018955978b5291519686029384019690965551919096015586518088018852600581528085019586526101068054808401825581895291519184027fc9ef9fceea91e87b2c84ea400a44fde78842aae8aa24cd4b502ce5fb4d91e63b8181019390935596517fc9ef9fceea91e87b2c84ea400a44fde78842aae8aa24cd4b502ce5fb4d91e63c978801558851808a018a529485526101f485870190815281548085018355828a52955195850280840196909655519487019490945587518089018952603281526102bc81870190815285548085018755868a5291519185028084019290925551908701558751808901909852606488526103e8948801948552835491820184559290955294519390940293840192909255905191015580156111b6576000805461ff0019169055604051600181527f7f26b83ff96e1f2b6a682f133852f6798a09c465da95921460cefb38474024989060200160405180910390a15b505050565b60ff81815481106111cb57600080fd5b60009182526020909120600290910201805460019091015490915082565b6111f1611651565b6111fb60006117fd565b565b60975433906001600160a01b0316811461126b5760405162461bcd60e51b815260206004820152602960248201527f4f776e61626c6532537465703a2063616c6c6572206973206e6f7420746865206044820152683732bb9037bbb732b960b91b6064820152608401610756565b611274816117fd565b50565b600080805b610106548110156112ee57610106818154811061129b5761129b611caf565b90600052602060002090600202016000015484106112dc5761010681815481106112c7576112c7611caf565b90600052602060002090600202016001015491505b806112e681611cc5565b91505061127c565b5092915050565b60fb546001600160a01b031633146113445760405162461bcd60e51b815260206004820152601260248201527127b7363c90383937bc3c9036b0b730b3b2b960711b6044820152606401610756565b8015801561136a57506001600160a01b0382166000908152610105602052604090205415155b156113ec576001600160a01b03821660009081526101056020526040812080549161139483611d3d565b90915550506001600160a01b038216600081815261010560209081526040918290205491519182527fad33e07839b794fbe43ff46168b0df91fc810a1dff35da022bf8a2925f51f357910160405180910390a2611437565b670de0b6b3a764000081146114375760405162461bcd60e51b8152602060048201526011602482015270125b9d985b1a59081c1b185e4818dbdcdd607a1b6044820152606401610756565b61143f611816565b60006101f461146e6101015460405180604001604052806005815260200164424f4e555360d81b8152506116ab565b610101805492909111159250600061148583611cc5565b90915550506040805160c0810182526001600160a01b03851681526000602082015242818301528151600380825260808201909352909160608301919081602001602082028036833701905050815282151560208083019190915260006040928301819052610101548152610100825282902083518154858401511515600160a01b026001600160a81b03199091166001600160a01b0390921691909117178155918301516001830155606083015180516115469260028501920190611a0d565b50608082015160038201805460ff191691151591909117905560a090910151600490910155610102805490600061157c83611cc5565b9190505550826001600160a01b0316610101547fbc0699d39a841f5917765a96bad29ca3fc6af279e1a5d2b2dd6ac382cc11aaf0836040516115c2911515815260200190565b60405180910390a3505050565b6115d7611651565b609780546001600160a01b0383166001600160a01b031990911681179091556116086065546001600160a01b031690565b6001600160a01b03167f38d16b8cac22d99fc7c124b9cd0de2d3fa1faef420bfe791d8c362d765e2270060405160405180910390a350565b61010681815481106111cb57600080fd5b6065546001600160a01b031633146111fb5760405162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e65726044820152606401610756565b6000612710424485856040516020016116c79493929190611d54565b6040516020818303038152906040528051906020012060001c6116ea9190611d9d565b90505b92915050565b60008160018151811061170857611708611caf565b60200260200101518260008151811061172357611723611caf565b60200260200101511480156116ed57508160028151811061174657611746611caf565b60200260200101518260018151811061176157611761611caf565b60200260200101511492915050565b600054610100900460ff166117975760405162461bcd60e51b815260040161075690611db1565b6111fb61192a565b600054610100900460ff166117c65760405162461bcd60e51b815260040161075690611db1565b6111fb61195d565b600054610100900460ff166117f55760405162461bcd60e51b815260040161075690611db1565b6111fb61198d565b609780546001600160a01b0319169055611274816119bb565b60fe5460ff16158015611838575060fd546118349062015180611c9c565b4210155b80156118725750600161186f426040518060400160405280600c81526020016b26a2a3a0afaa2924a3a3a2a960a11b8152506116ab565b11155b156118c55760fe805460ff191660011790554260fd8190556040805191825261025860208301527fa85dc7964b8b25b9951840e354cc63e587d589e5b132d44eeba47fd2eb749d0d910160405180910390a15b60fe5460ff1680156118e6575061025860fd546118e29190611c9c565b4210155b156111fb5760fe805460ff191690556040514281527f6c4a9d3b0ebd1b87178cad599c022a7da0376a3ae0719e3a9876c8690ca6d8089060200160405180910390a1565b600054610100900460ff166119515760405162461bcd60e51b815260040161075690611db1565b6033805460ff19169055565b600054610100900460ff166119845760405162461bcd60e51b815260040161075690611db1565b6111fb336117fd565b600054610100900460ff166119b45760405162461bcd60e51b815260040161075690611db1565b600160c955565b606580546001600160a01b038381166001600160a01b0319831681179093556040519116919082907f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e090600090a35050565b828054828255906000526020600020908101928215611a48579160200282015b82811115611a48578251825591602001919060010190611a2d565b50611a54929150611a58565b5090565b5b80821115611a545760008155600101611a59565b80356001600160a01b0381168114611a8457600080fd5b919050565b600060208284031215611a9b57600080fd5b6116ea82611a6d565b600060208284031215611ab657600080fd5b5035919050565b600060c0820160018060a01b038916835260208815158185015287604085015260c0606085015281875180845260e086019150828901935060005b81811015611b1457845183529383019391830191600101611af8565b5050951515608085015250505060a00152949350505050565b600080600060408486031215611b4257600080fd5b83359250602084013567ffffffffffffffff80821115611b6157600080fd5b818601915086601f830112611b7557600080fd5b813581811115611b8457600080fd5b8760208260051b8501011115611b9957600080fd5b6020830194508093505050509250925092565b60008060408385031215611bbf57600080fd5b611bc883611a6d565b9150611bd660208401611a6d565b90509250929050565b60008060408385031215611bf257600080fd5b611bfb83611a6d565b946020939093013593505050565b634e487b7160e01b600052601160045260246000fd5b80820281158282048414176116ed576116ed611c09565b818103818111156116ed576116ed611c09565b634e487b7160e01b600052601260045260246000fd5b600082611c6e57611c6e611c49565b500490565b600060208284031215611c8557600080fd5b81518015158114611c9557600080fd5b9392505050565b808201808211156116ed576116ed611c09565b634e487b7160e01b600052603260045260246000fd5b600060018201611cd757611cd7611c09565b5060010190565b6000608082016080835280875480835260a08501915088600052602092508260002060005b82811015611d1f57815484529284019260019182019101611d03565b50505095151590830152506040810192909252606090910152919050565b600081611d4c57611d4c611c09565b506000190190565b848152600060208581840152846040840152835160005b81811015611d8757858101830151858201606001528201611d6b565b5060009301606001928352509095945050505050565b600082611dac57611dac611c49565b500690565b6020808252602b908201527f496e697469616c697a61626c653a20636f6e7472616374206973206e6f74206960408201526a6e697469616c697a696e6760a81b60608201526080019056fea26469706673582212204c950c3678846aa58e17b3a1719d78a70f8357c843fa6f8d1cce8c0e474b5abe64736f6c63430008130033
Loading...
Loading
Loading...
Loading
Multichain Portfolio | 30 Chains
Chain | Token | Portfolio % | Price | Amount | Value |
---|
[ Download: CSV Export ]
A contract address hosts a smart contract, which is a set of code stored on the blockchain that runs when predetermined conditions are met. Learn more about addresses in our Knowledge Base.