Source Code
Overview
ETH Balance
ETH Value
$0.00| Transaction Hash |
|
Block
|
From
|
To
|
|||||
|---|---|---|---|---|---|---|---|---|---|
Latest 25 internal transactions (View All)
Advanced mode:
| Parent Transaction Hash | Block | From | To | |||
|---|---|---|---|---|---|---|
| 26043677 | 61 days ago | 0 ETH | ||||
| 25670958 | 72 days ago | 0 ETH | ||||
| 25356351 | 81 days ago | 0 ETH | ||||
| 24971325 | 91 days ago | 0 ETH | ||||
| 24771882 | 97 days ago | 0 ETH | ||||
| 23837526 | 122 days ago | 0 ETH | ||||
| 23352158 | 133 days ago | 0 ETH | ||||
| 23352148 | 133 days ago | 0 ETH | ||||
| 23066842 | 140 days ago | 0 ETH | ||||
| 22620591 | 151 days ago | 0 ETH | ||||
| 22620577 | 151 days ago | 0 ETH | ||||
| 21749397 | 172 days ago | 0 ETH | ||||
| 21145736 | 188 days ago | 0 ETH | ||||
| 21114343 | 189 days ago | 0 ETH | ||||
| 20975598 | 193 days ago | 0 ETH | ||||
| 20821803 | 197 days ago | 0 ETH | ||||
| 20759880 | 199 days ago | 0 ETH | ||||
| 20759875 | 199 days ago | 0 ETH | ||||
| 20712140 | 201 days ago | 0 ETH | ||||
| 20408768 | 212 days ago | 0 ETH | ||||
| 20394453 | 212 days ago | 0 ETH | ||||
| 20394442 | 212 days ago | 0 ETH | ||||
| 19768782 | 233 days ago | 0 ETH | ||||
| 19270882 | 248 days ago | 0 ETH | ||||
| 19253635 | 249 days ago | 0 ETH |
Cross-Chain Transactions
Loading...
Loading
Contract Name:
DCBPlatformVesting
Compiler Version
v0.8.19+commit.7dd6d404
Optimization Enabled:
Yes with 10000 runs
Other Settings:
default evmVersion
Contract Source Code (Solidity Standard Json-Input format)
// SPDX-License-Identifier: MIT
//** DCB vesting Contract */
//** Author Aaron & Aceson : DCB 2023.2 */
pragma solidity 0.8.19;
import { Ownable } from "openzeppelin-contracts/contracts/access/Ownable.sol";
import { SafeERC20, IERC20 } from "openzeppelin-contracts/contracts/token/ERC20/utils/SafeERC20.sol";
import { ERC20 } from "openzeppelin-contracts/contracts/token/ERC20/ERC20.sol";
import { Initializable } from "openzeppelin-contracts/contracts/proxy/utils/Initializable.sol";
import { IUniswapV2Router02 } from "v2-periphery/interfaces/IUniswapV2Router02.sol";
import { IDCBPlatformVesting } from "./interfaces/IDCBPlatformVesting.sol";
import { DateTime } from "./libraries/DateTime.sol";
interface ICrowdFunding {
function userAllocation(address user)
external
view
returns (uint8 tier, uint8 multi, uint256 shares, uint256 deposits, bool active);
}
contract DCBPlatformVesting is Ownable, DateTime, Initializable, IDCBPlatformVesting {
using SafeERC20 for IERC20;
VestingPool public vestingPool;
// refund total values
uint256 public totalVestedValue;
uint256 public totalRefunded;
uint256 public totalVestedToken;
uint256 public totalReturnedToken;
uint256 public totalTokenOnSale;
uint256 public platformFee;
uint256 public gracePeriod;
address public innovator;
address public paymentReceiver;
address public router;
address[] public path;
bool public claimed;
uint256[] public refundFees;
uint32 internal nativeChainId;
IERC20 public vestedToken;
IERC20 public paymentToken;
address public factory;
uint8 public constant VERSION = 1;
event CrowdfundingInitialized(ContractSetup c, VestingSetup p, BuybackSetup b);
event CrowdFundingSet(ContractSetup c);
event TokenClaimInitialized(address _token, VestingSetup p);
event VestingStrategyAdded(uint256 _cliff, uint256 _start, uint256 _duration, uint256 _initialUnlockPercent);
event RaisedFundsClaimed(uint256 payment, uint256 remaining);
event BuybackAndBurn(uint256 amount);
event SetVestingParams(uint256 _cliff, uint256 _start, uint256 _duration, uint256 _initialUnlockPercent);
modifier onlyInnovator() {
require(msg.sender == innovator, "Invalid access");
_;
}
modifier onlyFactory() {
require(msg.sender == factory, "Only factory");
_;
}
modifier userInWhitelist(address _wallet) {
require(vestingPool.hasWhitelist[_wallet].active, "Not in whitelist");
_;
}
function initializeCrowdfunding(
ContractSetup calldata c,
VestingSetup calldata p,
BuybackSetup calldata b
)
external
initializer
{
innovator = c._innovator;
paymentReceiver = c._paymentReceiver;
vestedToken = IERC20(c._vestedToken);
paymentToken = IERC20(c._paymentToken);
gracePeriod = c._gracePeriod;
totalTokenOnSale = c._totalTokenOnSale;
nativeChainId = c._nativeChainId;
refundFees = c._refundFees;
router = b.router;
path = b.path;
paymentToken.approve(router, type(uint256).max);
_transferOwnership(msg.sender);
factory = msg.sender;
platformFee = 800; // 8%
addVestingStrategy(p._cliff, p._startTime, p._duration, p._initialUnlockPercent);
emit CrowdfundingInitialized(c, p, b);
}
function setCrowdFundingParams(ContractSetup calldata c, uint256 _platformFee) external onlyFactory {
require(block.timestamp < vestingPool.start, "Vesting already started");
innovator = c._innovator;
paymentReceiver = c._paymentReceiver;
vestedToken = IERC20(c._vestedToken);
paymentToken = IERC20(c._paymentToken);
gracePeriod = c._gracePeriod;
totalTokenOnSale = c._totalTokenOnSale;
refundFees = c._refundFees;
platformFee = _platformFee;
paymentToken.approve(router, type(uint256).max);
emit CrowdFundingSet(c);
}
function initializeTokenClaim(
address _token,
VestingSetup calldata p,
uint32 _nativeChainId
)
external
initializer
{
vestedToken = IERC20(_token);
_transferOwnership(msg.sender);
factory = msg.sender;
nativeChainId = _nativeChainId;
addVestingStrategy(p._cliff, p._startTime, p._duration, p._initialUnlockPercent);
emit TokenClaimInitialized(_token, p);
}
function addVestingStrategy(
uint256 _cliff,
uint256 _start,
uint256 _duration,
uint256 _initialUnlockPercent
)
internal
returns (bool)
{
vestingPool.cliff = _start + _cliff;
vestingPool.start = _start;
vestingPool.duration = _duration;
vestingPool.initialUnlockPercent = _initialUnlockPercent;
emit VestingStrategyAdded(_cliff, _start, _duration, _initialUnlockPercent);
return true;
}
function setVestingParams(
uint256 _cliff,
uint256 _start,
uint256 _duration,
uint256 _initialUnlockPercent
)
external
onlyFactory
{
require(block.timestamp < vestingPool.start, "Vesting already started");
addVestingStrategy(_cliff, _start, _duration, _initialUnlockPercent);
emit SetVestingParams(_cliff, _start, _duration, _initialUnlockPercent);
}
function setToken(address _token) external onlyFactory {
vestedToken = IERC20(_token);
}
function rescueTokens(address _receiver, uint256 _amount) external onlyFactory {
require(
block.timestamp < vestingPool.start || vestedToken.balanceOf(address(this)) - totalVestedToken >= _amount,
"Invalid amount"
);
vestedToken.transfer(_receiver, _amount);
}
function refund() external userInWhitelist(msg.sender) {
uint256 idx = vestingPool.hasWhitelist[msg.sender].arrIdx;
WhitelistInfo storage whitelist = vestingPool.whitelistPool[idx];
require(
block.timestamp < vestingPool.start + gracePeriod && block.timestamp > vestingPool.start,
"Not in grace period"
);
require(!whitelist.refunded, "user already refunded");
require(whitelist.distributedAmount == 0, "user already claimed");
(uint256 tier, uint256 multi,,,) = ICrowdFunding(address(owner())).userAllocation(msg.sender);
uint256 refundFee = refundFees[tier];
if (multi > 1) {
uint256 multiReduction = (multi - 1) * 50;
refundFee = refundFee > multiReduction ? refundFee - multiReduction : 0;
}
uint256 fee = whitelist.value * refundFee / 10_000;
uint256 refundAmount = whitelist.value - fee;
whitelist.refunded = true;
whitelist.refundDate = uint256(block.timestamp);
totalRefunded += whitelist.value;
totalReturnedToken += whitelist.amount;
// Transfer BUSD to user sub some percent of fee
paymentToken.safeTransfer(msg.sender, refundAmount);
if (fee > 0) {
if (block.chainid == nativeChainId) {
_doBuybackAndBurn(fee);
} else {
paymentToken.safeTransfer(paymentReceiver, fee);
}
}
emit Refund(msg.sender, refundAmount);
}
function transferOwnership(address newOwner) public override(Ownable, IDCBPlatformVesting) onlyOwner {
super.transferOwnership(newOwner);
}
function claimRaisedFunds() external onlyInnovator {
require(block.timestamp > gracePeriod + vestingPool.start, "grace period in progress");
require(!claimed, "already claimed");
// payment amount = total value - total refunded
uint256 amountPayment = totalVestedValue - totalRefunded;
// calculate fee of 5%
uint256 decubateFee = amountPayment * platformFee / 10_000;
amountPayment -= decubateFee;
// amount of project tokens to return = amount not sold + amount refunded
uint256 amountTokenToReturn = totalTokenOnSale - totalVestedToken + totalReturnedToken;
claimed = true;
// transfer payment + refunded tokens to project
if (amountPayment > 0) {
paymentToken.safeTransfer(innovator, amountPayment);
}
if (amountTokenToReturn > 0) {
vestedToken.safeTransfer(innovator, amountTokenToReturn);
}
// transfer crowdfunding fee to payment receiver wallet
if (decubateFee > 0) {
paymentToken.safeTransfer(paymentReceiver, decubateFee);
}
emit RaisedFundsClaimed(amountPayment, amountTokenToReturn);
}
function getWhitelist(address _wallet) external view userInWhitelist(_wallet) returns (WhitelistInfo memory) {
uint256 idx = vestingPool.hasWhitelist[_wallet].arrIdx;
return vestingPool.whitelistPool[idx];
}
function getTotalToken(address _addr) external view returns (uint256) {
IERC20 _token = IERC20(_addr);
return _token.balanceOf(address(this));
}
function hasWhitelist(address _wallet) external view returns (bool) {
return vestingPool.hasWhitelist[_wallet].active;
}
function getVestAmount(address _wallet) external view returns (uint256) {
return calculateVestAmount(_wallet);
}
function getReleasableAmount(address _wallet) external view returns (uint256) {
return calculateReleasableAmount(_wallet);
}
function getWhitelistPool() external view returns (WhitelistInfo[] memory) {
return vestingPool.whitelistPool;
}
function claimDistribution(address _wallet) public returns (bool) {
uint256 idx = vestingPool.hasWhitelist[_wallet].arrIdx;
WhitelistInfo storage whitelist = vestingPool.whitelistPool[idx];
require(!whitelist.refunded, "user already refunded");
uint256 releaseAmount = calculateReleasableAmount(_wallet);
require(releaseAmount > 0, "Zero amount");
whitelist.distributedAmount = whitelist.distributedAmount + releaseAmount;
vestedToken.safeTransfer(_wallet, releaseAmount);
emit Claim(_wallet, releaseAmount, block.timestamp);
return true;
}
function setTokenClaimWhitelist(address _wallet, uint256 _amount) public onlyOwner {
require(!vestingPool.hasWhitelist[_wallet].active, "Already registered");
_setWhitelist(_wallet, _amount, 0);
}
function setCrowdfundingWhitelist(address _wallet, uint256 _amount, uint256 _value) public onlyOwner {
uint256 paymentAmount = !vestingPool.hasWhitelist[_wallet].active
? _value
: _value - vestingPool.whitelistPool[vestingPool.hasWhitelist[_wallet].arrIdx].value;
paymentToken.safeTransferFrom(_wallet, address(this), paymentAmount);
_setWhitelist(_wallet, _amount, _value);
}
function _setWhitelist(address _wallet, uint256 _amount, uint256 _value) internal {
HasWhitelist storage whitelist = vestingPool.hasWhitelist[_wallet];
if (!whitelist.active) {
whitelist.active = true;
whitelist.arrIdx = vestingPool.whitelistPool.length;
vestingPool.whitelistPool.push(
WhitelistInfo({
wallet: _wallet,
amount: _amount,
distributedAmount: 0,
value: _value,
joinDate: block.timestamp,
refundDate: 0,
refunded: false
})
);
totalVestedValue += _value;
totalVestedToken += _amount;
} else {
WhitelistInfo storage w = vestingPool.whitelistPool[whitelist.arrIdx];
totalVestedValue += _value - w.value;
totalVestedToken += _amount - w.amount;
w.amount = _amount;
w.value = _value;
}
emit SetWhitelist(_wallet, _amount, _value);
}
function _doBuybackAndBurn(uint256 amount) internal {
IUniswapV2Router02 _router = IUniswapV2Router02(router);
uint256[] memory amountsOut = _router.getAmountsOut(amount, path);
uint256 amountOut = (amountsOut[amountsOut.length - 1] * 99) / 100; //1% slippage
_router.swapExactTokensForTokens(amount, amountOut, path, address(0xdead), block.timestamp);
emit BuybackAndBurn(amount);
}
function getVestingInfo() public view returns (VestingInfo memory) {
return VestingInfo({
cliff: vestingPool.cliff,
start: vestingPool.start,
duration: vestingPool.duration,
initialUnlockPercent: vestingPool.initialUnlockPercent
});
}
function calculateVestAmount(address _wallet) internal view userInWhitelist(_wallet) returns (uint256 amount) {
uint256 idx = vestingPool.hasWhitelist[_wallet].arrIdx;
uint256 _amount = vestingPool.whitelistPool[idx].amount;
VestingPool storage vest = vestingPool;
if (block.timestamp < vest.start) {
return 0;
} else if (block.timestamp >= vest.start && block.timestamp < vest.cliff) {
return (_amount * vest.initialUnlockPercent / 1000);
} else if (block.timestamp >= vest.cliff) {
return calculateVestAmountForLinear(_amount, vest);
}
}
function calculateVestAmountForLinear(uint256 _amount, VestingPool storage vest) internal view returns (uint256) {
uint256 initial = _amount * vest.initialUnlockPercent / 1000;
uint256 remaining = _amount - initial;
if (block.timestamp >= vest.cliff + vest.duration) {
return _amount;
} else {
return initial + remaining * (block.timestamp - vest.cliff) / vest.duration;
}
}
function calculateReleasableAmount(address _wallet) internal view userInWhitelist(_wallet) returns (uint256) {
uint256 idx = vestingPool.hasWhitelist[_wallet].arrIdx;
return calculateVestAmount(_wallet) - (vestingPool.whitelistPool[idx].distributedAmount);
}
}// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts (last updated v4.7.0) (access/Ownable.sol)
pragma solidity ^0.8.0;
import "../utils/Context.sol";
/**
* @dev Contract module which provides a basic access control mechanism, where
* there is an account (an owner) that can be granted exclusive access to
* specific functions.
*
* By default, the owner account will be the one that deploys the contract. This
* can later be changed with {transferOwnership}.
*
* This module is used through inheritance. It will make available the modifier
* `onlyOwner`, which can be applied to your functions to restrict their use to
* the owner.
*/
abstract contract Ownable is Context {
address private _owner;
event OwnershipTransferred(address indexed previousOwner, address indexed newOwner);
/**
* @dev Initializes the contract setting the deployer as the initial owner.
*/
constructor() {
_transferOwnership(_msgSender());
}
/**
* @dev 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);
}
}// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts (last updated v4.8.0) (proxy/utils/Initializable.sol)
pragma solidity ^0.8.2;
import "../../utils/Address.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) || (!Address.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.8.0) (token/ERC20/ERC20.sol)
pragma solidity ^0.8.0;
import "./IERC20.sol";
import "./extensions/IERC20Metadata.sol";
import "../../utils/Context.sol";
/**
* @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.openzeppelin.com/t/how-to-implement-erc20-supply-mechanisms/226[How
* to implement supply mechanisms].
*
* The default value of {decimals} is 18. To change this, you should override
* this function so it returns a different value.
*
* We have followed general OpenZeppelin Contracts guidelines: functions revert
* instead 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, IERC20Metadata {
mapping(address => uint256) private _balances;
mapping(address => mapping(address => uint256)) private _allowances;
uint256 private _totalSupply;
string private _name;
string private _symbol;
/**
* @dev Sets the values for {name} and {symbol}.
*
* All two 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 name of the token.
*/
function name() public view virtual override returns (string memory) {
return _name;
}
/**
* @dev Returns the symbol of the token, usually a shorter version of the
* name.
*/
function symbol() public view virtual override returns (string memory) {
return _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 default value returned by this function, unless
* it's overridden.
*
* 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 override 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:
*
* - `to` cannot be the zero address.
* - the caller must have a balance of at least `amount`.
*/
function transfer(address to, uint256 amount) public virtual override returns (bool) {
address owner = _msgSender();
_transfer(owner, to, 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}.
*
* NOTE: If `amount` is the maximum `uint256`, the allowance is not updated on
* `transferFrom`. This is semantically equivalent to an infinite approval.
*
* Requirements:
*
* - `spender` cannot be the zero address.
*/
function approve(address spender, uint256 amount) public virtual override returns (bool) {
address owner = _msgSender();
_approve(owner, 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}.
*
* NOTE: Does not update the allowance if the current allowance
* is the maximum `uint256`.
*
* Requirements:
*
* - `from` and `to` cannot be the zero address.
* - `from` must have a balance of at least `amount`.
* - the caller must have allowance for ``from``'s tokens of at least
* `amount`.
*/
function transferFrom(address from, address to, uint256 amount) public virtual override returns (bool) {
address spender = _msgSender();
_spendAllowance(from, spender, amount);
_transfer(from, to, 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) {
address owner = _msgSender();
_approve(owner, spender, allowance(owner, 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) {
address owner = _msgSender();
uint256 currentAllowance = allowance(owner, spender);
require(currentAllowance >= subtractedValue, "ERC20: decreased allowance below zero");
unchecked {
_approve(owner, spender, currentAllowance - subtractedValue);
}
return true;
}
/**
* @dev Moves `amount` of tokens from `from` to `to`.
*
* This 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:
*
* - `from` cannot be the zero address.
* - `to` cannot be the zero address.
* - `from` must have a balance of at least `amount`.
*/
function _transfer(address from, address to, uint256 amount) internal virtual {
require(from != address(0), "ERC20: transfer from the zero address");
require(to != address(0), "ERC20: transfer to the zero address");
_beforeTokenTransfer(from, to, amount);
uint256 fromBalance = _balances[from];
require(fromBalance >= amount, "ERC20: transfer amount exceeds balance");
unchecked {
_balances[from] = fromBalance - amount;
// Overflow not possible: the sum of all balances is capped by totalSupply, and the sum is preserved by
// decrementing then incrementing.
_balances[to] += amount;
}
emit Transfer(from, to, amount);
_afterTokenTransfer(from, to, 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:
*
* - `account` cannot be the zero address.
*/
function _mint(address account, uint256 amount) internal virtual {
require(account != address(0), "ERC20: mint to the zero address");
_beforeTokenTransfer(address(0), account, amount);
_totalSupply += amount;
unchecked {
// Overflow not possible: balance + amount is at most totalSupply + amount, which is checked above.
_balances[account] += amount;
}
emit Transfer(address(0), account, amount);
_afterTokenTransfer(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");
_beforeTokenTransfer(account, address(0), amount);
uint256 accountBalance = _balances[account];
require(accountBalance >= amount, "ERC20: burn amount exceeds balance");
unchecked {
_balances[account] = accountBalance - amount;
// Overflow not possible: amount <= accountBalance <= totalSupply.
_totalSupply -= amount;
}
emit Transfer(account, address(0), amount);
_afterTokenTransfer(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);
}
/**
* @dev Updates `owner` s allowance for `spender` based on spent `amount`.
*
* Does not update the allowance amount in case of infinite allowance.
* Revert if not enough allowance is available.
*
* Might emit an {Approval} event.
*/
function _spendAllowance(address owner, address spender, uint256 amount) internal virtual {
uint256 currentAllowance = allowance(owner, spender);
if (currentAllowance != type(uint256).max) {
require(currentAllowance >= amount, "ERC20: insufficient allowance");
unchecked {
_approve(owner, spender, currentAllowance - amount);
}
}
}
/**
* @dev Hook that is called before any transfer of tokens. This includes
* minting and burning.
*
* Calling conditions:
*
* - when `from` and `to` are both non-zero, `amount` of ``from``'s tokens
* will be transferred to `to`.
* - when `from` is zero, `amount` tokens will be minted for `to`.
* - when `to` is zero, `amount` of ``from``'s tokens will be burned.
* - `from` and `to` are never both zero.
*
* To learn more about hooks, head to xref:ROOT:extending-contracts.adoc#using-hooks[Using Hooks].
*/
function _beforeTokenTransfer(address from, address to, uint256 amount) internal virtual {}
/**
* @dev Hook that is called after any transfer of tokens. This includes
* minting and burning.
*
* Calling conditions:
*
* - when `from` and `to` are both non-zero, `amount` of ``from``'s tokens
* has been transferred to `to`.
* - when `from` is zero, `amount` tokens have been minted for `to`.
* - when `to` is zero, `amount` of ``from``'s tokens have been burned.
* - `from` and `to` are never both zero.
*
* To learn more about hooks, head to xref:ROOT:extending-contracts.adoc#using-hooks[Using Hooks].
*/
function _afterTokenTransfer(address from, address to, uint256 amount) internal virtual {}
}// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts v4.4.1 (token/ERC20/extensions/IERC20Metadata.sol)
pragma solidity ^0.8.0;
import "../IERC20.sol";
/**
* @dev Interface for the optional metadata functions from the ERC20 standard.
*
* _Available since v4.1._
*/
interface IERC20Metadata is IERC20 {
/**
* @dev Returns the name of the token.
*/
function name() external view returns (string memory);
/**
* @dev Returns the symbol of the token.
*/
function symbol() external view returns (string memory);
/**
* @dev Returns the decimals places of the token.
*/
function decimals() external view returns (uint8);
}// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts v4.4.1 (token/ERC20/extensions/IERC20Permit.sol)
pragma solidity ^0.8.0;
/**
* @dev Interface of the ERC20 Permit extension allowing approvals to be made via signatures, as defined in
* https://eips.ethereum.org/EIPS/eip-2612[EIP-2612].
*
* Adds the {permit} method, which can be used to change an account's ERC20 allowance (see {IERC20-allowance}) by
* presenting a message signed by the account. By not relying on {IERC20-approve}, the token holder account doesn't
* need to send a transaction, and thus is not required to hold Ether at all.
*/
interface IERC20Permit {
/**
* @dev Sets `value` as the allowance of `spender` over ``owner``'s tokens,
* given ``owner``'s signed approval.
*
* IMPORTANT: The same issues {IERC20-approve} has related to transaction
* ordering also apply here.
*
* Emits an {Approval} event.
*
* Requirements:
*
* - `spender` cannot be the zero address.
* - `deadline` must be a timestamp in the future.
* - `v`, `r` and `s` must be a valid `secp256k1` signature from `owner`
* over the EIP712-formatted function arguments.
* - the signature must use ``owner``'s current nonce (see {nonces}).
*
* For more information on the signature format, see the
* https://eips.ethereum.org/EIPS/eip-2612#specification[relevant EIP
* section].
*/
function permit(
address owner,
address spender,
uint256 value,
uint256 deadline,
uint8 v,
bytes32 r,
bytes32 s
) external;
/**
* @dev Returns the current nonce for `owner`. This value must be
* included whenever a signature is generated for {permit}.
*
* Every successful call to {permit} increases ``owner``'s nonce by one. This
* prevents a signature from being used multiple times.
*/
function nonces(address owner) external view returns (uint256);
/**
* @dev Returns the domain separator used in the encoding of the signature for {permit}, as defined by {EIP712}.
*/
// solhint-disable-next-line func-name-mixedcase
function DOMAIN_SEPARATOR() external view returns (bytes32);
}// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts (last updated v4.6.0) (token/ERC20/IERC20.sol)
pragma solidity ^0.8.0;
/**
* @dev Interface of the ERC20 standard as defined in the EIP.
*/
interface IERC20 {
/**
* @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 Returns the amount of tokens in existence.
*/
function totalSupply() external view returns (uint256);
/**
* @dev Returns the amount of tokens owned by `account`.
*/
function balanceOf(address account) external view returns (uint256);
/**
* @dev Moves `amount` tokens from the caller's account to `to`.
*
* Returns a boolean value indicating whether the operation succeeded.
*
* Emits a {Transfer} event.
*/
function transfer(address to, uint256 amount) external returns (bool);
/**
* @dev Returns the remaining number of tokens that `spender` will be
* allowed to spend on behalf of `owner` through {transferFrom}. This is
* zero by default.
*
* This value changes when {approve} or {transferFrom} are called.
*/
function allowance(address owner, address spender) external view returns (uint256);
/**
* @dev Sets `amount` as the allowance of `spender` over the caller's tokens.
*
* Returns a boolean value indicating whether the operation succeeded.
*
* IMPORTANT: Beware that changing an allowance with this method brings the risk
* that someone may use both the old and the new allowance by unfortunate
* transaction ordering. One possible solution to mitigate this race
* condition is to first reduce the spender's allowance to 0 and set the
* desired value afterwards:
* https://github.com/ethereum/EIPs/issues/20#issuecomment-263524729
*
* Emits an {Approval} event.
*/
function approve(address spender, uint256 amount) external returns (bool);
/**
* @dev Moves `amount` tokens from `from` to `to` using the
* allowance mechanism. `amount` is then deducted from the caller's
* allowance.
*
* Returns a boolean value indicating whether the operation succeeded.
*
* Emits a {Transfer} event.
*/
function transferFrom(address from, address to, uint256 amount) external returns (bool);
}// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts (last updated v4.8.0) (token/ERC20/utils/SafeERC20.sol)
pragma solidity ^0.8.0;
import "../IERC20.sol";
import "../extensions/IERC20Permit.sol";
import "../../../utils/Address.sol";
/**
* @title SafeERC20
* @dev Wrappers around ERC20 operations that throw on failure (when the token
* contract returns false). Tokens that return no value (and instead revert or
* throw on failure) are also supported, non-reverting calls are assumed to be
* successful.
* To use this library you can add a `using SafeERC20 for IERC20;` statement to your contract,
* which allows you to call the safe operations as `token.safeTransfer(...)`, etc.
*/
library SafeERC20 {
using Address for address;
/**
* @dev Transfer `value` amount of `token` from the calling contract to `to`. If `token` returns no value,
* non-reverting calls are assumed to be successful.
*/
function safeTransfer(IERC20 token, address to, uint256 value) internal {
_callOptionalReturn(token, abi.encodeWithSelector(token.transfer.selector, to, value));
}
/**
* @dev Transfer `value` amount of `token` from `from` to `to`, spending the approval given by `from` to the
* calling contract. If `token` returns no value, non-reverting calls are assumed to be successful.
*/
function safeTransferFrom(IERC20 token, address from, address to, uint256 value) internal {
_callOptionalReturn(token, abi.encodeWithSelector(token.transferFrom.selector, from, to, value));
}
/**
* @dev Deprecated. This function has issues similar to the ones found in
* {IERC20-approve}, and its usage is discouraged.
*
* Whenever possible, use {safeIncreaseAllowance} and
* {safeDecreaseAllowance} instead.
*/
function safeApprove(IERC20 token, address spender, uint256 value) internal {
// safeApprove should only be called when setting an initial allowance,
// or when resetting it to zero. To increase and decrease it, use
// 'safeIncreaseAllowance' and 'safeDecreaseAllowance'
require(
(value == 0) || (token.allowance(address(this), spender) == 0),
"SafeERC20: approve from non-zero to non-zero allowance"
);
_callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, value));
}
/**
* @dev Increase the calling contract's allowance toward `spender` by `value`. If `token` returns no value,
* non-reverting calls are assumed to be successful.
*/
function safeIncreaseAllowance(IERC20 token, address spender, uint256 value) internal {
uint256 oldAllowance = token.allowance(address(this), spender);
_callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, oldAllowance + value));
}
/**
* @dev Decrease the calling contract's allowance toward `spender` by `value`. If `token` returns no value,
* non-reverting calls are assumed to be successful.
*/
function safeDecreaseAllowance(IERC20 token, address spender, uint256 value) internal {
unchecked {
uint256 oldAllowance = token.allowance(address(this), spender);
require(oldAllowance >= value, "SafeERC20: decreased allowance below zero");
_callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, oldAllowance - value));
}
}
/**
* @dev Set the calling contract's allowance toward `spender` to `value`. If `token` returns no value,
* non-reverting calls are assumed to be successful. Compatible with tokens that require the approval to be set to
* 0 before setting it to a non-zero value.
*/
function forceApprove(IERC20 token, address spender, uint256 value) internal {
bytes memory approvalCall = abi.encodeWithSelector(token.approve.selector, spender, value);
if (!_callOptionalReturnBool(token, approvalCall)) {
_callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, 0));
_callOptionalReturn(token, approvalCall);
}
}
/**
* @dev Use a ERC-2612 signature to set the `owner` approval toward `spender` on `token`.
* Revert on invalid signature.
*/
function safePermit(
IERC20Permit token,
address owner,
address spender,
uint256 value,
uint256 deadline,
uint8 v,
bytes32 r,
bytes32 s
) internal {
uint256 nonceBefore = token.nonces(owner);
token.permit(owner, spender, value, deadline, v, r, s);
uint256 nonceAfter = token.nonces(owner);
require(nonceAfter == nonceBefore + 1, "SafeERC20: permit did not succeed");
}
/**
* @dev Imitates a Solidity high-level call (i.e. a regular function call to a contract), relaxing the requirement
* on the return value: the return value is optional (but if data is returned, it must not be false).
* @param token The token targeted by the call.
* @param data The call data (encoded using abi.encode or one of its variants).
*/
function _callOptionalReturn(IERC20 token, bytes memory data) private {
// We need to perform a low level call here, to bypass Solidity's return data size checking mechanism, since
// we're implementing it ourselves. We use {Address-functionCall} to perform this call, which verifies that
// the target address contains contract code and also asserts for success in the low-level call.
bytes memory returndata = address(token).functionCall(data, "SafeERC20: low-level call failed");
require(returndata.length == 0 || abi.decode(returndata, (bool)), "SafeERC20: ERC20 operation did not succeed");
}
/**
* @dev Imitates a Solidity high-level call (i.e. a regular function call to a contract), relaxing the requirement
* on the return value: the return value is optional (but if data is returned, it must not be false).
* @param token The token targeted by the call.
* @param data The call data (encoded using abi.encode or one of its variants).
*
* This is a variant of {_callOptionalReturn} that silents catches all reverts and returns a bool instead.
*/
function _callOptionalReturnBool(IERC20 token, bytes memory data) private returns (bool) {
// We need to perform a low level call here, to bypass Solidity's return data size checking mechanism, since
// we're implementing it ourselves. We cannot use {Address-functionCall} here since this should return false
// and not revert is the subcall reverts.
(bool success, bytes memory returndata) = address(token).call(data);
return
success && (returndata.length == 0 || abi.decode(returndata, (bool))) && Address.isContract(address(token));
}
}// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts (last updated v4.8.0) (utils/Address.sol)
pragma solidity ^0.8.1;
/**
* @dev Collection of functions related to the address type
*/
library Address {
/**
* @dev Returns true if `account` is a contract.
*
* [IMPORTANT]
* ====
* It is unsafe to assume that an address for which this function returns
* false is an externally-owned account (EOA) and not a contract.
*
* Among others, `isContract` will return false for the following
* types of addresses:
*
* - an externally-owned account
* - a contract in construction
* - an address where a contract will be created
* - an address where a contract lived, but was destroyed
*
* 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.5.11/security-considerations.html#use-the-checks-effects-interactions-pattern[checks-effects-interactions pattern].
*/
function sendValue(address payable recipient, uint256 amount) internal {
require(address(this).balance >= amount, "Address: insufficient balance");
(bool success, ) = recipient.call{value: amount}("");
require(success, "Address: unable to send value, recipient may have reverted");
}
/**
* @dev Performs a Solidity function call using a low level `call`. A
* plain `call` is an unsafe replacement for a function call: use this
* function instead.
*
* If `target` reverts with a revert reason, it is bubbled up by this
* function (like regular Solidity function calls).
*
* Returns the raw returned data. To convert to the expected return value,
* use https://solidity.readthedocs.io/en/latest/units-and-global-variables.html?highlight=abi.decode#abi-encoding-and-decoding-functions[`abi.decode`].
*
* Requirements:
*
* - `target` must be a contract.
* - calling `target` with `data` must not revert.
*
* _Available since v3.1._
*/
function functionCall(address target, bytes memory data) internal returns (bytes memory) {
return 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 v4.4.1 (utils/Context.sol)
pragma solidity ^0.8.0;
/**
* @dev Provides information about the current execution context, including the
* sender of the transaction and its data. While these are generally available
* via msg.sender and msg.data, they should not be accessed in such a direct
* manner, since when dealing with meta-transactions the account sending and
* paying for execution may not be the actual sender (as far as an application
* is concerned).
*
* This contract is only required for intermediate, library-like contracts.
*/
abstract contract Context {
function _msgSender() internal view virtual returns (address) {
return msg.sender;
}
function _msgData() internal view virtual returns (bytes calldata) {
return msg.data;
}
}pragma solidity >=0.6.2;
interface IUniswapV2Router01 {
function factory() external pure returns (address);
function WETH() external pure returns (address);
function addLiquidity(
address tokenA,
address tokenB,
uint amountADesired,
uint amountBDesired,
uint amountAMin,
uint amountBMin,
address to,
uint deadline
) external returns (uint amountA, uint amountB, uint liquidity);
function addLiquidityETH(
address token,
uint amountTokenDesired,
uint amountTokenMin,
uint amountETHMin,
address to,
uint deadline
) external payable returns (uint amountToken, uint amountETH, uint liquidity);
function removeLiquidity(
address tokenA,
address tokenB,
uint liquidity,
uint amountAMin,
uint amountBMin,
address to,
uint deadline
) external returns (uint amountA, uint amountB);
function removeLiquidityETH(
address token,
uint liquidity,
uint amountTokenMin,
uint amountETHMin,
address to,
uint deadline
) external returns (uint amountToken, uint amountETH);
function removeLiquidityWithPermit(
address tokenA,
address tokenB,
uint liquidity,
uint amountAMin,
uint amountBMin,
address to,
uint deadline,
bool approveMax, uint8 v, bytes32 r, bytes32 s
) external returns (uint amountA, uint amountB);
function removeLiquidityETHWithPermit(
address token,
uint liquidity,
uint amountTokenMin,
uint amountETHMin,
address to,
uint deadline,
bool approveMax, uint8 v, bytes32 r, bytes32 s
) external returns (uint amountToken, uint amountETH);
function swapExactTokensForTokens(
uint amountIn,
uint amountOutMin,
address[] calldata path,
address to,
uint deadline
) external returns (uint[] memory amounts);
function swapTokensForExactTokens(
uint amountOut,
uint amountInMax,
address[] calldata path,
address to,
uint deadline
) external returns (uint[] memory amounts);
function swapExactETHForTokens(uint amountOutMin, address[] calldata path, address to, uint deadline)
external
payable
returns (uint[] memory amounts);
function swapTokensForExactETH(uint amountOut, uint amountInMax, address[] calldata path, address to, uint deadline)
external
returns (uint[] memory amounts);
function swapExactTokensForETH(uint amountIn, uint amountOutMin, address[] calldata path, address to, uint deadline)
external
returns (uint[] memory amounts);
function swapETHForExactTokens(uint amountOut, address[] calldata path, address to, uint deadline)
external
payable
returns (uint[] memory amounts);
function quote(uint amountA, uint reserveA, uint reserveB) external pure returns (uint amountB);
function getAmountOut(uint amountIn, uint reserveIn, uint reserveOut) external pure returns (uint amountOut);
function getAmountIn(uint amountOut, uint reserveIn, uint reserveOut) external pure returns (uint amountIn);
function getAmountsOut(uint amountIn, address[] calldata path) external view returns (uint[] memory amounts);
function getAmountsIn(uint amountOut, address[] calldata path) external view returns (uint[] memory amounts);
}pragma solidity >=0.6.2;
import './IUniswapV2Router01.sol';
interface IUniswapV2Router02 is IUniswapV2Router01 {
function removeLiquidityETHSupportingFeeOnTransferTokens(
address token,
uint liquidity,
uint amountTokenMin,
uint amountETHMin,
address to,
uint deadline
) external returns (uint amountETH);
function removeLiquidityETHWithPermitSupportingFeeOnTransferTokens(
address token,
uint liquidity,
uint amountTokenMin,
uint amountETHMin,
address to,
uint deadline,
bool approveMax, uint8 v, bytes32 r, bytes32 s
) external returns (uint amountETH);
function swapExactTokensForTokensSupportingFeeOnTransferTokens(
uint amountIn,
uint amountOutMin,
address[] calldata path,
address to,
uint deadline
) external;
function swapExactETHForTokensSupportingFeeOnTransferTokens(
uint amountOutMin,
address[] calldata path,
address to,
uint deadline
) external payable;
function swapExactTokensForETHSupportingFeeOnTransferTokens(
uint amountIn,
uint amountOutMin,
address[] calldata path,
address to,
uint deadline
) external;
}// SPDX-License-Identifier: MIT
//** DCB Vesting Interface */
pragma solidity 0.8.19;
interface IDCBPlatformVesting {
struct VestingInfo {
uint256 cliff;
uint256 start;
uint256 duration;
uint256 initialUnlockPercent;
}
struct VestingPool {
uint256 cliff;
uint256 start;
uint256 duration;
uint256 initialUnlockPercent;
WhitelistInfo[] whitelistPool;
mapping(address => HasWhitelist) hasWhitelist;
}
/**
*
* @dev WhiteInfo is the struct type which store whitelist information
*
*/
struct WhitelistInfo {
address wallet;
uint256 amount;
uint256 distributedAmount;
uint256 value; // price * amount in decimals of payment token
uint256 joinDate;
uint256 refundDate;
bool refunded;
}
struct HasWhitelist {
uint256 arrIdx;
bool active;
}
struct ContractSetup {
address _innovator;
address _paymentReceiver;
address _vestedToken;
address _paymentToken;
uint32 _nativeChainId;
uint256 _totalTokenOnSale;
uint256 _gracePeriod;
uint256[] _refundFees;
}
struct VestingSetup {
uint256 _startTime;
uint256 _cliff;
uint256 _duration;
uint256 _initialUnlockPercent;
}
struct BuybackSetup {
address router;
address[] path;
}
event Claim(address indexed token, uint256 amount, uint256 time);
event SetWhitelist(address indexed wallet, uint256 amount, uint256 value);
event Refund(address indexed wallet, uint256 amount);
function initializeCrowdfunding(ContractSetup memory c, VestingSetup memory p, BuybackSetup memory b) external;
function initializeTokenClaim(address _token, VestingSetup memory p, uint32 _nativeChainId) external;
function setCrowdfundingWhitelist(address _wallet, uint256 _amount, uint256 _value) external;
function setTokenClaimWhitelist(address _wallet, uint256 _amount) external;
function claimDistribution(address _wallet) external returns (bool);
function getWhitelist(address _wallet) external view returns (WhitelistInfo memory);
function getWhitelistPool() external view returns (WhitelistInfo[] memory);
function transferOwnership(address _newOwner) external;
function setVestingParams(
uint256 _cliff,
uint256 _start,
uint256 _duration,
uint256 _initialUnlockPercent
)
external;
function setCrowdFundingParams(ContractSetup calldata c, uint256 _platformFee) external;
function setToken(address _newToken) external;
function rescueTokens(address _receiver, uint256 _amount) external;
/**
*
* inherit functions will be used in contract
*
*/
function getVestAmount(address _wallet) external view returns (uint256);
function getReleasableAmount(address _wallet) external view returns (uint256);
function getVestingInfo() external view returns (VestingInfo memory);
}//SPDX-License-Identifier: UNLICENSED
pragma solidity 0.8.19;
contract DateTime {
/*
* Date and Time utilities for ethereum contracts
*
*/
struct Time {
uint16 year;
uint8 month;
uint8 day;
uint8 hour;
uint8 minute;
uint8 second;
uint8 weekday;
}
uint256 internal constant DAY_IN_SECONDS = 86_400;
uint256 internal constant YEAR_IN_SECONDS = 31_536_000;
uint256 internal constant LEAP_YEAR_IN_SECONDS = 31_622_400;
uint256 internal constant HOUR_IN_SECONDS = 3600;
uint256 internal constant MINUTE_IN_SECONDS = 60;
uint16 internal constant ORIGIN_YEAR = 1970;
function isLeapYear(uint16 year) public pure returns (bool) {
if (year % 4 != 0) {
return false;
}
if (year % 100 != 0) {
return true;
}
if (year % 400 != 0) {
return false;
}
return true;
}
function leapYearsBefore(uint256 year) public pure returns (uint256) {
year -= 1;
return year / 4 - year / 100 + year / 400;
}
function getDaysInMonth(uint8 month, uint16 year) public pure returns (uint8) {
if (month == 1 || month == 3 || month == 5 || month == 7 || month == 8 || month == 10 || month == 12) {
return 31;
} else if (month == 4 || month == 6 || month == 9 || month == 11) {
return 30;
} else if (isLeapYear(year)) {
return 29;
} else {
return 28;
}
}
function getYear(uint256 timestamp) public pure returns (uint16) {
uint256 secondsAccountedFor = 0;
uint16 year;
uint256 numLeapYears;
// Year
year = uint16(ORIGIN_YEAR + timestamp / YEAR_IN_SECONDS);
numLeapYears = leapYearsBefore(year) - leapYearsBefore(ORIGIN_YEAR);
secondsAccountedFor += LEAP_YEAR_IN_SECONDS * numLeapYears;
secondsAccountedFor += YEAR_IN_SECONDS * (year - ORIGIN_YEAR - numLeapYears);
while (secondsAccountedFor > timestamp) {
if (isLeapYear(uint16(year - 1))) {
secondsAccountedFor -= LEAP_YEAR_IN_SECONDS;
} else {
secondsAccountedFor -= YEAR_IN_SECONDS;
}
year -= 1;
}
return year;
}
function getMonth(uint256 timestamp) public pure returns (uint8) {
return parseTimestamp(timestamp).month;
}
function getDay(uint256 timestamp) public pure returns (uint8) {
return parseTimestamp(timestamp).day;
}
function getHour(uint256 timestamp) public pure returns (uint8) {
return uint8((timestamp / 60 / 60) % 24);
}
function getMinute(uint256 timestamp) public pure returns (uint8) {
return uint8((timestamp / 60) % 60);
}
function getSecond(uint256 timestamp) public pure returns (uint8) {
return uint8(timestamp % 60);
}
function getWeekday(uint256 timestamp) public pure returns (uint8) {
return uint8((timestamp / DAY_IN_SECONDS + 4) % 7);
}
function toTimestamp(uint16 year, uint8 month, uint8 day) public pure returns (uint256 timestamp) {
return toTimestamp(year, month, day, 0, 0, 0);
}
function toTimestamp(uint16 year, uint8 month, uint8 day, uint8 hour) public pure returns (uint256 timestamp) {
return toTimestamp(year, month, day, hour, 0, 0);
}
function toTimestamp(
uint16 year,
uint8 month,
uint8 day,
uint8 hour,
uint8 minute
)
public
pure
returns (uint256 timestamp)
{
return toTimestamp(year, month, day, hour, minute, 0);
}
function toTimestamp(
uint16 year,
uint8 month,
uint8 day,
uint8 hour,
uint8 minute,
uint8 second
)
public
pure
returns (uint256 timestamp)
{
uint16 i;
// Year
for (i = ORIGIN_YEAR; i < year; i++) {
if (isLeapYear(i)) {
timestamp += LEAP_YEAR_IN_SECONDS;
} else {
timestamp += YEAR_IN_SECONDS;
}
}
// Month
uint8[12] memory monthDayCounts;
monthDayCounts[0] = 31;
if (isLeapYear(year)) {
monthDayCounts[1] = 29;
} else {
monthDayCounts[1] = 28;
}
monthDayCounts[2] = 31;
monthDayCounts[3] = 30;
monthDayCounts[4] = 31;
monthDayCounts[5] = 30;
monthDayCounts[6] = 31;
monthDayCounts[7] = 31;
monthDayCounts[8] = 30;
monthDayCounts[9] = 31;
monthDayCounts[10] = 30;
monthDayCounts[11] = 31;
for (i = 1; i < month; i++) {
timestamp += DAY_IN_SECONDS * monthDayCounts[i - 1];
}
// Day
timestamp += DAY_IN_SECONDS * (day - 1);
// Hour
timestamp += HOUR_IN_SECONDS * (hour);
// Minute
timestamp += MINUTE_IN_SECONDS * (minute);
// Second
timestamp += second;
return timestamp;
}
function parseTimestamp(uint256 timestamp) internal pure returns (Time memory dt) {
uint256 secondsAccountedFor = 0;
uint256 buf;
uint8 i;
// Year
dt.year = getYear(timestamp);
buf = leapYearsBefore(dt.year) - leapYearsBefore(ORIGIN_YEAR);
secondsAccountedFor += LEAP_YEAR_IN_SECONDS * buf;
secondsAccountedFor += YEAR_IN_SECONDS * (dt.year - ORIGIN_YEAR - buf);
// Month
uint256 secondsInMonth;
for (i = 1; i <= 12; i++) {
secondsInMonth = DAY_IN_SECONDS * getDaysInMonth(i, dt.year);
if (secondsInMonth + secondsAccountedFor > timestamp) {
dt.month = i;
break;
}
secondsAccountedFor += secondsInMonth;
}
// Day
for (i = 1; i <= getDaysInMonth(dt.month, dt.year); i++) {
if (DAY_IN_SECONDS + secondsAccountedFor > timestamp) {
dt.day = i;
break;
}
secondsAccountedFor += DAY_IN_SECONDS;
}
// Hour
dt.hour = getHour(timestamp);
// Minute
dt.minute = getMinute(timestamp);
// Second
dt.second = getSecond(timestamp);
// Day of week.
dt.weekday = getWeekday(timestamp);
}
}{
"metadata": {
"bytecodeHash": "none"
},
"optimizer": {
"enabled": true,
"runs": 10000
},
"outputSelection": {
"*": {
"*": [
"evm.bytecode",
"evm.deployedBytecode",
"devdoc",
"userdoc",
"metadata",
"abi"
]
}
},
"remappings": [
"@prb/test/=lib/prb-math/lib/prb-test/src/",
"ds-test/=lib/forge-std/lib/ds-test/src/",
"erc4626-tests/=lib/openzeppelin-contracts/lib/erc4626-tests/",
"forge-std/=lib/forge-std/src/",
"openzeppelin-contracts-upgradeable/=lib/openzeppelin-contracts-upgradeable/",
"openzeppelin-contracts/=lib/openzeppelin-contracts/",
"src/=src/",
"v2-periphery/=lib/v2-periphery/contracts/",
"layerzero/=lib/LayerZero/contracts/",
"LayerZero/=lib/LayerZero/contracts/",
"prb-math/=lib/prb-math/src/",
"prb-test/=lib/prb-test/src/"
],
"libraries": {}
}Contract Security Audit
- No Contract Security Audit Submitted- Submit Audit Here
Contract ABI
API[{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"amount","type":"uint256"}],"name":"BuybackAndBurn","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"token","type":"address"},{"indexed":false,"internalType":"uint256","name":"amount","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"time","type":"uint256"}],"name":"Claim","type":"event"},{"anonymous":false,"inputs":[{"components":[{"internalType":"address","name":"_innovator","type":"address"},{"internalType":"address","name":"_paymentReceiver","type":"address"},{"internalType":"address","name":"_vestedToken","type":"address"},{"internalType":"address","name":"_paymentToken","type":"address"},{"internalType":"uint32","name":"_nativeChainId","type":"uint32"},{"internalType":"uint256","name":"_totalTokenOnSale","type":"uint256"},{"internalType":"uint256","name":"_gracePeriod","type":"uint256"},{"internalType":"uint256[]","name":"_refundFees","type":"uint256[]"}],"indexed":false,"internalType":"struct IDCBPlatformVesting.ContractSetup","name":"c","type":"tuple"}],"name":"CrowdFundingSet","type":"event"},{"anonymous":false,"inputs":[{"components":[{"internalType":"address","name":"_innovator","type":"address"},{"internalType":"address","name":"_paymentReceiver","type":"address"},{"internalType":"address","name":"_vestedToken","type":"address"},{"internalType":"address","name":"_paymentToken","type":"address"},{"internalType":"uint32","name":"_nativeChainId","type":"uint32"},{"internalType":"uint256","name":"_totalTokenOnSale","type":"uint256"},{"internalType":"uint256","name":"_gracePeriod","type":"uint256"},{"internalType":"uint256[]","name":"_refundFees","type":"uint256[]"}],"indexed":false,"internalType":"struct IDCBPlatformVesting.ContractSetup","name":"c","type":"tuple"},{"components":[{"internalType":"uint256","name":"_startTime","type":"uint256"},{"internalType":"uint256","name":"_cliff","type":"uint256"},{"internalType":"uint256","name":"_duration","type":"uint256"},{"internalType":"uint256","name":"_initialUnlockPercent","type":"uint256"}],"indexed":false,"internalType":"struct IDCBPlatformVesting.VestingSetup","name":"p","type":"tuple"},{"components":[{"internalType":"address","name":"router","type":"address"},{"internalType":"address[]","name":"path","type":"address[]"}],"indexed":false,"internalType":"struct IDCBPlatformVesting.BuybackSetup","name":"b","type":"tuple"}],"name":"CrowdfundingInitialized","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint8","name":"version","type":"uint8"}],"name":"Initialized","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":"uint256","name":"payment","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"remaining","type":"uint256"}],"name":"RaisedFundsClaimed","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"wallet","type":"address"},{"indexed":false,"internalType":"uint256","name":"amount","type":"uint256"}],"name":"Refund","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"_cliff","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"_start","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"_duration","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"_initialUnlockPercent","type":"uint256"}],"name":"SetVestingParams","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"wallet","type":"address"},{"indexed":false,"internalType":"uint256","name":"amount","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"value","type":"uint256"}],"name":"SetWhitelist","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"_token","type":"address"},{"components":[{"internalType":"uint256","name":"_startTime","type":"uint256"},{"internalType":"uint256","name":"_cliff","type":"uint256"},{"internalType":"uint256","name":"_duration","type":"uint256"},{"internalType":"uint256","name":"_initialUnlockPercent","type":"uint256"}],"indexed":false,"internalType":"struct IDCBPlatformVesting.VestingSetup","name":"p","type":"tuple"}],"name":"TokenClaimInitialized","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"_cliff","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"_start","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"_duration","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"_initialUnlockPercent","type":"uint256"}],"name":"VestingStrategyAdded","type":"event"},{"inputs":[],"name":"VERSION","outputs":[{"internalType":"uint8","name":"","type":"uint8"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_wallet","type":"address"}],"name":"claimDistribution","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"claimRaisedFunds","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"claimed","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"factory","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"timestamp","type":"uint256"}],"name":"getDay","outputs":[{"internalType":"uint8","name":"","type":"uint8"}],"stateMutability":"pure","type":"function"},{"inputs":[{"internalType":"uint8","name":"month","type":"uint8"},{"internalType":"uint16","name":"year","type":"uint16"}],"name":"getDaysInMonth","outputs":[{"internalType":"uint8","name":"","type":"uint8"}],"stateMutability":"pure","type":"function"},{"inputs":[{"internalType":"uint256","name":"timestamp","type":"uint256"}],"name":"getHour","outputs":[{"internalType":"uint8","name":"","type":"uint8"}],"stateMutability":"pure","type":"function"},{"inputs":[{"internalType":"uint256","name":"timestamp","type":"uint256"}],"name":"getMinute","outputs":[{"internalType":"uint8","name":"","type":"uint8"}],"stateMutability":"pure","type":"function"},{"inputs":[{"internalType":"uint256","name":"timestamp","type":"uint256"}],"name":"getMonth","outputs":[{"internalType":"uint8","name":"","type":"uint8"}],"stateMutability":"pure","type":"function"},{"inputs":[{"internalType":"address","name":"_wallet","type":"address"}],"name":"getReleasableAmount","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"timestamp","type":"uint256"}],"name":"getSecond","outputs":[{"internalType":"uint8","name":"","type":"uint8"}],"stateMutability":"pure","type":"function"},{"inputs":[{"internalType":"address","name":"_addr","type":"address"}],"name":"getTotalToken","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_wallet","type":"address"}],"name":"getVestAmount","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"getVestingInfo","outputs":[{"components":[{"internalType":"uint256","name":"cliff","type":"uint256"},{"internalType":"uint256","name":"start","type":"uint256"},{"internalType":"uint256","name":"duration","type":"uint256"},{"internalType":"uint256","name":"initialUnlockPercent","type":"uint256"}],"internalType":"struct IDCBPlatformVesting.VestingInfo","name":"","type":"tuple"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"timestamp","type":"uint256"}],"name":"getWeekday","outputs":[{"internalType":"uint8","name":"","type":"uint8"}],"stateMutability":"pure","type":"function"},{"inputs":[{"internalType":"address","name":"_wallet","type":"address"}],"name":"getWhitelist","outputs":[{"components":[{"internalType":"address","name":"wallet","type":"address"},{"internalType":"uint256","name":"amount","type":"uint256"},{"internalType":"uint256","name":"distributedAmount","type":"uint256"},{"internalType":"uint256","name":"value","type":"uint256"},{"internalType":"uint256","name":"joinDate","type":"uint256"},{"internalType":"uint256","name":"refundDate","type":"uint256"},{"internalType":"bool","name":"refunded","type":"bool"}],"internalType":"struct IDCBPlatformVesting.WhitelistInfo","name":"","type":"tuple"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"getWhitelistPool","outputs":[{"components":[{"internalType":"address","name":"wallet","type":"address"},{"internalType":"uint256","name":"amount","type":"uint256"},{"internalType":"uint256","name":"distributedAmount","type":"uint256"},{"internalType":"uint256","name":"value","type":"uint256"},{"internalType":"uint256","name":"joinDate","type":"uint256"},{"internalType":"uint256","name":"refundDate","type":"uint256"},{"internalType":"bool","name":"refunded","type":"bool"}],"internalType":"struct IDCBPlatformVesting.WhitelistInfo[]","name":"","type":"tuple[]"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"timestamp","type":"uint256"}],"name":"getYear","outputs":[{"internalType":"uint16","name":"","type":"uint16"}],"stateMutability":"pure","type":"function"},{"inputs":[],"name":"gracePeriod","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_wallet","type":"address"}],"name":"hasWhitelist","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"components":[{"internalType":"address","name":"_innovator","type":"address"},{"internalType":"address","name":"_paymentReceiver","type":"address"},{"internalType":"address","name":"_vestedToken","type":"address"},{"internalType":"address","name":"_paymentToken","type":"address"},{"internalType":"uint32","name":"_nativeChainId","type":"uint32"},{"internalType":"uint256","name":"_totalTokenOnSale","type":"uint256"},{"internalType":"uint256","name":"_gracePeriod","type":"uint256"},{"internalType":"uint256[]","name":"_refundFees","type":"uint256[]"}],"internalType":"struct IDCBPlatformVesting.ContractSetup","name":"c","type":"tuple"},{"components":[{"internalType":"uint256","name":"_startTime","type":"uint256"},{"internalType":"uint256","name":"_cliff","type":"uint256"},{"internalType":"uint256","name":"_duration","type":"uint256"},{"internalType":"uint256","name":"_initialUnlockPercent","type":"uint256"}],"internalType":"struct IDCBPlatformVesting.VestingSetup","name":"p","type":"tuple"},{"components":[{"internalType":"address","name":"router","type":"address"},{"internalType":"address[]","name":"path","type":"address[]"}],"internalType":"struct IDCBPlatformVesting.BuybackSetup","name":"b","type":"tuple"}],"name":"initializeCrowdfunding","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_token","type":"address"},{"components":[{"internalType":"uint256","name":"_startTime","type":"uint256"},{"internalType":"uint256","name":"_cliff","type":"uint256"},{"internalType":"uint256","name":"_duration","type":"uint256"},{"internalType":"uint256","name":"_initialUnlockPercent","type":"uint256"}],"internalType":"struct IDCBPlatformVesting.VestingSetup","name":"p","type":"tuple"},{"internalType":"uint32","name":"_nativeChainId","type":"uint32"}],"name":"initializeTokenClaim","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"innovator","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint16","name":"year","type":"uint16"}],"name":"isLeapYear","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"pure","type":"function"},{"inputs":[{"internalType":"uint256","name":"year","type":"uint256"}],"name":"leapYearsBefore","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"pure","type":"function"},{"inputs":[],"name":"owner","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"}],"name":"path","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"paymentReceiver","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"paymentToken","outputs":[{"internalType":"contract IERC20","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"platformFee","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"refund","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"}],"name":"refundFees","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"renounceOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_receiver","type":"address"},{"internalType":"uint256","name":"_amount","type":"uint256"}],"name":"rescueTokens","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"router","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"components":[{"internalType":"address","name":"_innovator","type":"address"},{"internalType":"address","name":"_paymentReceiver","type":"address"},{"internalType":"address","name":"_vestedToken","type":"address"},{"internalType":"address","name":"_paymentToken","type":"address"},{"internalType":"uint32","name":"_nativeChainId","type":"uint32"},{"internalType":"uint256","name":"_totalTokenOnSale","type":"uint256"},{"internalType":"uint256","name":"_gracePeriod","type":"uint256"},{"internalType":"uint256[]","name":"_refundFees","type":"uint256[]"}],"internalType":"struct IDCBPlatformVesting.ContractSetup","name":"c","type":"tuple"},{"internalType":"uint256","name":"_platformFee","type":"uint256"}],"name":"setCrowdFundingParams","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_wallet","type":"address"},{"internalType":"uint256","name":"_amount","type":"uint256"},{"internalType":"uint256","name":"_value","type":"uint256"}],"name":"setCrowdfundingWhitelist","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_token","type":"address"}],"name":"setToken","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_wallet","type":"address"},{"internalType":"uint256","name":"_amount","type":"uint256"}],"name":"setTokenClaimWhitelist","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_cliff","type":"uint256"},{"internalType":"uint256","name":"_start","type":"uint256"},{"internalType":"uint256","name":"_duration","type":"uint256"},{"internalType":"uint256","name":"_initialUnlockPercent","type":"uint256"}],"name":"setVestingParams","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint16","name":"year","type":"uint16"},{"internalType":"uint8","name":"month","type":"uint8"},{"internalType":"uint8","name":"day","type":"uint8"},{"internalType":"uint8","name":"hour","type":"uint8"},{"internalType":"uint8","name":"minute","type":"uint8"}],"name":"toTimestamp","outputs":[{"internalType":"uint256","name":"timestamp","type":"uint256"}],"stateMutability":"pure","type":"function"},{"inputs":[{"internalType":"uint16","name":"year","type":"uint16"},{"internalType":"uint8","name":"month","type":"uint8"},{"internalType":"uint8","name":"day","type":"uint8"},{"internalType":"uint8","name":"hour","type":"uint8"}],"name":"toTimestamp","outputs":[{"internalType":"uint256","name":"timestamp","type":"uint256"}],"stateMutability":"pure","type":"function"},{"inputs":[{"internalType":"uint16","name":"year","type":"uint16"},{"internalType":"uint8","name":"month","type":"uint8"},{"internalType":"uint8","name":"day","type":"uint8"}],"name":"toTimestamp","outputs":[{"internalType":"uint256","name":"timestamp","type":"uint256"}],"stateMutability":"pure","type":"function"},{"inputs":[{"internalType":"uint16","name":"year","type":"uint16"},{"internalType":"uint8","name":"month","type":"uint8"},{"internalType":"uint8","name":"day","type":"uint8"},{"internalType":"uint8","name":"hour","type":"uint8"},{"internalType":"uint8","name":"minute","type":"uint8"},{"internalType":"uint8","name":"second","type":"uint8"}],"name":"toTimestamp","outputs":[{"internalType":"uint256","name":"timestamp","type":"uint256"}],"stateMutability":"pure","type":"function"},{"inputs":[],"name":"totalRefunded","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"totalReturnedToken","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"totalTokenOnSale","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"totalVestedToken","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"totalVestedValue","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"newOwner","type":"address"}],"name":"transferOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"vestedToken","outputs":[{"internalType":"contract IERC20","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"vestingPool","outputs":[{"internalType":"uint256","name":"cliff","type":"uint256"},{"internalType":"uint256","name":"start","type":"uint256"},{"internalType":"uint256","name":"duration","type":"uint256"},{"internalType":"uint256","name":"initialUnlockPercent","type":"uint256"}],"stateMutability":"view","type":"function"}]Contract Creation Code
60806040523480156200001157600080fd5b506200001d3362000023565b62000073565b600080546001600160a01b038381166001600160a01b0319831681178455604051919092169283917f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e09190a35050565b61421e80620000836000396000f3fe608060405234801561001057600080fd5b50600436106103575760003560e01c80638da5cb5b116101c8578063c140240811610104578063e63318d5116100a2578063f887ea401161007c578063f887ea4014610770578063f95c81ff14610783578063fa93f88314610796578063ffa1ad74146107a957600080fd5b8063e63318d51461073d578063e834a83414610750578063f2fde38b1461075d57600080fd5b8063cb37f3b2116100de578063cb37f3b2146106d3578063d9082962146106e6578063d96eb767146106ef578063dc25a3001461070257600080fd5b8063c1402408146106a4578063c45a0155146106ad578063cb18583f146106c057600080fd5b8063a19d21b011610171578063af6d1fe41161014b578063af6d1fe414610663578063b199993714610676578063b238ad0e14610689578063b392972d1461069c57600080fd5b8063a19d21b01461062a578063a324ad241461063d578063a6f0e5771461065057600080fd5b80639054bdec116101a25780639054bdec146105e857806392d66313146105fb578063a06db7dc1461062157600080fd5b80638da5cb5b146105bb5780638dfd6a89146105cc5780638e25805b146105df57600080fd5b8063583c321d11610297578063669b0529116102405780637f7918331161021a5780637f791833146105535780638aa001fc146105665780638c8d98a0146105795780638cdb7e8b1461058c57600080fd5b8063669b05291461052f5780636c4a844314610538578063715018a61461054b57600080fd5b806362015bdc1161027157806362015bdc146104e657806362ba96871461050957806365c728401461051c57600080fd5b8063583c321d146104b0578063590e1ae3146104c35780635de29741146104cb57600080fd5b80633e239e1a116103045780634ac1ad78116102de5780634ac1ad7814610462578063522ce386146104755780635357a94a1461048a578063573761981461049d57600080fd5b80633e239e1a146104215780633e5ce385146104465780634344f6c51461045957600080fd5b80633013ce29116103355780633013ce29146103a057806330edc0f5146103cb5780633154b10e146103eb57600080fd5b8063144fa6d71461035c57806326232a2e146103715780632afd1a7d1461038d575b600080fd5b61036f61036a3660046135da565b6107b1565b005b61037a600c5481565b6040519081526020015b60405180910390f35b61037a61039b3660046135da565b610852565b6015546103b3906001600160a01b031681565b6040516001600160a01b039091168152602001610384565b6103de6103d93660046135da565b610863565b60405161038491906135f5565b6001546002546003546004546104019392919084565b604080519485526020850193909352918301526060820152608001610384565b61043461042f36600461364d565b6109bf565b60405160ff9091168152602001610384565b61036f610454366004613666565b6109e3565b61037a600b5481565b61043461047036600461364d565b610a8d565b61047d610aa9565b6040516103849190613699565b600e546103b3906001600160a01b031681565b61036f6104ab366004613731565b610b59565b61037a6104be36600461364d565b610d46565b61036f610d67565b6014546103b39064010000000090046001600160a01b031681565b6104f96104f43660046135da565b611153565b6040519015158152602001610384565b61037a61051736600461377c565b6112ca565b61043461052a36600461364d565b6112e5565b61037a60075481565b61036f610546366004613811565b6112fa565b61036f6115ca565b61037a610561366004613855565b6115de565b61043461057436600461364d565b6115fa565b61037a6105873660046138af565b611607565b6104f961059a3660046135da565b6001600160a01b031660009081526006602052604090206001015460ff1690565b6000546001600160a01b03166103b3565b61036f6105da3660046138f8565b611623565b61037a600a5481565b61037a6105f636600461392a565b611720565b61060e61060936600461364d565b6118d1565b60405161ffff9091168152602001610384565b61037a600d5481565b61037a6106383660046135da565b6119c1565b61043461064b36600461364d565b611a47565b6104f961065e3660046139aa565b611a5c565b6103b361067136600461364d565b611abb565b61037a61068436600461364d565b611ae5565b6104346106973660046139c5565b611b2a565b61036f611bf1565b61037a60095481565b6016546103b3906001600160a01b031681565b61036f6106ce366004613731565b611e36565b600f546103b3906001600160a01b031681565b61037a60085481565b61036f6106fd366004613a0d565b611eba565b61070a612359565b60405161038491908151815260208083015190820152604080830151908201526060918201519181019190915260800190565b61037a61074b3660046135da565b6123b1565b6012546104f99060ff1681565b61036f61076b3660046135da565b6123bc565b6010546103b3906001600160a01b031681565b61036f610791366004613a7e565b6123d0565b6104346107a436600461364d565b6126bd565b610434600181565b6016546001600160a01b031633146108105760405162461bcd60e51b815260206004820152600c60248201527f4f6e6c7920666163746f7279000000000000000000000000000000000000000060448201526064015b60405180910390fd5b601480546001600160a01b03909216640100000000027fffffffffffffffff0000000000000000000000000000000000000000ffffffff909216919091179055565b600061085d826126cb565b92915050565b6108ae6040518060e0016040528060006001600160a01b0316815260200160008152602001600081526020016000815260200160008152602001600081526020016000151581525090565b6001600160a01b038216600090815260066020526040902060010154829060ff1661091b5760405162461bcd60e51b815260206004820152601060248201527f4e6f7420696e2077686974656c697374000000000000000000000000000000006044820152606401610807565b6001600160a01b038316600090815260066020526040902054600580548290811061094857610948613ac3565b60009182526020918290206040805160e081018252600790930290910180546001600160a01b031683526001810154938301939093526002830154908201526003820154606082015260048201546080820152600582015460a082015260069091015460ff16151560c08201529250505b50919050565b60006018603c6109cf8185613b50565b6109d99190613b50565b61085d9190613b64565b6109eb61278a565b6001600160a01b03831660009081526006602052604081206001015460ff1615610a5f576001600160a01b038416600090815260066020526040902054600580549091908110610a3d57610a3d613ac3565b90600052602060002090600702016003015482610a5a9190613b78565b610a61565b815b601554909150610a7c906001600160a01b03168530846127e4565b610a878484846128b3565b50505050565b60006007610a9e6201518084613b50565b6109d9906004613b8b565b60606001600401805480602002602001604051908101604052809291908181526020016000905b82821015610b505760008481526020908190206040805160e0810182526007860290920180546001600160a01b03168352600180820154848601526002820154928401929092526003810154606084015260048101546080840152600581015460a08401526006015460ff16151560c08301529083529092019101610ad0565b50505050905090565b6016546001600160a01b03163314610bb35760405162461bcd60e51b815260206004820152600c60248201527f4f6e6c7920666163746f727900000000000000000000000000000000000000006044820152606401610807565b600254421080610c5c57506009546014546040517f70a0823100000000000000000000000000000000000000000000000000000000815230600482015283929164010000000090046001600160a01b0316906370a0823190602401602060405180830381865afa158015610c2b573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610c4f9190613b9e565b610c599190613b78565b10155b610ca85760405162461bcd60e51b815260206004820152600e60248201527f496e76616c696420616d6f756e740000000000000000000000000000000000006044820152606401610807565b6014546040517fa9059cbb0000000000000000000000000000000000000000000000000000000081526001600160a01b038481166004830152602482018490526401000000009092049091169063a9059cbb906044016020604051808303816000875af1158015610d1d573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610d419190613bc7565b505050565b60138181548110610d5657600080fd5b600091825260209091200154905081565b3360008181526006602052604090206001015460ff16610dc95760405162461bcd60e51b815260206004820152601060248201527f4e6f7420696e2077686974656c697374000000000000000000000000000000006044820152606401610807565b336000908152600660205260408120546005805491929183908110610df057610df0613ac3565b90600052602060002090600702019050600d546001800154610e129190613b8b565b42108015610e21575060025442115b610e6d5760405162461bcd60e51b815260206004820152601360248201527f4e6f7420696e20677261636520706572696f64000000000000000000000000006044820152606401610807565b600681015460ff1615610ec25760405162461bcd60e51b815260206004820152601560248201527f7573657220616c726561647920726566756e64656400000000000000000000006044820152606401610807565b600281015415610f145760405162461bcd60e51b815260206004820152601460248201527f7573657220616c726561647920636c61696d65640000000000000000000000006044820152606401610807565b600080610f296000546001600160a01b031690565b6040517f1a0cb3a70000000000000000000000000000000000000000000000000000000081523360048201526001600160a01b039190911690631a0cb3a79060240160a060405180830381865afa158015610f88573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610fac9190613be2565b50505060ff16915060ff169150600060138381548110610fce57610fce613ac3565b90600052602060002001549050600182111561101a576000610ff1600184613b78565b610ffc906032613c3f565b905080821161100c576000611016565b6110168183613b78565b9150505b600061271082866003015461102f9190613c3f565b6110399190613b50565b9050600081866003015461104d9190613b78565b6006870180547fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff00166001179055426005880155600387015460088054929350909160009061109c908490613b8b565b90915550506001860154600a80546000906110b8908490613b8b565b90915550506015546110d4906001600160a01b03163383612ba3565b81156111145760145463ffffffff1646036110f7576110f282612bec565b611114565b600f54601554611114916001600160a01b03918216911684612ba3565b60405181815233907fbb28353e4598c3b9199101a66e0989549b659a59a54d2c27fbb183f1932c8e6d9060200160405180910390a25050505050505050565b6001600160a01b038116600090815260066020526040812054600580548391908390811061118357611183613ac3565b60009182526020909120600790910201600681015490915060ff16156111eb5760405162461bcd60e51b815260206004820152601560248201527f7573657220616c726561647920726566756e64656400000000000000000000006044820152606401610807565b60006111f6856126cb565b9050600081116112485760405162461bcd60e51b815260206004820152600b60248201527f5a65726f20616d6f756e740000000000000000000000000000000000000000006044820152606401610807565b8082600201546112589190613b8b565b600283015560145461127c9064010000000090046001600160a01b03168683612ba3565b604080518281524260208201526001600160a01b038716917f34fcbac0073d7c3d388e51312faf357774904998eeb8fca628b9e6f65ee1cbf7910160405180910390a2506001949350505050565b60006112db86868686866000611720565b9695505050505050565b60006112f082612dcc565b6040015192915050565b6000547501000000000000000000000000000000000000000000900460ff1615808015611345575060005460017401000000000000000000000000000000000000000090910460ff16105b806113775750303b158015611377575060005474010000000000000000000000000000000000000000900460ff166001145b6113e95760405162461bcd60e51b815260206004820152602e60248201527f496e697469616c697a61626c653a20636f6e747261637420697320616c72656160448201527f647920696e697469616c697a65640000000000000000000000000000000000006064820152608401610807565b600080547fffffffffffffffffffffff00ffffffffffffffffffffffffffffffffffffffff1674010000000000000000000000000000000000000000179055801561146f57600080547fffffffffffffffffffff00ffffffffffffffffffffffffffffffffffffffffff1675010000000000000000000000000000000000000000001790555b601480547fffffffffffffffff0000000000000000000000000000000000000000ffffffff166401000000006001600160a01b038716021790556114b233612fb4565b601680547fffffffffffffffffffffffff000000000000000000000000000000000000000016331790556014805463ffffffff84167fffffffffffffffffffffffffffffffffffffffffffffffffffffffff00000000909116179055611527602084013584356040860135606087013561301c565b507f702283685135875ce459087727893f6d49d6fe02203812686d668fcb5812e9918484604051611559929190613c56565b60405180910390a18015610a8757600080547fffffffffffffffffffff00ffffffffffffffffffffffffffffffffffffffffff169055604051600181527f7f26b83ff96e1f2b6a682f133852f6798a09c465da95921460cefb3847402498906020015b60405180910390a150505050565b6115d261278a565b6115dc6000612fb4565b565b60006115ef85858585600080611720565b90505b949350505050565b600061085d603c83613b64565b60006116198484846000806000611720565b90505b9392505050565b6016546001600160a01b0316331461167d5760405162461bcd60e51b815260206004820152600c60248201527f4f6e6c7920666163746f727900000000000000000000000000000000000000006044820152606401610807565b60025442106116ce5760405162461bcd60e51b815260206004820152601760248201527f56657374696e6720616c726561647920737461727465640000000000000000006044820152606401610807565b6116da8484848461301c565b506040805185815260208101859052908101839052606081018290527feb4c1ae991ee614a820b5fdac6055e11ea335582cbf83319f7fe59dee3178a04906080016115bc565b60006107b25b8761ffff168161ffff16101561177c5761173f81611a5c565b15611759576117526301e2850083613b8b565b915061176a565b6117676301e1338083613b8b565b91505b8061177481613c94565b915050611726565b6117846134d4565b601f815261179189611a5c565b156117a257601d60208201526117aa565b601c60208201525b601f60408201819052601e606083018190526080830182905260a0830181905260c0830182905260e0830182905261010083018190526101208301829052610140830152610160820152600191505b8760ff168261ffff16101561185b5780611814600184613cb5565b61ffff16600c811061182857611828613ac3565b602002015161183d9060ff1662015180613c3f565b6118479084613b8b565b92508161185381613c94565b9250506117f9565b611866600188613cd7565b6118769060ff1662015180613c3f565b6118809084613b8b565b925061189160ff8716610e10613c3f565b61189b9084613b8b565b92506118ab60ff8616603c613c3f565b6118b59084613b8b565b92506118c460ff851684613b8b565b9998505050505050505050565b60008080806118e46301e1338086613b50565b6118f0906107b2613b8b565b91506118fd6107b2611ae5565b61190a8361ffff16611ae5565b6119149190613b78565b9050611924816301e28500613c3f565b61192e9084613b8b565b92508061193d6107b284613cb5565b61ffff1661194b9190613b78565b611959906301e13380613c3f565b6119639084613b8b565b92505b848311156119b95761197c61065e600184613cb5565b156119965761198f6301e2850084613b78565b92506119a7565b6119a46301e1338084613b78565b92505b6119b2600183613cb5565b9150611966565b509392505050565b6040517f70a0823100000000000000000000000000000000000000000000000000000000815230600482015260009082906001600160a01b038216906370a0823190602401602060405180830381865afa158015611a23573d6000803e3d6000fd5b505050506040513d601f19601f8201168201806040525081019061161c9190613b9e565b6000611a5282612dcc565b6020015192915050565b6000611a69600483613cf0565b61ffff1615611a7a57506000919050565b611a85606483613cf0565b61ffff1615611a9657506001919050565b611aa261019083613cf0565b61ffff1615611ab357506000919050565b506001919050565b60118181548110611acb57600080fd5b6000918252602090912001546001600160a01b0316905081565b6000611af2600183613b78565b9150611b0061019083613b50565b611b0b606484613b50565b611b16600485613b50565b611b209190613b78565b61085d9190613b8b565b60008260ff1660011480611b4157508260ff166003145b80611b4f57508260ff166005145b80611b5d57508260ff166007145b80611b6b57508260ff166008145b80611b7957508260ff16600a145b80611b8757508260ff16600c145b15611b945750601f61085d565b8260ff1660041480611ba957508260ff166006145b80611bb757508260ff166009145b80611bc557508260ff16600b145b15611bd25750601e61085d565b611bdb82611a5c565b15611be85750601d61085d565b50601c92915050565b600e546001600160a01b03163314611c4b5760405162461bcd60e51b815260206004820152600e60248201527f496e76616c6964206163636573730000000000000000000000000000000000006044820152606401610807565b600254600d54611c5b9190613b8b565b4211611ca95760405162461bcd60e51b815260206004820152601860248201527f677261636520706572696f6420696e2070726f677265737300000000000000006044820152606401610807565b60125460ff1615611cfc5760405162461bcd60e51b815260206004820152600f60248201527f616c726561647920636c61696d656400000000000000000000000000000000006044820152606401610807565b6000600854600754611d0e9190613b78565b90506000612710600c5483611d239190613c3f565b611d2d9190613b50565b9050611d398183613b78565b91506000600a54600954600b54611d509190613b78565b611d5a9190613b8b565b601280547fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff0016600117905590508215611daa57600e54601554611daa916001600160a01b03918216911685612ba3565b8015611dd557600e54601454611dd5916001600160a01b036401000000009092048216911683612ba3565b8115611df857600f54601554611df8916001600160a01b03918216911684612ba3565b60408051848152602081018390527f6414532af447bc814fac232567913fadd11eb1714ee73e08ab1f005a893a9a00910160405180910390a1505050565b611e3e61278a565b6001600160a01b03821660009081526006602052604090206001015460ff1615611eaa5760405162461bcd60e51b815260206004820152601260248201527f416c7265616479207265676973746572656400000000000000000000000000006044820152606401610807565b611eb6828260006128b3565b5050565b6000547501000000000000000000000000000000000000000000900460ff1615808015611f05575060005460017401000000000000000000000000000000000000000090910460ff16105b80611f375750303b158015611f37575060005474010000000000000000000000000000000000000000900460ff166001145b611fa95760405162461bcd60e51b815260206004820152602e60248201527f496e697469616c697a61626c653a20636f6e747261637420697320616c72656160448201527f647920696e697469616c697a65640000000000000000000000000000000000006064820152608401610807565b600080547fffffffffffffffffffffff00ffffffffffffffffffffffffffffffffffffffff1674010000000000000000000000000000000000000000179055801561202f57600080547fffffffffffffffffffff00ffffffffffffffffffffffffffffffffffffffffff1675010000000000000000000000000000000000000000001790555b61203c60208501856135da565b600e80547fffffffffffffffffffffffff0000000000000000000000000000000000000000166001600160a01b039290921691909117905561208460408501602086016135da565b600f80547fffffffffffffffffffffffff0000000000000000000000000000000000000000166001600160a01b03929092169190911790556120cc60608501604086016135da565b601480546001600160a01b0392909216640100000000027fffffffffffffffff0000000000000000000000000000000000000000ffffffff90921691909117905561211d60808501606086016135da565b601580547fffffffffffffffffffffffff0000000000000000000000000000000000000000166001600160a01b039290921691909117905560c0840135600d5560a084018035600b556121739060808601613d11565b601480547fffffffffffffffffffffffffffffffffffffffffffffffffffffffff000000001663ffffffff929092169190911790556121b560e0850185613d2c565b6121c1916013916134f3565b506121cf60208301836135da565b601080547fffffffffffffffffffffffff0000000000000000000000000000000000000000166001600160a01b03929092169190911790556122146020830183613d2c565b6122209160119161353e565b506015546010546040517f095ea7b30000000000000000000000000000000000000000000000000000000081526001600160a01b0391821660048201527fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff602482015291169063095ea7b3906044016020604051808303816000875af11580156122ae573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906122d29190613bc7565b506122dc33612fb4565b601680547fffffffffffffffffffffffff00000000000000000000000000000000000000001633179055610320600c55612325602084013584356040860135606087013561301c565b507f60539c54c534813a30c613b1b79c550f4b2e23341cea11cea6d0732e7038e15c84848460405161155993929190613eec565b6123846040518060800160405280600081526020016000815260200160008152602001600081525090565b50604080516080810182526001548152600254602082015260035491810191909152600454606082015290565b600061085d8261308d565b6123c461278a565b6123cd816131b3565b50565b6016546001600160a01b0316331461242a5760405162461bcd60e51b815260206004820152600c60248201527f4f6e6c7920666163746f727900000000000000000000000000000000000000006044820152606401610807565b600254421061247b5760405162461bcd60e51b815260206004820152601760248201527f56657374696e6720616c726561647920737461727465640000000000000000006044820152606401610807565b61248860208301836135da565b600e80547fffffffffffffffffffffffff0000000000000000000000000000000000000000166001600160a01b03929092169190911790556124d060408301602084016135da565b600f80547fffffffffffffffffffffffff0000000000000000000000000000000000000000166001600160a01b039290921691909117905561251860608301604084016135da565b601480546001600160a01b0392909216640100000000027fffffffffffffffff0000000000000000000000000000000000000000ffffffff90921691909117905561256960808301606084016135da565b601580547fffffffffffffffffffffffff0000000000000000000000000000000000000000166001600160a01b039290921691909117905560c0820135600d5560a0820135600b556125be60e0830183613d2c565b6125ca916013916134f3565b50600c8190556015546010546040517f095ea7b30000000000000000000000000000000000000000000000000000000081526001600160a01b0391821660048201527fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff602482015291169063095ea7b3906044016020604051808303816000875af115801561265d573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906126819190613bc7565b507f69a08d1ff0e6d976f9c703dccabe21e76e5484a8b7652f0190bc587d37550e0e826040516126b19190613fa5565b60405180910390a15050565b6000603c6109d98184613b50565b6001600160a01b038116600090815260066020526040812060010154829060ff166127385760405162461bcd60e51b815260206004820152601060248201527f4e6f7420696e2077686974656c697374000000000000000000000000000000006044820152606401610807565b6001600160a01b038316600090815260066020526040902054600580548290811061276557612765613ac3565b9060005260206000209060070201600201546127808561308d565b6115f29190613b78565b6000546001600160a01b031633146115dc5760405162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e65726044820152606401610807565b6040516001600160a01b0380851660248301528316604482015260648101829052610a879085907f23b872dd00000000000000000000000000000000000000000000000000000000906084015b604080517fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe08184030181529190526020810180517bffffffffffffffffffffffffffffffffffffffffffffffffffffffff167fffffffff0000000000000000000000000000000000000000000000000000000090931692909217909152613240565b6001600160a01b0383166000908152600660205260409020600181015460ff16612ad857600181810180547fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff009081168317909155600580548085556040805160e0810182526001600160a01b038a81168252602082018a81526000938301848152606084018b8152426080860190815260a0860187815260c087018881529b89018a5598875294517f036b6384b5eca791c62761152d0c79bb0604c104a5fb6f4eb0703f3154bb3db0600798890290810180547fffffffffffffffffffffffff000000000000000000000000000000000000000016929096169190911790945591517f036b6384b5eca791c62761152d0c79bb0604c104a5fb6f4eb0703f3154bb3db1840155517f036b6384b5eca791c62761152d0c79bb0604c104a5fb6f4eb0703f3154bb3db2830155517f036b6384b5eca791c62761152d0c79bb0604c104a5fb6f4eb0703f3154bb3db382015590517f036b6384b5eca791c62761152d0c79bb0604c104a5fb6f4eb0703f3154bb3db482015592517f036b6384b5eca791c62761152d0c79bb0604c104a5fb6f4eb0703f3154bb3db584015593517f036b6384b5eca791c62761152d0c79bb0604c104a5fb6f4eb0703f3154bb3db69092018054909316911515919091179091558054849290612ab4908490613b8b565b925050819055508260096000828254612acd9190613b8b565b90915550612b599050565b805460058054600092908110612af057612af0613ac3565b90600052602060002090600702019050806003015483612b109190613b78565b60076000828254612b219190613b8b565b90915550506001810154612b359085613b78565b60096000828254612b469190613b8b565b9091555050600181018490556003018290555b60408051848152602081018490526001600160a01b038616917fdb76cc8d16d8e51a4a8b73f914539ac9f9e4e42ff5f0140ddd580c48e8947798910160405180910390a250505050565b6040516001600160a01b038316602482015260448101829052610d419084907fa9059cbb0000000000000000000000000000000000000000000000000000000090606401612831565b6010546040517fd06ca61f0000000000000000000000000000000000000000000000000000000081526001600160a01b0390911690600090829063d06ca61f90612c3d908690601190600401614001565b600060405180830381865afa158015612c5a573d6000803e3d6000fd5b505050506040513d6000823e601f3d9081017fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe0168201604052612ca09190810190614049565b9050600060648260018451612cb59190613b78565b81518110612cc557612cc5613ac3565b60200260200101516063612cd99190613c3f565b612ce39190613b50565b6040517f38ed17390000000000000000000000000000000000000000000000000000000081529091506001600160a01b038416906338ed173990612d36908790859060119061dead904290600401614125565b6000604051808303816000875af1158015612d55573d6000803e3d6000fd5b505050506040513d6000823e601f3d9081017fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe0168201604052612d9b9190810190614049565b506040518481527ffcfa54313b92fceed71362d6bfec038726e592d2fa73d69c461986eeedf74400906020016115bc565b6040805160e081018252600080825260208201819052918101829052606081018290526080810182905260a0810182905260c08101829052908080612e10856118d1565b61ffff168452612e216107b2611ae5565b8451612e309061ffff16611ae5565b612e3a9190613b78565b9150612e4a826301e28500613c3f565b612e549084613b8b565b9250816107b28560000151612e699190613cb5565b61ffff16612e779190613b78565b612e85906301e13380613c3f565b612e8f9084613b8b565b92506000600191505b600c8260ff1611612f0057612eb1828660000151611b2a565b612ec19060ff1662015180613c3f565b905085612ece8583613b8b565b1115612ee25760ff82166020860152612f00565b612eec8185613b8b565b935081612ef881614161565b925050612e98565b600191505b612f1785602001518660000151611b2a565b60ff168260ff1611612f665785612f318562015180613b8b565b1115612f455760ff82166040860152612f66565b612f526201518085613b8b565b935081612f5e81614161565b925050612f05565b612f6f866109bf565b60ff166060860152612f80866126bd565b60ff166080860152612f91866115fa565b60ff1660a0860152612fa286610a8d565b60ff1660c08601525092949350505050565b600080546001600160a01b038381167fffffffffffffffffffffffff0000000000000000000000000000000000000000831681178455604051919092169283917f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e09190a35050565b60006130288585613b8b565b6001556002849055600383905560048290556040805186815260208101869052908101849052606081018390527f8b2bfe599bf9e0b744a947eec87d2b85b086f600a3c5f8901ca919b9c30f034b9060800160405180910390a1506001949350505050565b6001600160a01b038116600090815260066020526040812060010154829060ff166130fa5760405162461bcd60e51b815260206004820152601060248201527f4e6f7420696e2077686974656c697374000000000000000000000000000000006044820152606401610807565b6001600160a01b038316600090815260066020526040812054600580549192918390811061312a5761312a613ac3565b60009182526020909120600160079092020181015460025490925042101561315857600094505050506109b9565b8060010154421015801561316c5750805442105b15613198576103e88160030154836131849190613c3f565b61318e9190613b50565b94505050506109b9565b805442106131aa5761318e8282613328565b50505050919050565b6131bb61278a565b6001600160a01b0381166132375760405162461bcd60e51b815260206004820152602660248201527f4f776e61626c653a206e6577206f776e657220697320746865207a65726f206160448201527f64647265737300000000000000000000000000000000000000000000000000006064820152608401610807565b6123cd81612fb4565b6000613295826040518060400160405280602081526020017f5361666545524332303a206c6f772d6c6576656c2063616c6c206661696c6564815250856001600160a01b03166133b19092919063ffffffff16565b90508051600014806132b65750808060200190518101906132b69190613bc7565b610d415760405162461bcd60e51b815260206004820152602a60248201527f5361666545524332303a204552433230206f7065726174696f6e20646964206e60448201527f6f742073756363656564000000000000000000000000000000000000000000006064820152608401610807565b6000806103e883600301548561333e9190613c3f565b6133489190613b50565b905060006133568286613b78565b6002850154855491925061336991613b8b565b421061337957849250505061085d565b6002840154845461338a9042613b78565b6133949083613c3f565b61339e9190613b50565b6133a89083613b8b565b9250505061085d565b6060611619848460008585600080866001600160a01b031685876040516133d891906141a4565b60006040518083038185875af1925050503d8060008114613415576040519150601f19603f3d011682016040523d82523d6000602084013e61341a565b606091505b509150915061342b87838387613436565b979650505050505050565b606083156134a557825160000361349e576001600160a01b0385163b61349e5760405162461bcd60e51b815260206004820152601d60248201527f416464726573733a2063616c6c20746f206e6f6e2d636f6e74726163740000006044820152606401610807565b50816115f2565b6115f283838151156134ba5781518083602001fd5b8060405162461bcd60e51b815260040161080791906141c0565b604051806101800160405280600c906020820280368337509192915050565b82805482825590600052602060002090810192821561352e579160200282015b8281111561352e578235825591602001919060010190613513565b5061353a9291506135a9565b5090565b82805482825590600052602060002090810192821561352e579160200282015b8281111561352e5781547fffffffffffffffffffffffff0000000000000000000000000000000000000000166001600160a01b0384351617825560209092019160019091019061355e565b5b8082111561353a57600081556001016135aa565b80356001600160a01b03811681146135d557600080fd5b919050565b6000602082840312156135ec57600080fd5b61161c826135be565b60e0810161085d82846001600160a01b0381511682526020810151602083015260408101516040830152606081015160608301526080810151608083015260a081015160a083015260c0810151151560c08301525050565b60006020828403121561365f57600080fd5b5035919050565b60008060006060848603121561367b57600080fd5b613684846135be565b95602085013595506040909401359392505050565b6020808252825182820181905260009190848201906040850190845b81811015613725576137128385516001600160a01b0381511682526020810151602083015260408101516040830152606081015160608301526080810151608083015260a081015160a083015260c0810151151560c08301525050565b9284019260e092909201916001016136b5565b50909695505050505050565b6000806040838503121561374457600080fd5b61374d836135be565b946020939093013593505050565b803561ffff811681146135d557600080fd5b60ff811681146123cd57600080fd5b600080600080600060a0868803121561379457600080fd5b61379d8661375b565b945060208601356137ad8161376d565b935060408601356137bd8161376d565b925060608601356137cd8161376d565b915060808601356137dd8161376d565b809150509295509295909350565b6000608082840312156109b957600080fd5b803563ffffffff811681146135d557600080fd5b600080600060c0848603121561382657600080fd5b61382f846135be565b925061383e85602086016137eb565b915061384c60a085016137fd565b90509250925092565b6000806000806080858703121561386b57600080fd5b6138748561375b565b935060208501356138848161376d565b925060408501356138948161376d565b915060608501356138a48161376d565b939692955090935050565b6000806000606084860312156138c457600080fd5b6138cd8461375b565b925060208401356138dd8161376d565b915060408401356138ed8161376d565b809150509250925092565b6000806000806080858703121561390e57600080fd5b5050823594602084013594506040840135936060013592509050565b60008060008060008060c0878903121561394357600080fd5b61394c8761375b565b9550602087013561395c8161376d565b9450604087013561396c8161376d565b9350606087013561397c8161376d565b9250608087013561398c8161376d565b915060a087013561399c8161376d565b809150509295509295509295565b6000602082840312156139bc57600080fd5b61161c8261375b565b600080604083850312156139d857600080fd5b82356139e38161376d565b91506139f16020840161375b565b90509250929050565b600061010082840312156109b957600080fd5b600080600060c08486031215613a2257600080fd5b833567ffffffffffffffff80821115613a3a57600080fd5b613a46878388016139fa565b9450613a5587602088016137eb565b935060a0860135915080821115613a6b57600080fd5b508401604081870312156138ed57600080fd5b60008060408385031215613a9157600080fd5b823567ffffffffffffffff811115613aa857600080fd5b613ab4858286016139fa565b95602094909401359450505050565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052603260045260246000fd5b7f4e487b7100000000000000000000000000000000000000000000000000000000600052601260045260246000fd5b7f4e487b7100000000000000000000000000000000000000000000000000000000600052601160045260246000fd5b600082613b5f57613b5f613af2565b500490565b600082613b7357613b73613af2565b500690565b8181038181111561085d5761085d613b21565b8082018082111561085d5761085d613b21565b600060208284031215613bb057600080fd5b5051919050565b805180151581146135d557600080fd5b600060208284031215613bd957600080fd5b61161c82613bb7565b600080600080600060a08688031215613bfa57600080fd5b8551613c058161376d565b6020870151909550613c168161376d565b6040870151606088015191955093509150613c3360808701613bb7565b90509295509295909350565b808202811582820484141761085d5761085d613b21565b6001600160a01b038316815260a0810161161c6020830184803582526020810135602083015260408101356040830152606081013560608301525050565b600061ffff808316818103613cab57613cab613b21565b6001019392505050565b61ffff828116828216039080821115613cd057613cd0613b21565b5092915050565b60ff828116828216039081111561085d5761085d613b21565b600061ffff80841680613d0557613d05613af2565b92169190910692915050565b600060208284031215613d2357600080fd5b61161c826137fd565b60008083357fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe1843603018112613d6157600080fd5b83018035915067ffffffffffffffff821115613d7c57600080fd5b6020019150600581901b3603821315613d9457600080fd5b9250929050565b60008083357fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe1843603018112613dd057600080fd5b830160208101925035905067ffffffffffffffff811115613df057600080fd5b8060051b3603821315613d9457600080fd5b81835260007f07ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff831115613e3457600080fd5b8260051b80836020870137939093016020019392505050565b60006101006001600160a01b0380613e64856135be565b16855280613e74602086016135be565b16602086015280613e87604086016135be565b16604086015280613e9a606086016135be565b1660608601525063ffffffff613eb2608085016137fd565b16608085015260a083013560a085015260c083013560c0850152613ed960e0840184613d9b565b8260e08701526112db8387018284613e02565b60c081526000613eff60c0830186613e4d565b6020613f2e81850187803582526020810135602083015260408101356040830152606081013560608301525050565b83820360a0850152604082016001600160a01b0380613f4c886135be565b168452613f5b83880188613d9b565b604086860152928390529360009285916060015b81851015613f965783613f81886135be565b16815295850195600194909401938501613f6f565b9b9a5050505050505050505050565b60208152600061161c6020830184613e4d565b6000815480845260208085019450836000528060002060005b83811015613ff65781546001600160a01b031687529582019560019182019101613fd1565b509495945050505050565b8281526040602082015260006116196040830184613fb8565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052604160045260246000fd5b6000602080838503121561405c57600080fd5b825167ffffffffffffffff8082111561407457600080fd5b818501915085601f83011261408857600080fd5b81518181111561409a5761409a61401a565b8060051b6040517fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe0603f830116810181811085821117156140dd576140dd61401a565b6040529182528482019250838101850191888311156140fb57600080fd5b938501935b8285101561411957845184529385019392850192614100565b98975050505050505050565b85815284602082015260a06040820152600061414460a0830186613fb8565b6001600160a01b0394909416606083015250608001529392505050565b600060ff821660ff810361417757614177613b21565b60010192915050565b60005b8381101561419b578181015183820152602001614183565b50506000910152565b600082516141b6818460208701614180565b9190910192915050565b60208152600082518060208401526141df816040850160208701614180565b601f017fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe016919091016040019291505056fea164736f6c6343000813000a
Deployed Bytecode
0x608060405234801561001057600080fd5b50600436106103575760003560e01c80638da5cb5b116101c8578063c140240811610104578063e63318d5116100a2578063f887ea401161007c578063f887ea4014610770578063f95c81ff14610783578063fa93f88314610796578063ffa1ad74146107a957600080fd5b8063e63318d51461073d578063e834a83414610750578063f2fde38b1461075d57600080fd5b8063cb37f3b2116100de578063cb37f3b2146106d3578063d9082962146106e6578063d96eb767146106ef578063dc25a3001461070257600080fd5b8063c1402408146106a4578063c45a0155146106ad578063cb18583f146106c057600080fd5b8063a19d21b011610171578063af6d1fe41161014b578063af6d1fe414610663578063b199993714610676578063b238ad0e14610689578063b392972d1461069c57600080fd5b8063a19d21b01461062a578063a324ad241461063d578063a6f0e5771461065057600080fd5b80639054bdec116101a25780639054bdec146105e857806392d66313146105fb578063a06db7dc1461062157600080fd5b80638da5cb5b146105bb5780638dfd6a89146105cc5780638e25805b146105df57600080fd5b8063583c321d11610297578063669b0529116102405780637f7918331161021a5780637f791833146105535780638aa001fc146105665780638c8d98a0146105795780638cdb7e8b1461058c57600080fd5b8063669b05291461052f5780636c4a844314610538578063715018a61461054b57600080fd5b806362015bdc1161027157806362015bdc146104e657806362ba96871461050957806365c728401461051c57600080fd5b8063583c321d146104b0578063590e1ae3146104c35780635de29741146104cb57600080fd5b80633e239e1a116103045780634ac1ad78116102de5780634ac1ad7814610462578063522ce386146104755780635357a94a1461048a578063573761981461049d57600080fd5b80633e239e1a146104215780633e5ce385146104465780634344f6c51461045957600080fd5b80633013ce29116103355780633013ce29146103a057806330edc0f5146103cb5780633154b10e146103eb57600080fd5b8063144fa6d71461035c57806326232a2e146103715780632afd1a7d1461038d575b600080fd5b61036f61036a3660046135da565b6107b1565b005b61037a600c5481565b6040519081526020015b60405180910390f35b61037a61039b3660046135da565b610852565b6015546103b3906001600160a01b031681565b6040516001600160a01b039091168152602001610384565b6103de6103d93660046135da565b610863565b60405161038491906135f5565b6001546002546003546004546104019392919084565b604080519485526020850193909352918301526060820152608001610384565b61043461042f36600461364d565b6109bf565b60405160ff9091168152602001610384565b61036f610454366004613666565b6109e3565b61037a600b5481565b61043461047036600461364d565b610a8d565b61047d610aa9565b6040516103849190613699565b600e546103b3906001600160a01b031681565b61036f6104ab366004613731565b610b59565b61037a6104be36600461364d565b610d46565b61036f610d67565b6014546103b39064010000000090046001600160a01b031681565b6104f96104f43660046135da565b611153565b6040519015158152602001610384565b61037a61051736600461377c565b6112ca565b61043461052a36600461364d565b6112e5565b61037a60075481565b61036f610546366004613811565b6112fa565b61036f6115ca565b61037a610561366004613855565b6115de565b61043461057436600461364d565b6115fa565b61037a6105873660046138af565b611607565b6104f961059a3660046135da565b6001600160a01b031660009081526006602052604090206001015460ff1690565b6000546001600160a01b03166103b3565b61036f6105da3660046138f8565b611623565b61037a600a5481565b61037a6105f636600461392a565b611720565b61060e61060936600461364d565b6118d1565b60405161ffff9091168152602001610384565b61037a600d5481565b61037a6106383660046135da565b6119c1565b61043461064b36600461364d565b611a47565b6104f961065e3660046139aa565b611a5c565b6103b361067136600461364d565b611abb565b61037a61068436600461364d565b611ae5565b6104346106973660046139c5565b611b2a565b61036f611bf1565b61037a60095481565b6016546103b3906001600160a01b031681565b61036f6106ce366004613731565b611e36565b600f546103b3906001600160a01b031681565b61037a60085481565b61036f6106fd366004613a0d565b611eba565b61070a612359565b60405161038491908151815260208083015190820152604080830151908201526060918201519181019190915260800190565b61037a61074b3660046135da565b6123b1565b6012546104f99060ff1681565b61036f61076b3660046135da565b6123bc565b6010546103b3906001600160a01b031681565b61036f610791366004613a7e565b6123d0565b6104346107a436600461364d565b6126bd565b610434600181565b6016546001600160a01b031633146108105760405162461bcd60e51b815260206004820152600c60248201527f4f6e6c7920666163746f7279000000000000000000000000000000000000000060448201526064015b60405180910390fd5b601480546001600160a01b03909216640100000000027fffffffffffffffff0000000000000000000000000000000000000000ffffffff909216919091179055565b600061085d826126cb565b92915050565b6108ae6040518060e0016040528060006001600160a01b0316815260200160008152602001600081526020016000815260200160008152602001600081526020016000151581525090565b6001600160a01b038216600090815260066020526040902060010154829060ff1661091b5760405162461bcd60e51b815260206004820152601060248201527f4e6f7420696e2077686974656c697374000000000000000000000000000000006044820152606401610807565b6001600160a01b038316600090815260066020526040902054600580548290811061094857610948613ac3565b60009182526020918290206040805160e081018252600790930290910180546001600160a01b031683526001810154938301939093526002830154908201526003820154606082015260048201546080820152600582015460a082015260069091015460ff16151560c08201529250505b50919050565b60006018603c6109cf8185613b50565b6109d99190613b50565b61085d9190613b64565b6109eb61278a565b6001600160a01b03831660009081526006602052604081206001015460ff1615610a5f576001600160a01b038416600090815260066020526040902054600580549091908110610a3d57610a3d613ac3565b90600052602060002090600702016003015482610a5a9190613b78565b610a61565b815b601554909150610a7c906001600160a01b03168530846127e4565b610a878484846128b3565b50505050565b60006007610a9e6201518084613b50565b6109d9906004613b8b565b60606001600401805480602002602001604051908101604052809291908181526020016000905b82821015610b505760008481526020908190206040805160e0810182526007860290920180546001600160a01b03168352600180820154848601526002820154928401929092526003810154606084015260048101546080840152600581015460a08401526006015460ff16151560c08301529083529092019101610ad0565b50505050905090565b6016546001600160a01b03163314610bb35760405162461bcd60e51b815260206004820152600c60248201527f4f6e6c7920666163746f727900000000000000000000000000000000000000006044820152606401610807565b600254421080610c5c57506009546014546040517f70a0823100000000000000000000000000000000000000000000000000000000815230600482015283929164010000000090046001600160a01b0316906370a0823190602401602060405180830381865afa158015610c2b573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610c4f9190613b9e565b610c599190613b78565b10155b610ca85760405162461bcd60e51b815260206004820152600e60248201527f496e76616c696420616d6f756e740000000000000000000000000000000000006044820152606401610807565b6014546040517fa9059cbb0000000000000000000000000000000000000000000000000000000081526001600160a01b038481166004830152602482018490526401000000009092049091169063a9059cbb906044016020604051808303816000875af1158015610d1d573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610d419190613bc7565b505050565b60138181548110610d5657600080fd5b600091825260209091200154905081565b3360008181526006602052604090206001015460ff16610dc95760405162461bcd60e51b815260206004820152601060248201527f4e6f7420696e2077686974656c697374000000000000000000000000000000006044820152606401610807565b336000908152600660205260408120546005805491929183908110610df057610df0613ac3565b90600052602060002090600702019050600d546001800154610e129190613b8b565b42108015610e21575060025442115b610e6d5760405162461bcd60e51b815260206004820152601360248201527f4e6f7420696e20677261636520706572696f64000000000000000000000000006044820152606401610807565b600681015460ff1615610ec25760405162461bcd60e51b815260206004820152601560248201527f7573657220616c726561647920726566756e64656400000000000000000000006044820152606401610807565b600281015415610f145760405162461bcd60e51b815260206004820152601460248201527f7573657220616c726561647920636c61696d65640000000000000000000000006044820152606401610807565b600080610f296000546001600160a01b031690565b6040517f1a0cb3a70000000000000000000000000000000000000000000000000000000081523360048201526001600160a01b039190911690631a0cb3a79060240160a060405180830381865afa158015610f88573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610fac9190613be2565b50505060ff16915060ff169150600060138381548110610fce57610fce613ac3565b90600052602060002001549050600182111561101a576000610ff1600184613b78565b610ffc906032613c3f565b905080821161100c576000611016565b6110168183613b78565b9150505b600061271082866003015461102f9190613c3f565b6110399190613b50565b9050600081866003015461104d9190613b78565b6006870180547fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff00166001179055426005880155600387015460088054929350909160009061109c908490613b8b565b90915550506001860154600a80546000906110b8908490613b8b565b90915550506015546110d4906001600160a01b03163383612ba3565b81156111145760145463ffffffff1646036110f7576110f282612bec565b611114565b600f54601554611114916001600160a01b03918216911684612ba3565b60405181815233907fbb28353e4598c3b9199101a66e0989549b659a59a54d2c27fbb183f1932c8e6d9060200160405180910390a25050505050505050565b6001600160a01b038116600090815260066020526040812054600580548391908390811061118357611183613ac3565b60009182526020909120600790910201600681015490915060ff16156111eb5760405162461bcd60e51b815260206004820152601560248201527f7573657220616c726561647920726566756e64656400000000000000000000006044820152606401610807565b60006111f6856126cb565b9050600081116112485760405162461bcd60e51b815260206004820152600b60248201527f5a65726f20616d6f756e740000000000000000000000000000000000000000006044820152606401610807565b8082600201546112589190613b8b565b600283015560145461127c9064010000000090046001600160a01b03168683612ba3565b604080518281524260208201526001600160a01b038716917f34fcbac0073d7c3d388e51312faf357774904998eeb8fca628b9e6f65ee1cbf7910160405180910390a2506001949350505050565b60006112db86868686866000611720565b9695505050505050565b60006112f082612dcc565b6040015192915050565b6000547501000000000000000000000000000000000000000000900460ff1615808015611345575060005460017401000000000000000000000000000000000000000090910460ff16105b806113775750303b158015611377575060005474010000000000000000000000000000000000000000900460ff166001145b6113e95760405162461bcd60e51b815260206004820152602e60248201527f496e697469616c697a61626c653a20636f6e747261637420697320616c72656160448201527f647920696e697469616c697a65640000000000000000000000000000000000006064820152608401610807565b600080547fffffffffffffffffffffff00ffffffffffffffffffffffffffffffffffffffff1674010000000000000000000000000000000000000000179055801561146f57600080547fffffffffffffffffffff00ffffffffffffffffffffffffffffffffffffffffff1675010000000000000000000000000000000000000000001790555b601480547fffffffffffffffff0000000000000000000000000000000000000000ffffffff166401000000006001600160a01b038716021790556114b233612fb4565b601680547fffffffffffffffffffffffff000000000000000000000000000000000000000016331790556014805463ffffffff84167fffffffffffffffffffffffffffffffffffffffffffffffffffffffff00000000909116179055611527602084013584356040860135606087013561301c565b507f702283685135875ce459087727893f6d49d6fe02203812686d668fcb5812e9918484604051611559929190613c56565b60405180910390a18015610a8757600080547fffffffffffffffffffff00ffffffffffffffffffffffffffffffffffffffffff169055604051600181527f7f26b83ff96e1f2b6a682f133852f6798a09c465da95921460cefb3847402498906020015b60405180910390a150505050565b6115d261278a565b6115dc6000612fb4565b565b60006115ef85858585600080611720565b90505b949350505050565b600061085d603c83613b64565b60006116198484846000806000611720565b90505b9392505050565b6016546001600160a01b0316331461167d5760405162461bcd60e51b815260206004820152600c60248201527f4f6e6c7920666163746f727900000000000000000000000000000000000000006044820152606401610807565b60025442106116ce5760405162461bcd60e51b815260206004820152601760248201527f56657374696e6720616c726561647920737461727465640000000000000000006044820152606401610807565b6116da8484848461301c565b506040805185815260208101859052908101839052606081018290527feb4c1ae991ee614a820b5fdac6055e11ea335582cbf83319f7fe59dee3178a04906080016115bc565b60006107b25b8761ffff168161ffff16101561177c5761173f81611a5c565b15611759576117526301e2850083613b8b565b915061176a565b6117676301e1338083613b8b565b91505b8061177481613c94565b915050611726565b6117846134d4565b601f815261179189611a5c565b156117a257601d60208201526117aa565b601c60208201525b601f60408201819052601e606083018190526080830182905260a0830181905260c0830182905260e0830182905261010083018190526101208301829052610140830152610160820152600191505b8760ff168261ffff16101561185b5780611814600184613cb5565b61ffff16600c811061182857611828613ac3565b602002015161183d9060ff1662015180613c3f565b6118479084613b8b565b92508161185381613c94565b9250506117f9565b611866600188613cd7565b6118769060ff1662015180613c3f565b6118809084613b8b565b925061189160ff8716610e10613c3f565b61189b9084613b8b565b92506118ab60ff8616603c613c3f565b6118b59084613b8b565b92506118c460ff851684613b8b565b9998505050505050505050565b60008080806118e46301e1338086613b50565b6118f0906107b2613b8b565b91506118fd6107b2611ae5565b61190a8361ffff16611ae5565b6119149190613b78565b9050611924816301e28500613c3f565b61192e9084613b8b565b92508061193d6107b284613cb5565b61ffff1661194b9190613b78565b611959906301e13380613c3f565b6119639084613b8b565b92505b848311156119b95761197c61065e600184613cb5565b156119965761198f6301e2850084613b78565b92506119a7565b6119a46301e1338084613b78565b92505b6119b2600183613cb5565b9150611966565b509392505050565b6040517f70a0823100000000000000000000000000000000000000000000000000000000815230600482015260009082906001600160a01b038216906370a0823190602401602060405180830381865afa158015611a23573d6000803e3d6000fd5b505050506040513d601f19601f8201168201806040525081019061161c9190613b9e565b6000611a5282612dcc565b6020015192915050565b6000611a69600483613cf0565b61ffff1615611a7a57506000919050565b611a85606483613cf0565b61ffff1615611a9657506001919050565b611aa261019083613cf0565b61ffff1615611ab357506000919050565b506001919050565b60118181548110611acb57600080fd5b6000918252602090912001546001600160a01b0316905081565b6000611af2600183613b78565b9150611b0061019083613b50565b611b0b606484613b50565b611b16600485613b50565b611b209190613b78565b61085d9190613b8b565b60008260ff1660011480611b4157508260ff166003145b80611b4f57508260ff166005145b80611b5d57508260ff166007145b80611b6b57508260ff166008145b80611b7957508260ff16600a145b80611b8757508260ff16600c145b15611b945750601f61085d565b8260ff1660041480611ba957508260ff166006145b80611bb757508260ff166009145b80611bc557508260ff16600b145b15611bd25750601e61085d565b611bdb82611a5c565b15611be85750601d61085d565b50601c92915050565b600e546001600160a01b03163314611c4b5760405162461bcd60e51b815260206004820152600e60248201527f496e76616c6964206163636573730000000000000000000000000000000000006044820152606401610807565b600254600d54611c5b9190613b8b565b4211611ca95760405162461bcd60e51b815260206004820152601860248201527f677261636520706572696f6420696e2070726f677265737300000000000000006044820152606401610807565b60125460ff1615611cfc5760405162461bcd60e51b815260206004820152600f60248201527f616c726561647920636c61696d656400000000000000000000000000000000006044820152606401610807565b6000600854600754611d0e9190613b78565b90506000612710600c5483611d239190613c3f565b611d2d9190613b50565b9050611d398183613b78565b91506000600a54600954600b54611d509190613b78565b611d5a9190613b8b565b601280547fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff0016600117905590508215611daa57600e54601554611daa916001600160a01b03918216911685612ba3565b8015611dd557600e54601454611dd5916001600160a01b036401000000009092048216911683612ba3565b8115611df857600f54601554611df8916001600160a01b03918216911684612ba3565b60408051848152602081018390527f6414532af447bc814fac232567913fadd11eb1714ee73e08ab1f005a893a9a00910160405180910390a1505050565b611e3e61278a565b6001600160a01b03821660009081526006602052604090206001015460ff1615611eaa5760405162461bcd60e51b815260206004820152601260248201527f416c7265616479207265676973746572656400000000000000000000000000006044820152606401610807565b611eb6828260006128b3565b5050565b6000547501000000000000000000000000000000000000000000900460ff1615808015611f05575060005460017401000000000000000000000000000000000000000090910460ff16105b80611f375750303b158015611f37575060005474010000000000000000000000000000000000000000900460ff166001145b611fa95760405162461bcd60e51b815260206004820152602e60248201527f496e697469616c697a61626c653a20636f6e747261637420697320616c72656160448201527f647920696e697469616c697a65640000000000000000000000000000000000006064820152608401610807565b600080547fffffffffffffffffffffff00ffffffffffffffffffffffffffffffffffffffff1674010000000000000000000000000000000000000000179055801561202f57600080547fffffffffffffffffffff00ffffffffffffffffffffffffffffffffffffffffff1675010000000000000000000000000000000000000000001790555b61203c60208501856135da565b600e80547fffffffffffffffffffffffff0000000000000000000000000000000000000000166001600160a01b039290921691909117905561208460408501602086016135da565b600f80547fffffffffffffffffffffffff0000000000000000000000000000000000000000166001600160a01b03929092169190911790556120cc60608501604086016135da565b601480546001600160a01b0392909216640100000000027fffffffffffffffff0000000000000000000000000000000000000000ffffffff90921691909117905561211d60808501606086016135da565b601580547fffffffffffffffffffffffff0000000000000000000000000000000000000000166001600160a01b039290921691909117905560c0840135600d5560a084018035600b556121739060808601613d11565b601480547fffffffffffffffffffffffffffffffffffffffffffffffffffffffff000000001663ffffffff929092169190911790556121b560e0850185613d2c565b6121c1916013916134f3565b506121cf60208301836135da565b601080547fffffffffffffffffffffffff0000000000000000000000000000000000000000166001600160a01b03929092169190911790556122146020830183613d2c565b6122209160119161353e565b506015546010546040517f095ea7b30000000000000000000000000000000000000000000000000000000081526001600160a01b0391821660048201527fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff602482015291169063095ea7b3906044016020604051808303816000875af11580156122ae573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906122d29190613bc7565b506122dc33612fb4565b601680547fffffffffffffffffffffffff00000000000000000000000000000000000000001633179055610320600c55612325602084013584356040860135606087013561301c565b507f60539c54c534813a30c613b1b79c550f4b2e23341cea11cea6d0732e7038e15c84848460405161155993929190613eec565b6123846040518060800160405280600081526020016000815260200160008152602001600081525090565b50604080516080810182526001548152600254602082015260035491810191909152600454606082015290565b600061085d8261308d565b6123c461278a565b6123cd816131b3565b50565b6016546001600160a01b0316331461242a5760405162461bcd60e51b815260206004820152600c60248201527f4f6e6c7920666163746f727900000000000000000000000000000000000000006044820152606401610807565b600254421061247b5760405162461bcd60e51b815260206004820152601760248201527f56657374696e6720616c726561647920737461727465640000000000000000006044820152606401610807565b61248860208301836135da565b600e80547fffffffffffffffffffffffff0000000000000000000000000000000000000000166001600160a01b03929092169190911790556124d060408301602084016135da565b600f80547fffffffffffffffffffffffff0000000000000000000000000000000000000000166001600160a01b039290921691909117905561251860608301604084016135da565b601480546001600160a01b0392909216640100000000027fffffffffffffffff0000000000000000000000000000000000000000ffffffff90921691909117905561256960808301606084016135da565b601580547fffffffffffffffffffffffff0000000000000000000000000000000000000000166001600160a01b039290921691909117905560c0820135600d5560a0820135600b556125be60e0830183613d2c565b6125ca916013916134f3565b50600c8190556015546010546040517f095ea7b30000000000000000000000000000000000000000000000000000000081526001600160a01b0391821660048201527fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff602482015291169063095ea7b3906044016020604051808303816000875af115801561265d573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906126819190613bc7565b507f69a08d1ff0e6d976f9c703dccabe21e76e5484a8b7652f0190bc587d37550e0e826040516126b19190613fa5565b60405180910390a15050565b6000603c6109d98184613b50565b6001600160a01b038116600090815260066020526040812060010154829060ff166127385760405162461bcd60e51b815260206004820152601060248201527f4e6f7420696e2077686974656c697374000000000000000000000000000000006044820152606401610807565b6001600160a01b038316600090815260066020526040902054600580548290811061276557612765613ac3565b9060005260206000209060070201600201546127808561308d565b6115f29190613b78565b6000546001600160a01b031633146115dc5760405162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e65726044820152606401610807565b6040516001600160a01b0380851660248301528316604482015260648101829052610a879085907f23b872dd00000000000000000000000000000000000000000000000000000000906084015b604080517fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe08184030181529190526020810180517bffffffffffffffffffffffffffffffffffffffffffffffffffffffff167fffffffff0000000000000000000000000000000000000000000000000000000090931692909217909152613240565b6001600160a01b0383166000908152600660205260409020600181015460ff16612ad857600181810180547fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff009081168317909155600580548085556040805160e0810182526001600160a01b038a81168252602082018a81526000938301848152606084018b8152426080860190815260a0860187815260c087018881529b89018a5598875294517f036b6384b5eca791c62761152d0c79bb0604c104a5fb6f4eb0703f3154bb3db0600798890290810180547fffffffffffffffffffffffff000000000000000000000000000000000000000016929096169190911790945591517f036b6384b5eca791c62761152d0c79bb0604c104a5fb6f4eb0703f3154bb3db1840155517f036b6384b5eca791c62761152d0c79bb0604c104a5fb6f4eb0703f3154bb3db2830155517f036b6384b5eca791c62761152d0c79bb0604c104a5fb6f4eb0703f3154bb3db382015590517f036b6384b5eca791c62761152d0c79bb0604c104a5fb6f4eb0703f3154bb3db482015592517f036b6384b5eca791c62761152d0c79bb0604c104a5fb6f4eb0703f3154bb3db584015593517f036b6384b5eca791c62761152d0c79bb0604c104a5fb6f4eb0703f3154bb3db69092018054909316911515919091179091558054849290612ab4908490613b8b565b925050819055508260096000828254612acd9190613b8b565b90915550612b599050565b805460058054600092908110612af057612af0613ac3565b90600052602060002090600702019050806003015483612b109190613b78565b60076000828254612b219190613b8b565b90915550506001810154612b359085613b78565b60096000828254612b469190613b8b565b9091555050600181018490556003018290555b60408051848152602081018490526001600160a01b038616917fdb76cc8d16d8e51a4a8b73f914539ac9f9e4e42ff5f0140ddd580c48e8947798910160405180910390a250505050565b6040516001600160a01b038316602482015260448101829052610d419084907fa9059cbb0000000000000000000000000000000000000000000000000000000090606401612831565b6010546040517fd06ca61f0000000000000000000000000000000000000000000000000000000081526001600160a01b0390911690600090829063d06ca61f90612c3d908690601190600401614001565b600060405180830381865afa158015612c5a573d6000803e3d6000fd5b505050506040513d6000823e601f3d9081017fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe0168201604052612ca09190810190614049565b9050600060648260018451612cb59190613b78565b81518110612cc557612cc5613ac3565b60200260200101516063612cd99190613c3f565b612ce39190613b50565b6040517f38ed17390000000000000000000000000000000000000000000000000000000081529091506001600160a01b038416906338ed173990612d36908790859060119061dead904290600401614125565b6000604051808303816000875af1158015612d55573d6000803e3d6000fd5b505050506040513d6000823e601f3d9081017fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe0168201604052612d9b9190810190614049565b506040518481527ffcfa54313b92fceed71362d6bfec038726e592d2fa73d69c461986eeedf74400906020016115bc565b6040805160e081018252600080825260208201819052918101829052606081018290526080810182905260a0810182905260c08101829052908080612e10856118d1565b61ffff168452612e216107b2611ae5565b8451612e309061ffff16611ae5565b612e3a9190613b78565b9150612e4a826301e28500613c3f565b612e549084613b8b565b9250816107b28560000151612e699190613cb5565b61ffff16612e779190613b78565b612e85906301e13380613c3f565b612e8f9084613b8b565b92506000600191505b600c8260ff1611612f0057612eb1828660000151611b2a565b612ec19060ff1662015180613c3f565b905085612ece8583613b8b565b1115612ee25760ff82166020860152612f00565b612eec8185613b8b565b935081612ef881614161565b925050612e98565b600191505b612f1785602001518660000151611b2a565b60ff168260ff1611612f665785612f318562015180613b8b565b1115612f455760ff82166040860152612f66565b612f526201518085613b8b565b935081612f5e81614161565b925050612f05565b612f6f866109bf565b60ff166060860152612f80866126bd565b60ff166080860152612f91866115fa565b60ff1660a0860152612fa286610a8d565b60ff1660c08601525092949350505050565b600080546001600160a01b038381167fffffffffffffffffffffffff0000000000000000000000000000000000000000831681178455604051919092169283917f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e09190a35050565b60006130288585613b8b565b6001556002849055600383905560048290556040805186815260208101869052908101849052606081018390527f8b2bfe599bf9e0b744a947eec87d2b85b086f600a3c5f8901ca919b9c30f034b9060800160405180910390a1506001949350505050565b6001600160a01b038116600090815260066020526040812060010154829060ff166130fa5760405162461bcd60e51b815260206004820152601060248201527f4e6f7420696e2077686974656c697374000000000000000000000000000000006044820152606401610807565b6001600160a01b038316600090815260066020526040812054600580549192918390811061312a5761312a613ac3565b60009182526020909120600160079092020181015460025490925042101561315857600094505050506109b9565b8060010154421015801561316c5750805442105b15613198576103e88160030154836131849190613c3f565b61318e9190613b50565b94505050506109b9565b805442106131aa5761318e8282613328565b50505050919050565b6131bb61278a565b6001600160a01b0381166132375760405162461bcd60e51b815260206004820152602660248201527f4f776e61626c653a206e6577206f776e657220697320746865207a65726f206160448201527f64647265737300000000000000000000000000000000000000000000000000006064820152608401610807565b6123cd81612fb4565b6000613295826040518060400160405280602081526020017f5361666545524332303a206c6f772d6c6576656c2063616c6c206661696c6564815250856001600160a01b03166133b19092919063ffffffff16565b90508051600014806132b65750808060200190518101906132b69190613bc7565b610d415760405162461bcd60e51b815260206004820152602a60248201527f5361666545524332303a204552433230206f7065726174696f6e20646964206e60448201527f6f742073756363656564000000000000000000000000000000000000000000006064820152608401610807565b6000806103e883600301548561333e9190613c3f565b6133489190613b50565b905060006133568286613b78565b6002850154855491925061336991613b8b565b421061337957849250505061085d565b6002840154845461338a9042613b78565b6133949083613c3f565b61339e9190613b50565b6133a89083613b8b565b9250505061085d565b6060611619848460008585600080866001600160a01b031685876040516133d891906141a4565b60006040518083038185875af1925050503d8060008114613415576040519150601f19603f3d011682016040523d82523d6000602084013e61341a565b606091505b509150915061342b87838387613436565b979650505050505050565b606083156134a557825160000361349e576001600160a01b0385163b61349e5760405162461bcd60e51b815260206004820152601d60248201527f416464726573733a2063616c6c20746f206e6f6e2d636f6e74726163740000006044820152606401610807565b50816115f2565b6115f283838151156134ba5781518083602001fd5b8060405162461bcd60e51b815260040161080791906141c0565b604051806101800160405280600c906020820280368337509192915050565b82805482825590600052602060002090810192821561352e579160200282015b8281111561352e578235825591602001919060010190613513565b5061353a9291506135a9565b5090565b82805482825590600052602060002090810192821561352e579160200282015b8281111561352e5781547fffffffffffffffffffffffff0000000000000000000000000000000000000000166001600160a01b0384351617825560209092019160019091019061355e565b5b8082111561353a57600081556001016135aa565b80356001600160a01b03811681146135d557600080fd5b919050565b6000602082840312156135ec57600080fd5b61161c826135be565b60e0810161085d82846001600160a01b0381511682526020810151602083015260408101516040830152606081015160608301526080810151608083015260a081015160a083015260c0810151151560c08301525050565b60006020828403121561365f57600080fd5b5035919050565b60008060006060848603121561367b57600080fd5b613684846135be565b95602085013595506040909401359392505050565b6020808252825182820181905260009190848201906040850190845b81811015613725576137128385516001600160a01b0381511682526020810151602083015260408101516040830152606081015160608301526080810151608083015260a081015160a083015260c0810151151560c08301525050565b9284019260e092909201916001016136b5565b50909695505050505050565b6000806040838503121561374457600080fd5b61374d836135be565b946020939093013593505050565b803561ffff811681146135d557600080fd5b60ff811681146123cd57600080fd5b600080600080600060a0868803121561379457600080fd5b61379d8661375b565b945060208601356137ad8161376d565b935060408601356137bd8161376d565b925060608601356137cd8161376d565b915060808601356137dd8161376d565b809150509295509295909350565b6000608082840312156109b957600080fd5b803563ffffffff811681146135d557600080fd5b600080600060c0848603121561382657600080fd5b61382f846135be565b925061383e85602086016137eb565b915061384c60a085016137fd565b90509250925092565b6000806000806080858703121561386b57600080fd5b6138748561375b565b935060208501356138848161376d565b925060408501356138948161376d565b915060608501356138a48161376d565b939692955090935050565b6000806000606084860312156138c457600080fd5b6138cd8461375b565b925060208401356138dd8161376d565b915060408401356138ed8161376d565b809150509250925092565b6000806000806080858703121561390e57600080fd5b5050823594602084013594506040840135936060013592509050565b60008060008060008060c0878903121561394357600080fd5b61394c8761375b565b9550602087013561395c8161376d565b9450604087013561396c8161376d565b9350606087013561397c8161376d565b9250608087013561398c8161376d565b915060a087013561399c8161376d565b809150509295509295509295565b6000602082840312156139bc57600080fd5b61161c8261375b565b600080604083850312156139d857600080fd5b82356139e38161376d565b91506139f16020840161375b565b90509250929050565b600061010082840312156109b957600080fd5b600080600060c08486031215613a2257600080fd5b833567ffffffffffffffff80821115613a3a57600080fd5b613a46878388016139fa565b9450613a5587602088016137eb565b935060a0860135915080821115613a6b57600080fd5b508401604081870312156138ed57600080fd5b60008060408385031215613a9157600080fd5b823567ffffffffffffffff811115613aa857600080fd5b613ab4858286016139fa565b95602094909401359450505050565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052603260045260246000fd5b7f4e487b7100000000000000000000000000000000000000000000000000000000600052601260045260246000fd5b7f4e487b7100000000000000000000000000000000000000000000000000000000600052601160045260246000fd5b600082613b5f57613b5f613af2565b500490565b600082613b7357613b73613af2565b500690565b8181038181111561085d5761085d613b21565b8082018082111561085d5761085d613b21565b600060208284031215613bb057600080fd5b5051919050565b805180151581146135d557600080fd5b600060208284031215613bd957600080fd5b61161c82613bb7565b600080600080600060a08688031215613bfa57600080fd5b8551613c058161376d565b6020870151909550613c168161376d565b6040870151606088015191955093509150613c3360808701613bb7565b90509295509295909350565b808202811582820484141761085d5761085d613b21565b6001600160a01b038316815260a0810161161c6020830184803582526020810135602083015260408101356040830152606081013560608301525050565b600061ffff808316818103613cab57613cab613b21565b6001019392505050565b61ffff828116828216039080821115613cd057613cd0613b21565b5092915050565b60ff828116828216039081111561085d5761085d613b21565b600061ffff80841680613d0557613d05613af2565b92169190910692915050565b600060208284031215613d2357600080fd5b61161c826137fd565b60008083357fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe1843603018112613d6157600080fd5b83018035915067ffffffffffffffff821115613d7c57600080fd5b6020019150600581901b3603821315613d9457600080fd5b9250929050565b60008083357fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe1843603018112613dd057600080fd5b830160208101925035905067ffffffffffffffff811115613df057600080fd5b8060051b3603821315613d9457600080fd5b81835260007f07ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff831115613e3457600080fd5b8260051b80836020870137939093016020019392505050565b60006101006001600160a01b0380613e64856135be565b16855280613e74602086016135be565b16602086015280613e87604086016135be565b16604086015280613e9a606086016135be565b1660608601525063ffffffff613eb2608085016137fd565b16608085015260a083013560a085015260c083013560c0850152613ed960e0840184613d9b565b8260e08701526112db8387018284613e02565b60c081526000613eff60c0830186613e4d565b6020613f2e81850187803582526020810135602083015260408101356040830152606081013560608301525050565b83820360a0850152604082016001600160a01b0380613f4c886135be565b168452613f5b83880188613d9b565b604086860152928390529360009285916060015b81851015613f965783613f81886135be565b16815295850195600194909401938501613f6f565b9b9a5050505050505050505050565b60208152600061161c6020830184613e4d565b6000815480845260208085019450836000528060002060005b83811015613ff65781546001600160a01b031687529582019560019182019101613fd1565b509495945050505050565b8281526040602082015260006116196040830184613fb8565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052604160045260246000fd5b6000602080838503121561405c57600080fd5b825167ffffffffffffffff8082111561407457600080fd5b818501915085601f83011261408857600080fd5b81518181111561409a5761409a61401a565b8060051b6040517fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe0603f830116810181811085821117156140dd576140dd61401a565b6040529182528482019250838101850191888311156140fb57600080fd5b938501935b8285101561411957845184529385019392850192614100565b98975050505050505050565b85815284602082015260a06040820152600061414460a0830186613fb8565b6001600160a01b0394909416606083015250608001529392505050565b600060ff821660ff810361417757614177613b21565b60010192915050565b60005b8381101561419b578181015183820152602001614183565b50506000910152565b600082516141b6818460208701614180565b9190910192915050565b60208152600082518060208401526141df816040850160208701614180565b601f017fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe016919091016040019291505056fea164736f6c6343000813000a
Loading...
Loading
Loading...
Loading
Loading...
Loading
Net Worth in USD
$0.00
Net Worth in ETH
Multichain Portfolio | 35 Chains
| Chain | Token | Portfolio % | Price | Amount | Value |
|---|
Loading...
Loading
Loading...
Loading
Loading...
Loading
[ Download: CSV Export ]
A contract address hosts a smart contract, which is a set of code stored on the blockchain that runs when predetermined conditions are met. Learn more about addresses in our Knowledge Base.