ETH Price: $2,408.11 (-0.27%)

Contract

0x87df4c7E6E8E76ba82C4C239261A8D070576E76F
 

Overview

ETH Balance

0 ETH

Eth Value

$0.00

Token Holdings

Multichain Info

No addresses found
Transaction Hash
Method
Block
From
To
Divest191202962024-01-30 15:43:35228 days ago1706629415IN
0x87df4c7E...70576E76F
0 ETH0.0135822866.91704876
Grant Role173862312023-06-01 13:23:23471 days ago1685625803IN
0x87df4c7E...70576E76F
0 ETH0.0020491240
0x61014060173862302023-06-01 13:23:11471 days ago1685625791IN
 Create: Strategy
0 ETH0.1507192440

View more zero value Internal Transactions in Advanced View mode

Advanced mode:
Loading...
Loading

Contract Source Code Verified (Exact Match)

Contract Name:
Strategy

Compiler Version
v0.8.15+commit.e14f2714

Optimization Enabled:
Yes with 100 runs

Other Settings:
default evmVersion
File 1 of 28 : Strategy.sol
// SPDX-License-Identifier: BUSL-1.1
// Audit of commit 9e6a33d at https://hackmd.io/7YB8QorOSs-nAAaz_f8EbQ

pragma solidity >=0.8.13;

import { IStrategy } from "./interfaces/IStrategy.sol";
import { StrategyMigrator } from "./StrategyMigrator.sol";
import { AccessControl } from "@yield-protocol/utils-v2/src/access/AccessControl.sol";
import { SafeERC20Namer } from "@yield-protocol/utils-v2/src/token/SafeERC20Namer.sol";
import { MinimalTransferHelper } from "@yield-protocol/utils-v2/src/token/MinimalTransferHelper.sol";
import { IERC20 } from "@yield-protocol/utils-v2/src/token/IERC20.sol";
import { ERC20Rewards } from "@yield-protocol/utils-v2/src/token/ERC20Rewards.sol";
import { IFYToken } from "@yield-protocol/vault-v2/src/interfaces/IFYToken.sol";
import { IPool } from "@yield-protocol/yieldspace-tv/src/interfaces/IPool.sol";

/// @dev The Strategy contract allows liquidity providers to provide liquidity in yieldspace
/// pool tokens and receive strategy tokens that represent a stake in a YieldSpace pool contract.
/// Upon maturity, the strategy can `divest` from the mature pool, becoming a proportional
/// ownership underlying vault. When not invested, the strategy can `invest` into a Pool using
/// all its underlying.
/// The strategy can also `eject` from a Pool before maturity. Any fyToken obtained will be available
/// to be bought by anyone at face value. If the pool tokens can't be burned, they will be ejected
/// and the strategy can be recapitalized.
contract Strategy is AccessControl, ERC20Rewards, StrategyMigrator { // TODO: I'd like to import IStrategy
    enum State {DEPLOYED, DIVESTED, INVESTED, EJECTED, DRAINED}
    using MinimalTransferHelper for IERC20;
    using MinimalTransferHelper for IFYToken;
    using MinimalTransferHelper for IPool;

    event Invested(address indexed pool, uint256 baseInvested, uint256 lpTokensObtained);
    event Divested(address indexed pool, uint256 lpTokenDivested, uint256 baseObtained);
    event Ejected(address indexed pool, uint256 lpTokenDivested, uint256 baseObtained, uint256 fyTokenObtained);
    event Drained(address indexed pool, uint256 lpTokenDivested);
    event SoldFYToken(uint256 soldFYToken, uint256 returnedBase);

    State public state;                          // The state determines which functions are available

    // IERC20 public immutable base;             // Base token for this strategy (inherited from StrategyMigrator)
    // IFYToken public override fyToken;         // Current fyToken for this strategy (inherited from StrategyMigrator)
    IPool public pool;                           // Current pool that this strategy invests in

    uint256 public baseCached;                   // Base tokens held by the strategy
    uint256 public poolCached;                   // Pool tokens held by the strategy
    uint256 public fyTokenCached;                // In emergencies, the strategy can keep fyToken

    constructor(string memory name_, string memory symbol_, IFYToken fyToken_)
        ERC20Rewards(name_, symbol_, SafeERC20Namer.tokenDecimals(address(fyToken_)))
        StrategyMigrator(
            IERC20(fyToken_.underlying()),
            fyToken_)
    {}

    modifier isState(State target) {
        require (
            target == state,
            "Not allowed in this state"
        );
        _;
    }

    /// @notice State and state variable management
    /// @param target State to transition to
    /// @param pool_ If transitioning to invested, update pool state variable with this parameter
    function _transition(State target, IPool pool_) internal {
        if (target == State.INVESTED) {
            pool = pool_;
            fyToken = IFYToken(address(pool_.fyToken()));
            maturity = pool_.maturity();
        } else if (target == State.DIVESTED) {
            delete fyToken;
            delete maturity;
            delete pool;
        } else if (target == State.EJECTED) {
            delete maturity;
            delete pool;
        } else if (target == State.DRAINED) {
            delete maturity;
            delete pool;
        }
        state = target;
    }

    /// @notice State and state variable management
    /// @param target State to transition to
    function _transition(State target) internal {
        require (target != State.INVESTED, "Must provide a pool");
        _transition(target, IPool(address(0)));
    }

    // ----------------------- INVEST & DIVEST --------------------------- //

    /// @notice Mint the first strategy tokens, without investing
    /// @dev Returns additional values to match the pool init function and allow for strategy migrations.
    /// It is expected that base has been transferred in, but no fyTokens
    /// @return baseIn Amount of base tokens found in contract
    /// @return fyTokenIn This is always returned as 0 since they aren't used
    /// @return minted Amount of strategy tokens minted from base tokens which is the same as baseIn
    function init(address to)
        external
        override
        auth
        returns (uint256 baseIn, uint256 fyTokenIn, uint256 minted)
    {
        fyTokenIn = 0; // Silence compiler warning
        baseIn = minted = _init(to);
    }

    /// @notice Mint the first strategy tokens, without investing
    /// @param to Recipient for the strategy tokens
    /// @return minted Amount of strategy tokens minted from base tokens
    function _init(address to)
        internal
        isState(State.DEPLOYED)
        returns (uint256 minted)
    {
        // Clear fyToken in case we initialized through `mint`
        delete fyToken;

        baseCached = minted = base.balanceOf(address(this));
        require (minted > 0, "Not enough base in");
        // Make sure that at the end of the transaction the strategy has enough tokens as to not expose itself to a rounding-down liquidity attack.
        _mint(to, minted);

        _transition(State.DIVESTED);
    }

    /// @notice Start the strategy investments in the next pool
    /// @param pool_ Pool to invest into
    /// @return poolTokensObtained Amount of pool tokens minted from base tokens
    /// @notice When calling this function for the first pool, some underlying needs to be transferred to the strategy first, using a batchable router.
    function invest(IPool pool_)
        external
        auth
        isState(State.DIVESTED)
        returns (uint256 poolTokensObtained)
    {
        // Caching
        IFYToken fyToken_ = IFYToken(address(pool_.fyToken()));
        uint256 baseCached_ = baseCached; // We could read the real balance, but this is a bit safer

        require(base == pool_.base(), "Mismatched base");

        // Mint LP tokens and initialize the pool
        delete baseCached;
        base.safeTransfer(address(pool_), baseCached_);
        (,, poolTokensObtained) = pool_.init(address(this));
        poolCached = poolTokensObtained;

        // Update state variables
        fyToken = fyToken_;
        maturity = pool_.maturity();
        pool = pool_;

        _transition(State.INVESTED, pool_);
        emit Invested(address(pool_), baseCached_, poolTokensObtained);
    }

    /// @notice Divest out of a pool once it has matured
    /// @return baseObtained Amount of base tokens obtained from burning pool tokens   
    function divest()
        external
        isState(State.INVESTED)
        returns (uint256 baseObtained)
    {
        // Caching
        IPool pool_ = pool;
        IFYToken fyToken_ = fyToken;
        require (uint32(block.timestamp) >= maturity, "Only after maturity");

        uint256 toDivest = pool_.balanceOf(address(this));

        // Burn lpTokens
        delete poolCached;
        pool_.safeTransfer(address(pool_), toDivest);
        (, uint256 baseFromBurn, uint256 fyTokenFromBurn) = pool_.burn(address(this), address(this), 0, type(uint256).max); // We don't care about slippage, because the strategy holds to maturity

        // Redeem any fyToken
        uint256 baseFromRedeem = fyToken_.redeem(address(this), fyTokenFromBurn);

        // Reset the base cache
        baseCached = base.balanceOf(address(this));

        // Transition to Divested
        _transition(State.DIVESTED, pool_);
        emit Divested(address(pool_), toDivest, baseObtained = baseFromBurn + baseFromRedeem);
    }

    // ----------------------- EMERGENCY --------------------------- //

    /// @notice Divest out of a pool at any time. If possible the pool tokens will be burnt for base and fyToken, the latter of which
    /// must be sold to return the strategy to a functional state. If the pool token burn reverts, the pool tokens will be transferred
    /// to the caller as a last resort.
    /// @return baseReceived Amount of base tokens received from pool tokens
    /// @return fyTokenReceived Amount of fyToken received from pool tokens
    /// @notice The caller must take care of slippage when selling fyToken, if relevant.
    function eject()
        external
        auth
        isState(State.INVESTED)
        returns (uint256 baseReceived, uint256 fyTokenReceived)
    {
        // Caching
        IPool pool_ = pool;
        uint256 toDivest = pool_.balanceOf(address(this));

        // Burn lpTokens, if not possible, eject the pool tokens out. Slippage should be managed by the caller.
        delete poolCached;
        try this.burnPoolTokens(pool_, toDivest) returns (uint256 baseReceived_, uint256 fyTokenReceived_) {
            baseCached = baseReceived = baseReceived_;
            fyTokenCached = fyTokenReceived = fyTokenReceived_;
            if (fyTokenReceived > 0) {
                _transition(State.EJECTED, pool_);
                emit Ejected(address(pool_), toDivest, baseReceived, fyTokenReceived);
            } else {
                _transition(State.DIVESTED, pool_);
                emit Divested(address(pool_), toDivest, baseReceived);
            }

        } catch {
            pool_.safeTransfer(msg.sender, toDivest);
            _transition(State.DRAINED, pool_);
            emit Drained(address(pool_), toDivest);
        }
    }

    /// @notice Burn an amount of pool tokens.
    /// @dev Only the Strategy itself can call this function. It is external and exists so that the transfer is reverted if the burn also reverts.
    /// @param pool_ Pool for the pool tokens.
    /// @param poolTokens Amount of tokens to burn.
    /// @return baseReceived Amount of base tokens received from pool tokens
    /// @return fyTokenReceived Amount of fyToken received from pool tokens
    function burnPoolTokens(IPool pool_, uint256 poolTokens)
        external
        returns (uint256 baseReceived, uint256 fyTokenReceived)
    {
        require (msg.sender ==  address(this), "Unauthorized");

        // Burn lpTokens
        pool_.safeTransfer(address(pool_), poolTokens);
        uint256 baseBalance = base.balanceOf(address(this));
        uint256 fyTokenBalance = fyToken.balanceOf(address(this));
        (, baseReceived, fyTokenReceived) = pool_.burn(address(this), address(this), 0, type(uint256).max);
        require(base.balanceOf(address(this)) - baseBalance == baseReceived, "Burn failed - base");
        require(fyToken.balanceOf(address(this)) - fyTokenBalance == fyTokenReceived, "Burn failed - fyToken");
    }

    /// @notice Buy ejected fyToken in the strategy at face value
    /// @param fyTokenTo Address to send the purchased fyToken to.
    /// @param baseTo Address to send any remaining base to.
    /// @return soldFYToken Amount of fyToken sold.
    /// @return returnedBase Amount of base unused and returned.
    function buyFYToken(address fyTokenTo, address baseTo)
        external
        isState(State.EJECTED)
        returns (uint256 soldFYToken, uint256 returnedBase)
    {
        // Caching
        IFYToken fyToken_ = fyToken;
        uint256 baseCached_ = baseCached;
        uint256 fyTokenCached_ = fyTokenCached;

        uint256 baseIn = base.balanceOf(address(this)) - baseCached_;
        (soldFYToken, returnedBase) = baseIn > fyTokenCached_ ? (fyTokenCached_, baseIn - fyTokenCached_) : (baseIn, 0);

        // Update base and fyToken cache
        baseCached = baseCached_ + soldFYToken; // soldFYToken is base not returned
        fyTokenCached = fyTokenCached_ -= soldFYToken;

        // Transition to divested if done
        if (fyTokenCached_ == 0) {
            // Transition to Divested
            _transition(State.DIVESTED);
            emit Divested(address(0), 0, 0);
        }

        // Transfer fyToken and base (if surplus)
        fyToken_.safeTransfer(fyTokenTo, soldFYToken);
        if (soldFYToken < baseIn) {
            base.safeTransfer(baseTo, baseIn - soldFYToken);
        }

        emit SoldFYToken(soldFYToken, returnedBase);
    }

    /// @notice If we drained the strategy, we can recapitalize it with base to avoid a forced migration
    /// @return baseIn Amount of base tokens used to restart
    function restart()
        external
        auth
        isState(State.DRAINED)
        returns (uint256 baseIn)
    {
        require((baseCached = baseIn = base.balanceOf(address(this))) > 0, "No base to restart");
        _transition(State.DIVESTED);
        emit Divested(address(0), 0, 0);
    }

    /// @notice If everything else goes wrong, use this to take corrective action
    function call(address target, bytes calldata data) external auth returns (bytes memory) {
        (bool success, bytes memory returndata) = target.call(data);
        require(success, "Call failed");
        return returndata;
    }

    // ----------------------- MINT & BURN --------------------------- //

    /// @notice Mint strategy tokens with pool tokens. It can be called only when invested.
    /// @param to Recipient for the strategy tokens
    /// @return minted Amount of strategy tokens minted
    /// @notice The pool tokens that the user contributes need to have been transferred previously, using a batchable router.
    function mint(address to)
        external
        isState(State.INVESTED)
        returns (uint256 minted)
    {
        // Caching
        IPool pool_ = pool;
        uint256 poolCached_ = poolCached;

        // minted = supply * value(deposit) / value(strategy)

        // Find how much was deposited
        uint256 deposit = pool_.balanceOf(address(this)) - poolCached_;

        // Update the pool cache
        poolCached = poolCached_ + deposit;

        // Mint strategy tokens
        minted = _totalSupply * deposit / poolCached_;
        _mint(to, minted);
    }

    /// @notice Burn strategy tokens to withdraw pool tokens. It can be called only when invested.
    /// @param to Recipient for the pool tokens
    /// @return poolTokensObtained Amount of pool tokens obtained
    /// @notice The strategy tokens that the user burns need to have been transferred previously, using a batchable router.
    function burn(address to)
        external
        isState(State.INVESTED)
        returns (uint256 poolTokensObtained)
    {
        // Caching
        IPool pool_ = pool;
        uint256 poolCached_ = poolCached;
        uint256 totalSupply_ = _totalSupply;

        // Burn strategy tokens
        uint256 burnt = _balanceOf[address(this)];
        _burn(address(this), burnt);

        poolTokensObtained = poolCached_ * burnt / totalSupply_;
        pool_.safeTransfer(address(to), poolTokensObtained);

        // Update pool cache
        poolCached = poolCached_ - poolTokensObtained;
    }

    /// @notice Mint strategy tokens with base tokens. It can be called only when not invested and not ejected.
    /// @param to Recipient for the strategy tokens
    /// @return minted Amount of strategy tokens minted
    /// @notice The base tokens that the user invests need to have been transferred previously, using a batchable router.
    function mintDivested(address to)
        external
        isState(State.DIVESTED)
        returns (uint256 minted)
    {
        // minted = supply * value(deposit) / value(strategy)
        uint256 baseCached_ = baseCached;
        uint256 deposit = base.balanceOf(address(this)) - baseCached_;
        baseCached = baseCached_ + deposit;

        minted = _totalSupply * deposit / baseCached_;

        _mint(to, minted);
    }

    /// @notice Burn strategy tokens to withdraw base tokens. It can be called when not invested and not ejected.
    /// @param to Recipient for the base tokens
    /// @return baseObtained Amount of base tokens obtained
    /// @notice The strategy tokens that the user burns need to have been transferred previously, using a batchable router.
    function burnDivested(address to)
        external
        isState(State.DIVESTED)
        returns (uint256 baseObtained)
    {
        // strategy * burnt/supply = withdrawal
        uint256 baseCached_ = baseCached;
        uint256 burnt = _balanceOf[address(this)];
        baseObtained = baseCached_ * burnt / _totalSupply;
        baseCached = baseCached_ - baseObtained;

        _burn(address(this), burnt);
        base.safeTransfer(to, baseObtained);
    }
}

File 2 of 28 : AccessControl.sol
// SPDX-License-Identifier: MIT

pragma solidity ^0.8.0;

/**
 * @dev Contract module that allows children to implement role-based access
 * control mechanisms.
 *
 * Roles are referred to by their `bytes4` identifier. These are expected to be the 
 * signatures for all the functions in the contract. Special roles should be exposed
 * in the external API and be unique:
 *
 * ```
 * bytes4 public constant ROOT = 0x00000000;
 * ```
 *
 * Roles represent restricted access to a function call. For that purpose, use {auth}:
 *
 * ```
 * function foo() public auth {
 *     ...
 * }
 * ```
 *
 * Roles can be granted and revoked dynamically via the {grantRole} and
 * {revokeRole} functions. Each role has an associated admin role, and only
 * accounts that have a role's admin role can call {grantRole} and {revokeRole}.
 *
 * By default, the admin role for all roles is `ROOT`, which means
 * that only accounts with this role will be able to grant or revoke other
 * roles. More complex role relationships can be created by using
 * {setRoleAdmin}.
 *
 * WARNING: The `ROOT` is also its own admin: it has permission to
 * grant and revoke this role. Extra precautions should be taken to secure
 * accounts that have been granted it.
 */
contract AccessControl {
    struct RoleData {
        mapping (address => bool) members;
        bytes4 adminRole;
    }

    mapping (bytes4 => RoleData) private _roles;

    bytes4 public constant ROOT = 0x00000000;
    bytes4 public constant ROOT4146650865 = 0x00000000; // Collision protection for ROOT, test with ROOT12007226833()
    bytes4 public constant LOCK = 0xFFFFFFFF;           // Used to disable further permissioning of a function
    bytes4 public constant LOCK8605463013 = 0xFFFFFFFF; // Collision protection for LOCK, test with LOCK10462387368()

    /**
     * @dev Emitted when `newAdminRole` is set as ``role``'s admin role
     *
     * `ROOT` is the starting admin for all roles, despite
     * {RoleAdminChanged} not being emitted signaling this.
     *
     * _Available since v3.1._
     */
    event RoleAdminChanged(bytes4 indexed role, bytes4 indexed newAdminRole);

    /**
     * @dev Emitted when `account` is granted `role`.
     *
     * `sender` is the account that originated the contract call.
     */
    event RoleGranted(bytes4 indexed role, address indexed account, address indexed sender);

    /**
     * @dev Emitted when `account` is revoked `role`.
     *
     * `sender` is the account that originated the contract call:
     *   - if using `revokeRole`, it is the admin role bearer
     *   - if using `renounceRole`, it is the role bearer (i.e. `account`)
     */
    event RoleRevoked(bytes4 indexed role, address indexed account, address indexed sender);

    /**
     * @dev Give msg.sender the ROOT role and create a LOCK role with itself as the admin role and no members. 
     * Calling setRoleAdmin(msg.sig, LOCK) means no one can grant that msg.sig role anymore.
     */
    constructor () {
        _grantRole(ROOT, msg.sender);   // Grant ROOT to msg.sender
        _setRoleAdmin(LOCK, LOCK);      // Create the LOCK role by setting itself as its own admin, creating an independent role tree
    }

    /**
     * @dev Each function in the contract has its own role, identified by their msg.sig signature.
     * ROOT can give and remove access to each function, lock any further access being granted to
     * a specific action, or even create other roles to delegate admin control over a function.
     */
    modifier auth() {
        require (_hasRole(msg.sig, msg.sender), "Access denied");
        _;
    }

    /**
     * @dev Allow only if the caller has been granted the admin role of `role`.
     */
    modifier admin(bytes4 role) {
        require (_hasRole(_getRoleAdmin(role), msg.sender), "Only admin");
        _;
    }

    /**
     * @dev Returns `true` if `account` has been granted `role`.
     */
    function hasRole(bytes4 role, address account) external view returns (bool) {
        return _hasRole(role, account);
    }

    /**
     * @dev Returns the admin role that controls `role`. See {grantRole} and
     * {revokeRole}.
     *
     * To change a role's admin, use {_setRoleAdmin}.
     */
    function getRoleAdmin(bytes4 role) external view returns (bytes4) {
        return _getRoleAdmin(role);
    }

    /**
     * @dev Sets `adminRole` as ``role``'s admin role.

     * If ``role``'s admin role is not `adminRole` emits a {RoleAdminChanged} event.
     *
     * Requirements:
     *
     * - the caller must have ``role``'s admin role.
     */
    function setRoleAdmin(bytes4 role, bytes4 adminRole) external virtual admin(role) {
        _setRoleAdmin(role, adminRole);
    }

    /**
     * @dev Grants `role` to `account`.
     *
     * If `account` had not been already granted `role`, emits a {RoleGranted}
     * event.
     *
     * Requirements:
     *
     * - the caller must have ``role``'s admin role.
     */
    function grantRole(bytes4 role, address account) external virtual admin(role) {
        _grantRole(role, account);
    }

    
    /**
     * @dev Grants all of `role` in `roles` to `account`.
     *
     * If `account` had not been already granted `role`, emits a {RoleGranted}
     * event.
     *
     * Requirements:
     *
     * - For each `role` in `roles`, the caller must have ``role``'s admin role.
     */
    function grantRoles(bytes4[] memory roles, address account) external virtual {
        for (uint256 i = 0; i < roles.length; i++) {
            require (_hasRole(_getRoleAdmin(roles[i]), msg.sender), "Only admin");
            _grantRole(roles[i], account);
        }
    }

    /**
     * @dev Sets LOCK as ``role``'s admin role. LOCK has no members, so this disables admin management of ``role``.

     * Emits a {RoleAdminChanged} event.
     *
     * Requirements:
     *
     * - the caller must have ``role``'s admin role.
     */
    function lockRole(bytes4 role) external virtual admin(role) {
        _setRoleAdmin(role, LOCK);
    }

    /**
     * @dev Revokes `role` from `account`.
     *
     * If `account` had been granted `role`, emits a {RoleRevoked} event.
     *
     * Requirements:
     *
     * - the caller must have ``role``'s admin role.
     */
    function revokeRole(bytes4 role, address account) external virtual admin(role) {
        _revokeRole(role, account);
    }

    /**
     * @dev Revokes all of `role` in `roles` from `account`.
     *
     * If `account` had been granted `role`, emits a {RoleRevoked} event.
     *
     * Requirements:
     *
     * - For each `role` in `roles`, the caller must have ``role``'s admin role.
     */
    function revokeRoles(bytes4[] memory roles, address account) external virtual {
        for (uint256 i = 0; i < roles.length; i++) {
            require (_hasRole(_getRoleAdmin(roles[i]), msg.sender), "Only admin");
            _revokeRole(roles[i], account);
        }
    }

    /**
     * @dev Revokes `role` from the calling account.
     *
     * Roles are often managed via {grantRole} and {revokeRole}: this function's
     * purpose is to provide a mechanism for accounts to lose their privileges
     * if they are compromised (such as when a trusted device is misplaced).
     *
     * If the calling account had been granted `role`, emits a {RoleRevoked}
     * event.
     *
     * Requirements:
     *
     * - the caller must be `account`.
     */
    function renounceRole(bytes4 role, address account) external virtual {
        require(account == msg.sender, "Renounce only for self");

        _revokeRole(role, account);
    }

    function _hasRole(bytes4 role, address account) internal view returns (bool) {
        return _roles[role].members[account];
    }

    function _getRoleAdmin(bytes4 role) internal view returns (bytes4) {
        return _roles[role].adminRole;
    }

    function _setRoleAdmin(bytes4 role, bytes4 adminRole) internal virtual {
        if (_getRoleAdmin(role) != adminRole) {
            _roles[role].adminRole = adminRole;
            emit RoleAdminChanged(role, adminRole);
        }
    }

    function _grantRole(bytes4 role, address account) internal {
        if (!_hasRole(role, account)) {
            _roles[role].members[account] = true;
            emit RoleGranted(role, account, msg.sender);
        }
    }

    function _revokeRole(bytes4 role, address account) internal {
        if (_hasRole(role, account)) {
            _roles[role].members[account] = false;
            emit RoleRevoked(role, account, msg.sender);
        }
    }
}

File 3 of 28 : MinimalTransferHelper.sol
// SPDX-License-Identifier: MIT
// Taken from https://github.com/Uniswap/uniswap-lib/blob/master/src/libraries/TransferHelper.sol

pragma solidity >=0.6.0;

import "./IERC20.sol";
import "../utils/RevertMsgExtractor.sol";


// helper methods for transferring ERC20 tokens that do not consistently return true/false
library MinimalTransferHelper {
    /// @notice Transfers tokens from msg.sender to a recipient
    /// @dev Errors with the underlying revert message if transfer fails
    /// @param token The contract address of the token which will be transferred
    /// @param to The recipient of the transfer
    /// @param value The value of the transfer
    function safeTransfer(
        IERC20 token,
        address to,
        uint256 value
    ) internal {
        (bool success, bytes memory data) = address(token).call(abi.encodeWithSelector(IERC20.transfer.selector, to, value));
        if (!(success && (data.length == 0 || abi.decode(data, (bool))))) revert(RevertMsgExtractor.getRevertMsg(data));
    }
}

File 4 of 28 : IERC20.sol
// SPDX-License-Identifier: MIT

pragma solidity ^0.8.0;

/**
 * @dev Interface of the ERC20 standard as defined in the EIP.
 */
interface IERC20 {
    /**
     * @dev Returns the amount of tokens in existence.
     */
    function totalSupply() external view returns (uint256);

    /**
     * @dev Returns the amount of tokens owned by `account`.
     */
    function balanceOf(address account) external view returns (uint256);

    /**
     * @dev Moves `amount` tokens from the caller's account to `recipient`.
     *
     * Returns a boolean value indicating whether the operation succeeded.
     *
     * Emits a {Transfer} event.
     */
    function transfer(address recipient, uint256 amount) external returns (bool);

    /**
     * @dev Returns the remaining number of tokens that `spender` will be
     * allowed to spend on behalf of `owner` through {transferFrom}. This is
     * zero by default.
     *
     * This value changes when {approve} or {transferFrom} are called.
     */
    function allowance(address owner, address spender) external view returns (uint256);

    /**
     * @dev Sets `amount` as the allowance of `spender` over the caller's tokens.
     *
     * Returns a boolean value indicating whether the operation succeeded.
     *
     * IMPORTANT: Beware that changing an allowance with this method brings the risk
     * that someone may use both the old and the new allowance by unfortunate
     * transaction ordering. One possible solution to mitigate this race
     * condition is to first reduce the spender's allowance to 0 and set the
     * desired value afterwards:
     * https://github.com/ethereum/EIPs/issues/20#issuecomment-263524729
     *
     * Emits an {Approval} event.
     */
    function approve(address spender, uint256 amount) external returns (bool);

    /**
     * @dev Moves `amount` tokens from `sender` to `recipient` using the
     * allowance mechanism. `amount` is then deducted from the caller's
     * allowance.
     *
     * Returns a boolean value indicating whether the operation succeeded.
     *
     * Emits a {Transfer} event.
     */
    function transferFrom(address sender, address recipient, uint256 amount) external returns (bool);

    /**
     * @dev Emitted when `value` tokens are moved from one account (`from`) to
     * another (`to`).
     *
     * Note that `value` may be zero.
     */
    event Transfer(address indexed from, address indexed to, uint256 value);

    /**
     * @dev Emitted when the allowance of a `spender` for an `owner` is set by
     * a call to {approve}. `value` is the new allowance.
     */
    event Approval(address indexed owner, address indexed spender, uint256 value);
}

File 5 of 28 : SafeERC20Namer.sol
// SPDX-License-Identifier: MIT
// Last audit: https://github.com/yieldprotocol/yield-utils-v2/commit/0d0b08b6b67cef6dfa69e6e6539bee542f49e25b
// Report: https://code4rena.com/reports/2021-05-yield

pragma solidity >=0.5.0;

import "../token/IERC20Metadata.sol";
import "../utils/AddressStringUtil.sol";

// produces token descriptors from inconsistent or absent ERC20 symbol implementations that can return string or bytes32
// this library will always produce a string symbol to represent the token
library SafeERC20Namer {
    function bytes32ToString(bytes32 x) private pure returns (string memory) {
        bytes memory bytesString = new bytes(32);
        uint256 charCount = 0;
        for (uint256 j = 0; j < 32; j++) {
            bytes1 char = x[j];
            if (char != 0) {
                bytesString[charCount] = char;
                charCount++;
            }
        }
        bytes memory bytesStringTrimmed = new bytes(charCount);
        for (uint256 j = 0; j < charCount; j++) {
            bytesStringTrimmed[j] = bytesString[j];
        }
        return string(bytesStringTrimmed);
    }

    // assumes the data is in position 2
    function parseStringData(bytes memory b) private pure returns (string memory) {
        uint256 charCount = 0;
        // first parse the charCount out of the data
        for (uint256 i = 32; i < 64; i++) {
            charCount <<= 8;
            charCount += uint8(b[i]);
        }

        bytes memory bytesStringTrimmed = new bytes(charCount);
        for (uint256 i = 0; i < charCount; i++) {
            bytesStringTrimmed[i] = b[i + 64];
        }

        return string(bytesStringTrimmed);
    }

    // uses a heuristic to produce a token name from the address
    // the heuristic returns the full hex of the address string in upper case
    function addressToName(address token) private pure returns (string memory) {
        return AddressStringUtil.toAsciiString(token, 40);
    }

    // uses a heuristic to produce a token symbol from the address
    // the heuristic returns the first 6 hex of the address string in upper case
    function addressToSymbol(address token) private pure returns (string memory) {
        return AddressStringUtil.toAsciiString(token, 6);
    }

    // calls an external view token contract method that returns a symbol or name, and parses the output into a string
    function callAndParseStringReturn(address token, bytes4 selector) private view returns (string memory) {
        (bool success, bytes memory data) = token.staticcall(abi.encodeWithSelector(selector));
        // if not implemented, or returns empty data, return empty string
        if (!success || data.length == 0) {
            return "";
        }
        // bytes32 data always has length 32
        if (data.length == 32) {
            bytes32 decoded = abi.decode(data, (bytes32));
            return bytes32ToString(decoded);
        } else if (data.length > 64) {
            return abi.decode(data, (string));
        }
        return "";
    }

    // attempts to extract the token symbol. if it does not implement symbol, returns a symbol derived from the address
    function tokenSymbol(address token) public view returns (string memory) {
        string memory symbol = callAndParseStringReturn(token, IERC20Metadata.symbol.selector);
        if (bytes(symbol).length == 0) {
            // fallback to 6 uppercase hex of address
            return addressToSymbol(token);
        }
        return symbol;
    }

    // attempts to extract the token name. if it does not implement name, returns a name derived from the address
    function tokenName(address token) public view returns (string memory) {
        string memory name = callAndParseStringReturn(token, IERC20Metadata.name.selector);
        if (bytes(name).length == 0) {
            // fallback to full hex of address
            return addressToName(token);
        }
        return name;
    }

    /// @notice Provides a safe ERC20.decimals version which returns '0' as fallback value.
    /// @param token The address of the ERC-20 token contract.
    /// @return (uint8) Token decimals.
    function tokenDecimals(address token) public view returns (uint8) {
        (bool success, bytes memory data) = token.staticcall(abi.encodeWithSelector(IERC20Metadata.decimals.selector));
        return success && data.length == 32 ? abi.decode(data, (uint8)) : 0;
    }
}

File 6 of 28 : StrategyMigrator.sol
// SPDX-License-Identifier: BUSL-1.1
pragma solidity ^0.8.13;

import {IStrategyMigrator} from "./interfaces/IStrategyMigrator.sol";
import {IFYToken} from "@yield-protocol/vault-v2/src/interfaces/IFYToken.sol";
import {IERC20} from "@yield-protocol/utils-v2/src/token/IERC20.sol";


/// @dev The Migrator contract poses as a Pool to receive all assets from a Strategy during an invest call.
/// TODO: For this to work, the implementing class must inherit from ERC20.
abstract contract StrategyMigrator is IStrategyMigrator {

    /// Mock pool base - Must match that of the calling strategy
    IERC20 public immutable base;

    /// Mock pool fyToken - Can be any address
    IFYToken public fyToken;

    /// Mock pool maturity - Can be set to a value far in the future to avoid `divest` calls
    uint32 public maturity;

    constructor(IERC20 base_, IFYToken fyToken_) {
        base = base_;
        fyToken = fyToken_;
    }

    /// @dev Mock pool init. Called within `invest`.
    function init(address)
        external
        virtual
        returns (uint256, uint256, uint256)
    {
        return (0, 0, 0);
    }

    /// @dev Mock pool burn that reverts so that `divest` never suceeds, but `eject` does.
    function burn(address, address, uint256, uint256)
        external
        virtual
        returns (uint256, uint256, uint256)
    {
        revert();
    }
}

File 7 of 28 : ERC20Rewards.sol
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.0;

import "./IERC20.sol";
import "./ERC20Permit.sol";
import "../access/AccessControl.sol";
import "../utils/RevertMsgExtractor.sol";
import "../token/MinimalTransferHelper.sol";
import "../utils/Cast.sol";


/// @dev A token inheriting from ERC20Rewards will reward token holders with a rewards token.
/// The rewarded amount will be a fixed wei per second, distributed proportionally to token holders
/// by the size of their holdings.
contract ERC20Rewards is AccessControl, ERC20Permit {
    using MinimalTransferHelper for IERC20;
    using Cast for uint256;

    event RewardsTokenSet(IERC20 token);
    event RewardsSet(uint32 start, uint32 end, uint256 rate);
    event RewardsPerTokenUpdated(uint256 accumulated);
    event UserRewardsUpdated(address user, uint256 userRewards, uint256 paidRewardPerToken);
    event Claimed(address user, address receiver, uint256 claimed);

    struct RewardsPeriod {
        uint32 start;                                   // Start time for the current rewardsToken schedule
        uint32 end;                                     // End time for the current rewardsToken schedule
    }

    struct RewardsPerToken {
        uint128 accumulated;                            // Accumulated rewards per token for the period, scaled up by 1e18
        uint32 lastUpdated;                             // Last time the rewards per token accumulator was updated
        uint96 rate;                                    // Wei rewarded per second among all token holders
    }

    struct UserRewards {
        uint128 accumulated;                            // Accumulated rewards for the user until the checkpoint
        uint128 checkpoint;                             // RewardsPerToken the last time the user rewards were updated
    }

    IERC20 public rewardsToken;                         // Token used as rewards
    RewardsPeriod public rewardsPeriod;                 // Period in which rewards are accumulated by users

    RewardsPerToken public rewardsPerToken;             // Accumulator to track rewards per token               
    mapping (address => UserRewards) public rewards;    // Rewards accumulated by users
    
    constructor(string memory name, string memory symbol, uint8 decimals)
        ERC20Permit(name, symbol, decimals)
    { }

    /// @dev Return the earliest of two timestamps
    function earliest(uint32 x, uint32 y) internal pure returns (uint32 z) {
        z = (x < y) ? x : y;
    }

    /// @dev Set a rewards token.
    /// @notice Careful, this can only be done once.
    function setRewardsToken(IERC20 rewardsToken_)
        external
        auth
    {
        require(rewardsToken == IERC20(address(0)), "Rewards token already set");
        rewardsToken = rewardsToken_;
        emit RewardsTokenSet(rewardsToken_);
    }

    /// @dev Set a rewards schedule
    function setRewards(uint32 start, uint32 end, uint96 rate)
        external
        auth
    {
        require(
            start <= end,
            "Incorrect input"
        );
        require(
            rewardsToken != IERC20(address(0)),
            "Rewards token not set"
        );
        // A new rewards program can be set if one is not running
        require(
            block.timestamp.u32() < rewardsPeriod.start || block.timestamp.u32() > rewardsPeriod.end,
            "Ongoing rewards"
        );

        // Update the rewards per token so that we don't lose any rewards
        _updateRewardsPerToken();

        rewardsPeriod.start = start;
        rewardsPeriod.end = end;

        // If setting up a new rewards program, the rewardsPerToken.accumulated is used and built upon
        // New rewards start accumulating from the new rewards program start
        // Any unaccounted rewards from last program can still be added to the user rewards
        // Any unclaimed rewards can still be claimed
        rewardsPerToken.lastUpdated = start;
        rewardsPerToken.rate = rate;

        emit RewardsSet(start, end, rate);
    }

    /// @dev Update the rewards per token accumulator.
    /// @notice Needs to be called on each liquidity event
    function _updateRewardsPerToken() internal {
        RewardsPerToken memory rewardsPerToken_ = rewardsPerToken;
        RewardsPeriod memory rewardsPeriod_ = rewardsPeriod;
        uint256 totalSupply_ = _totalSupply;

        // We skip the update if the program hasn't started
        if (block.timestamp.u32() < rewardsPeriod_.start) return;

        // Find out the unaccounted time
        uint32 end = earliest(block.timestamp.u32(), rewardsPeriod_.end);
        uint256 unaccountedTime = end - rewardsPerToken_.lastUpdated; // Cast to uint256 to avoid overflows later on
        if (unaccountedTime == 0) return; // We skip the storage changes if already updated in the same block

        // Calculate and update the new value of the accumulator. unaccountedTime casts it into uint256, which is desired.
        // If the first mint happens mid-program, we don't update the accumulator, no one gets the rewards for that period.
        if (totalSupply_ != 0) rewardsPerToken_.accumulated = (rewardsPerToken_.accumulated + 1e18 * unaccountedTime * rewardsPerToken_.rate / totalSupply_).u128(); // The rewards per token are scaled up for precision
        rewardsPerToken_.lastUpdated = end;
        rewardsPerToken = rewardsPerToken_;
        
        emit RewardsPerTokenUpdated(rewardsPerToken_.accumulated);
    }

    /// @dev Accumulate rewards for an user.
    /// @notice Needs to be called on each liquidity event, or when user balances change.
    function _updateUserRewards(address user) internal returns (uint128) {
        UserRewards memory userRewards_ = rewards[user];
        RewardsPerToken memory rewardsPerToken_ = rewardsPerToken;
        
        // Calculate and update the new value user reserves. _balanceOf[user] casts it into uint256, which is desired.
        userRewards_.accumulated = (userRewards_.accumulated + _balanceOf[user] * (rewardsPerToken_.accumulated - userRewards_.checkpoint) / 1e18).u128(); // We must scale down the rewards by the precision factor
        userRewards_.checkpoint = rewardsPerToken_.accumulated;
        rewards[user] = userRewards_;
        emit UserRewardsUpdated(user, userRewards_.accumulated, userRewards_.checkpoint);

        return userRewards_.accumulated;
    }

    /// @dev Mint tokens, after accumulating rewards for an user and update the rewards per token accumulator.
    function _mint(address dst, uint256 wad)
        internal virtual override
        returns (bool)
    {
        _updateRewardsPerToken();
        _updateUserRewards(dst);
        return super._mint(dst, wad);
    }

    /// @dev Burn tokens, after accumulating rewards for an user and update the rewards per token accumulator.
    function _burn(address src, uint256 wad)
        internal virtual override
        returns (bool)
    {
        _updateRewardsPerToken();
        _updateUserRewards(src);
        return super._burn(src, wad);
    }

    /// @dev Transfer tokens, after updating rewards for source and destination.
    function _transfer(address src, address dst, uint wad) internal virtual override returns (bool) {
        _updateRewardsPerToken();
        _updateUserRewards(src);
        _updateUserRewards(dst);
        return super._transfer(src, dst, wad);
    }

    /// @dev Claim all rewards from caller into a given address
    function claim(address to)
        external
        returns (uint256 claiming)
    {
        claiming = _claim(msg.sender, to);
    }

    /// @dev Trigger a claim for any user
    function remit(address user)
        external
        returns (uint256 claiming)
    {
        claiming = _claim(user, user);
    }

    /// @dev Claim all rewards from a user into an arbitrary receiver
    function _claim(address from, address to)
        internal
        returns (uint256 claiming)
    {
        _updateRewardsPerToken();
        claiming = _updateUserRewards(from);
        rewards[from].accumulated = 0; // A Claimed event implies the rewards were set to zero
        rewardsToken.safeTransfer(to, claiming);
        emit Claimed(from, to, claiming);
    }
}

File 8 of 28 : IStrategy.sol
// SPDX-License-Identifier: BUSL-1.1
pragma solidity >=0.8.13;

import {IStrategyMigrator} from "./IStrategyMigrator.sol";
import {IERC20} from "@yield-protocol/utils-v2/src/token/IERC20.sol";
import {IFYToken} from "@yield-protocol/vault-v2/src/interfaces/IFYToken.sol";
import {ICauldron} from "@yield-protocol/vault-v2/src/interfaces/ICauldron.sol";
import {ILadle} from "@yield-protocol/vault-v2/src/interfaces/ILadle.sol";
import {IPool} from "@yield-protocol/yieldspace-tv/src/interfaces/IPool.sol";


/// @dev The Strategy contract allows liquidity providers to provide liquidity in underlying
/// and receive strategy tokens that represent a stake in a YieldSpace pool contract.
/// Upon maturity, the strategy can `divest` from the mature pool, becoming a proportional
/// ownership underlying vault. When not invested, the strategy can `invest` into a Pool using
/// all its underlying.
/// The strategy can also `eject` from a Pool before maturity, immediately converting its assets
/// to underlying as much as possible. If any fyToken can't be exchanged for underlying, the
/// strategy will hold them until maturity when `redeemEjected` can be used.
interface IStrategy is IStrategyMigrator {
    enum State {DEPLOYED, DIVESTED, INVESTED, EJECTED, DRAINED}

    function state() external view returns(State);                          // The state determines which functions are available
    function base() external view returns(IERC20);                          // Base token for this strategy (inherited from StrategyMigrator)
    function fyToken() external view returns(IFYToken);                     // Current fyToken for this strategy (inherited from StrategyMigrator)
    function pool() external view returns(IPool);                           // Current pool that this strategy invests in
    function cached() external view returns(uint256);                       // Base tokens owned by the strategy after the last operation
    function fyTokenCached() external view returns(uint256);                // In emergencies, the strategy can keep fyToken of one series

    /// @dev Mint the first strategy tokens, without investing
    function init(address to)
        external
        returns (uint256 baseIn, uint256 fyTokenIn, uint256 minted);

    /// @dev Start the strategy investments in the next pool
    /// @notice When calling this function for the first pool, some underlying needs to be transferred to the strategy first, using a batchable router.
    function invest(IPool pool_)
        external
        returns (uint256 poolTokensObtained);


    /// @dev Divest out of a pool once it has matured
    function divest()
        external
        returns (uint256 baseObtained);

    /// @dev Divest out of a pool at any time. If possible the pool tokens will be burnt for base and fyToken, the latter of which
    /// must be sold to return the strategy to a functional state. If the pool token burn reverts, the pool tokens will be transferred
    /// to the caller as a last resort.
    /// @notice The caller must take care of slippage when selling fyToken, if relevant.
    function eject()
        external
        returns (uint256 baseObtained, uint256 fyTokenObtained);

    /// @dev Buy ejected fyToken in the strategy at face value
    /// @param fyTokenTo Address to send the purchased fyToken to.
    /// @param baseTo Address to send any remaining base to.
    /// @return soldFYToken Amount of fyToken sold.
    /// @return returnedBase Amount of base unused and returned.
    function buyFYToken(address fyTokenTo, address baseTo)
        external
        returns (uint256 soldFYToken, uint256 returnedBase);

    /// @dev If we ejected the pool tokens, we can recapitalize the strategy to avoid a forced migration
    function restart()
        external
        returns (uint256 baseIn);

    /// @dev Mint strategy tokens with pool tokens. It can be called only when invested.
    /// @notice The pool tokens that the user contributes need to have been transferred previously, using a batchable router.
    function mint(address to)
        external
        returns (uint256 minted);

    /// @dev Burn strategy tokens to withdraw pool tokens. It can be called only when invested.
    /// @notice The strategy tokens that the user burns need to have been transferred previously, using a batchable router.
    function burn(address to)
        external
        returns (uint256 poolTokensObtained);

    /// @dev Mint strategy tokens with base tokens. It can be called only when not invested and not ejected.
    /// @notice The base tokens that the user invests need to have been transferred previously, using a batchable router.
    function mintDivested(address to)
        external
        returns (uint256 minted);
    
    /// @dev Burn strategy tokens to withdraw base tokens. It can be called when not invested and not ejected.
    /// @notice The strategy tokens that the user burns need to have been transferred previously, using a batchable router.
    function burnDivested(address baseTo)
        external
        returns (uint256 baseObtained);

    /// @dev Token used as rewards
    function rewardsToken() external view returns(IERC20);
    
    /// @dev Rewards schedule
    function rewardsPeriod() external view returns(uint32 start, uint32 end);

    /// @dev Rewards per token
    function rewardsPerToken() external view returns(uint128 accumulated, uint32 lastUpdated, uint96 rate);
    
    /// @dev Rewards accumulated by users
    function rewards(address user) external view returns(uint128 accumulatedUserStart, uint128 accumulatedCheckpoint);

    /// @dev Set the rewards token
    function setRewardsToken(IERC20 rewardsToken_)
        external;

    /// @dev Set a rewards schedule
    function setRewards(uint32 start, uint32 end, uint96 rate)
        external;

    /// @dev Claim all rewards from caller into a given address
    function claim(address to)
        external
        returns (uint256 claiming);

    /// @dev Trigger a claim for any user
    function remit(address user)
        external
        returns (uint256 claiming);
}

File 9 of 28 : IPool.sol
// SPDX-License-Identifier: MIT
pragma solidity >= 0.8.0;
import "@yield-protocol/utils-v2/src/token/IERC20.sol";
import "@yield-protocol/utils-v2/src/token/IERC2612.sol";
import {IMaturingToken} from "./IMaturingToken.sol";
import {IERC20Metadata} from  "@yield-protocol/utils-v2/src/token/ERC20.sol";

interface IPool is IERC20, IERC2612 {
    function baseToken() external view returns(IERC20Metadata);
    function base() external view returns(IERC20);
    function burn(address baseTo, address fyTokenTo, uint256 minRatio, uint256 maxRatio) external returns (uint256, uint256, uint256);
    function burnForBase(address to, uint256 minRatio, uint256 maxRatio) external returns (uint256, uint256);
    function buyBase(address to, uint128 baseOut, uint128 max) external returns(uint128);
    function buyBasePreview(uint128 baseOut) external view returns(uint128);
    function buyFYToken(address to, uint128 fyTokenOut, uint128 max) external returns(uint128);
    function buyFYTokenPreview(uint128 fyTokenOut) external view returns(uint128);
    function currentCumulativeRatio() external view returns (uint256 currentCumulativeRatio_, uint256 blockTimestampCurrent);
    function cumulativeRatioLast() external view returns (uint256);
    function fyToken() external view returns(IMaturingToken);
    function g1() external view returns(int128);
    function g2() external view returns(int128);
    function getC() external view returns (int128);
    function getCurrentSharePrice() external view returns (uint256);
    function getCache() external view returns (uint104 baseCached, uint104 fyTokenCached, uint32 blockTimestampLast, uint16 g1Fee_);
    function getBaseBalance() external view returns(uint128);
    function getFYTokenBalance() external view returns(uint128);
    function getSharesBalance() external view returns(uint128);
    function init(address to) external returns (uint256, uint256, uint256);
    function maturity() external view returns(uint32);
    function mint(address to, address remainder, uint256 minRatio, uint256 maxRatio) external returns (uint256, uint256, uint256);
    function mu() external view returns (int128);
    function mintWithBase(address to, address remainder, uint256 fyTokenToBuy, uint256 minRatio, uint256 maxRatio) external returns (uint256, uint256, uint256);
    function retrieveBase(address to) external returns(uint128 retrieved);
    function retrieveFYToken(address to) external returns(uint128 retrieved);
    function retrieveShares(address to) external returns(uint128 retrieved);
    function scaleFactor() external view returns(uint96);
    function sellBase(address to, uint128 min) external returns(uint128);
    function sellBasePreview(uint128 baseIn) external view returns(uint128);
    function sellFYToken(address to, uint128 min) external returns(uint128);
    function sellFYTokenPreview(uint128 fyTokenIn) external view returns(uint128);
    function setFees(uint16 g1Fee_) external;
    function sharesToken() external view returns(IERC20Metadata);
    function ts() external view returns(int128);
    function wrap(address receiver) external returns (uint256 shares);
    function wrapPreview(uint256 assets) external view returns (uint256 shares);
    function unwrap(address receiver) external returns (uint256 assets);
    function unwrapPreview(uint256 shares) external view returns (uint256 assets);
    /// Returns the max amount of FYTokens that can be sold to the pool
    function maxFYTokenIn() external view returns (uint128) ;
    /// Returns the max amount of FYTokens that can be bought from the pool
    function maxFYTokenOut() external view returns (uint128) ;
    /// Returns the max amount of Base that can be sold to the pool
    function maxBaseIn() external view returns (uint128) ;
    /// Returns the max amount of Base that can be bought from the pool
    function maxBaseOut() external view returns (uint128);
    /// Returns the result of the total supply invariant function
    function invariant() external view returns (uint128);
}

File 10 of 28 : IFYToken.sol
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.0;
import "./IERC5095.sol";
import "./IJoin.sol";
import "./IOracle.sol";

interface IFYToken is IERC5095 {

    /// @dev Oracle for the savings rate.
    function oracle() view external returns (IOracle);

    /// @dev Source of redemption funds.
    function join() view external returns (IJoin); 

    /// @dev Asset to be paid out on redemption.
    function underlying() view external returns (address);

    /// @dev Yield id of the asset to be paid out on redemption.
    function underlyingId() view external returns (bytes6);

    /// @dev Time at which redemptions are enabled.
    function maturity() view external returns (uint256);

    /// @dev Spot price (exchange rate) between the base and an interest accruing token at maturity, set to 2^256-1 before maturity
    function chiAtMaturity() view external returns (uint256);
    
    /// @dev Record price data at maturity
    function mature() external;

    /// @dev Mint fyToken providing an equal amount of underlying to the protocol
    function mintWithUnderlying(address to, uint256 amount) external;

    /// @dev Burn fyToken after maturity for an amount of underlying.
    function redeem(address to, uint256 amount) external returns (uint256);

    /// @dev Mint fyToken.
    /// This function can only be called by other Yield contracts, not users directly.
    /// @param to Wallet to mint the fyToken in.
    /// @param fyTokenAmount Amount of fyToken to mint.
    function mint(address to, uint256 fyTokenAmount) external;

    /// @dev Burn fyToken.
    /// This function can only be called by other Yield contracts, not users directly.
    /// @param from Wallet to burn the fyToken from.
    /// @param fyTokenAmount Amount of fyToken to burn.
    function burn(address from, uint256 fyTokenAmount) external;
}

File 11 of 28 : RevertMsgExtractor.sol
// SPDX-License-Identifier: MIT
// Taken from https://github.com/sushiswap/BoringSolidity/blob/441e51c0544cf2451e6116fe00515e71d7c42e2c/src/BoringBatchable.sol

pragma solidity >=0.6.0;


library RevertMsgExtractor {
    /// @dev Helper function to extract a useful revert message from a failed call.
    /// If the returned data is malformed or not correctly abi encoded then this call can fail itself.
    function getRevertMsg(bytes memory returnData)
        internal pure
        returns (string memory)
    {
        // If the _res length is less than 68, then the transaction failed silently (without a revert message)
        if (returnData.length < 68) return "Transaction reverted silently";

        assembly {
            // Slice the sighash.
            returnData := add(returnData, 0x04)
        }
        return abi.decode(returnData, (string)); // All that remains is the revert string
    }
}

File 12 of 28 : IERC20Metadata.sol
// SPDX-License-Identifier: MIT
// Taken from https://github.com/OpenZeppelin/openzeppelin-contracts/blob/master/src/token/ERC20/extensions/IERC20Metadata.sol

pragma solidity ^0.8.0;

import "./IERC20.sol";

/**
 * @dev Interface for the optional metadata functions from the ERC20 standard.
 */
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);
}

File 13 of 28 : AddressStringUtil.sol
// SPDX-License-Identifier: MIT

pragma solidity >=0.5.0;

library AddressStringUtil {
    // converts an address to the uppercase hex string, extracting only len bytes (up to 20, multiple of 2)
    function toAsciiString(address addr, uint256 len) internal pure returns (string memory) {
        require(len % 2 == 0 && len > 0 && len <= 40, "AddressStringUtil: INVALID_LEN");
        bytes memory s = new bytes(len);
        uint256 addrNum = uint256(uint160(addr));
        for (uint256 ii = 0; ii < len ; ii +=2) {
            uint8 b = uint8(addrNum >> (4 * (38 - ii)));
            s[ii] = char(b >> 4);
            s[ii + 1] = char(b & 0x0f);
        }
        return string(s);
    }

    // hi and lo are only 4 bits and between 0 and 16
    // this method converts those values to the unicode/ascii code point for the hex representation
    // uses upper case for the characters
    function char(uint8 b) private pure returns (bytes1 c) {
        if (b < 10) {
            return bytes1(b + 0x30);
        } else {
            return bytes1(b + 0x37);
        }
    }
}

File 14 of 28 : IStrategyMigrator.sol
// SPDX-License-Identifier: BUSL-1.1
pragma solidity ^0.8.13;

import {IFYToken} from "@yield-protocol/vault-v2/src/interfaces/IFYToken.sol";
import {IERC20} from "@yield-protocol/utils-v2/src/token/IERC20.sol";


/// @dev The Migrator contract poses as a Pool to receive all assets from a Strategy
/// during a roll operation.
/// @notice The Pool and fyToken must exist. The fyToken needs to be not mature, and the pool needs to have no fyToken in it.
/// There will be no state changes on pool or fyToken.
interface IStrategyMigrator is IERC20 {

    /// @dev Mock pool base - Must match that of the calling strategy
    function base() external view returns(IERC20);

    /// @dev Mock pool fyToken - Can be any address, including address(0)
    function fyToken() external view returns(IFYToken);

    /// @dev Mock pool init. Called within `invest`.
    function init(address) external returns (uint256, uint256, uint256);

    /// @dev Mock pool burn and make it revert so that `endPool`never suceeds, and `burnForBase` can never be called.
    function burn(address, address, uint256, uint256) external returns  (uint256, uint256, uint256);
}

File 15 of 28 : IERC5095.sol
// SPDX-License-Identifier: MIT
pragma solidity >=0.8.13;

import "@yield-protocol/utils-v2/src/token/IERC20.sol";

interface IERC5095 is IERC20 {
    /// @dev Asset that is returned on redemption.
    function underlying() external view returns (address underlyingAddress);

    /// @dev Unix time at which redemption of fyToken for underlying are possible
    function maturity() external view returns (uint256 timestamp);

    /// @dev Converts a specified amount of principal to underlying
    function convertToUnderlying(uint256 principalAmount) external returns (uint256 underlyingAmount);

    /// @dev Converts a specified amount of underlying to principal
    function convertToPrincipal(uint256 underlyingAmount) external returns (uint256 principalAmount);

    /// @dev Gives the maximum amount an address holder can redeem in terms of the principal
    function maxRedeem(address holder) external view returns (uint256 maxPrincipalAmount);

    /// @dev Gives the amount in terms of underlying that the princiapl amount can be redeemed for plus accrual
    function previewRedeem(uint256 principalAmount) external returns (uint256 underlyingAmount);

    /// @dev Burn fyToken after maturity for an amount of principal.
    function redeem(uint256 principalAmount, address to, address from) external returns (uint256 underlyingAmount);

    /// @dev Gives the maximum amount an address holder can withdraw in terms of the underlying
    function maxWithdraw(address holder) external returns (uint256 maxUnderlyingAmount);

    /// @dev Gives the amount in terms of principal that the underlying amount can be withdrawn for plus accrual
    function previewWithdraw(uint256 underlyingAmount) external returns (uint256 principalAmount);

    /// @dev Burn fyToken after maturity for an amount of underlying.
    function withdraw(uint256 underlyingAmount, address to, address from) external returns (uint256 principalAmount);
}

File 16 of 28 : IOracle.sol
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.0;

interface IOracle {
    /**
     * @notice Doesn't refresh the price, but returns the latest value available without doing any transactional operations
     * @param base The asset in which the `amount` to be converted is represented
     * @param quote The asset in which the converted `value` will be represented
     * @param amount The amount to be converted from `base` to `quote`
     * @return value The converted value of `amount` from `base` to `quote`
     * @return updateTime The timestamp when the conversion price was taken
     */
    function peek(
        bytes32 base,
        bytes32 quote,
        uint256 amount
    ) external view returns (uint256 value, uint256 updateTime);

    /**
     * @notice Does whatever work or queries will yield the most up-to-date price, and returns it.
     * @param base The asset in which the `amount` to be converted is represented
     * @param quote The asset in which the converted `value` will be represented
     * @param amount The amount to be converted from `base` to `quote`
     * @return value The converted value of `amount` from `base` to `quote`
     * @return updateTime The timestamp when the conversion price was taken
     */
    function get(
        bytes32 base,
        bytes32 quote,
        uint256 amount
    ) external returns (uint256 value, uint256 updateTime);
}

File 17 of 28 : IJoin.sol
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.0;
import "@yield-protocol/utils-v2/src/token/IERC20.sol";

interface IJoin {
    /// @dev asset managed by this contract
    function asset() external view returns (address);

    /// @dev amount of assets held by this contract
    function storedBalance() external view returns (uint256);

    /// @dev Add tokens to this contract.
    function join(address user, uint128 wad) external returns (uint128);

    /// @dev Remove tokens to this contract.
    function exit(address user, uint128 wad) external returns (uint128);

    /// @dev Retrieve any tokens other than the `asset`. Useful for airdropped tokens.
    function retrieve(IERC20 token, address to) external;
}

File 18 of 28 : Cast.sol
// SPDX-License-Identifier: BUSL-1.1
pragma solidity ^0.8.13;

library Cast {
    ///@dev library for safe casting of value types

    function b12(bytes32 x) internal pure returns (bytes12 y) {
        require(bytes32(y = bytes12(x)) == x, "Cast overflow");
    }

    function b6(bytes32 x) internal pure returns (bytes6 y) {
        require(bytes32(y = bytes6(x)) == x, "Cast overflow");
    }

    function u256(int256 x) internal pure returns (uint256 y) {
        require(x >= 0, "Cast overflow");
        y = uint256(x);
    }

    function i256(uint256 x) internal pure returns (int256 y) {
        require(x <= uint256(type(int256).max), "Cast overflow");
        y = int256(x);
    }

    function u128(uint256 x) internal pure returns (uint128 y) {
        require(x <= type(uint128).max, "Cast overflow");
        y = uint128(x);
    }

    function u128(int256 x) internal pure returns (uint128 y) {
        require(x >= 0, "Cast overflow");
        y = uint128(uint256(x));
    }

    function i128(uint256 x) internal pure returns (int128) {
        require(x <= uint256(int256(type(int128).max)), "Cast overflow");
        return int128(int256(x));
    }

    function i128(int256 x) internal pure returns (int128) {
        require(x <= type(int128).max, "Cast overflow");
        require(x >= type(int128).min, "Cast overflow");
        return int128(x);
    }

    function u112(uint256 x) internal pure returns (uint112 y) {
        require(x <= type(uint112).max, "Cast overflow");
        y = uint112(x);
    }

    function u104(uint256 x) internal pure returns (uint104 y) {
        require(x <= type(uint104).max, "Cast overflow");
        y = uint104(x);
    }

    function u96(uint256 x) internal pure returns (uint96 y) {
        require(x <= type(uint96).max, "Cast overflow");
        y = uint96(x);
    }

    function u32(uint256 x) internal pure returns (uint32 y) {
        require(x <= type(uint32).max, "Cast overflow");
        y = uint32(x);
    }
}

File 19 of 28 : ERC20Permit.sol
// SPDX-License-Identifier: MIT
// Adapted from https://github.com/OpenZeppelin/openzeppelin-contracts/blob/53516bc555a454862470e7860a9b5254db4d00f5/src/token/ERC20/ERC20Permit.sol
pragma solidity ^0.8.0;

import "./ERC20.sol";
import "./IERC2612.sol";

/**
 * @dev Extension of {ERC20} that allows token holders to use their tokens
 * without sending any transactions by setting {IERC20-allowance} with a
 * signature using the {permit} method, and then spend them via
 * {IERC20-transferFrom}.
 *
 * The {permit} signature mechanism conforms to the {IERC2612} interface.
 */
abstract contract ERC20Permit is ERC20, IERC2612 {
    mapping (address => uint256) public override nonces;

    bytes32 public immutable PERMIT_TYPEHASH = keccak256("Permit(address owner,address spender,uint256 value,uint256 nonce,uint256 deadline)");
    bytes32 private immutable _DOMAIN_SEPARATOR;
    uint256 public immutable deploymentChainId;

    constructor(string memory name_, string memory symbol_, uint8 decimals_) ERC20(name_, symbol_, decimals_) {
        deploymentChainId = block.chainid;
        _DOMAIN_SEPARATOR = _calculateDomainSeparator(block.chainid);
    }

    /// @dev Calculate the DOMAIN_SEPARATOR.
    function _calculateDomainSeparator(uint256 chainId) private view returns (bytes32) {
        return keccak256(
            abi.encode(
                keccak256("EIP712Domain(string name,string version,uint256 chainId,address verifyingContract)"),
                keccak256(bytes(name)),
                keccak256(bytes(version())),
                chainId,
                address(this)
            )
        );
    }

    /// @dev Return the DOMAIN_SEPARATOR.
    function DOMAIN_SEPARATOR() external view returns (bytes32) {
        return block.chainid == deploymentChainId ? _DOMAIN_SEPARATOR : _calculateDomainSeparator(block.chainid);
    }

    /// @dev Setting the version as a function so that it can be overriden
    function version() public pure virtual returns(string memory) { return "1"; }

    /**
     * @dev See {IERC2612-permit}.
     *
     * In cases where the free option is not a concern, deadline can simply be
     * set to uint(-1), so it should be seen as an optional parameter
     */
    function permit(address owner, address spender, uint256 amount, uint256 deadline, uint8 v, bytes32 r, bytes32 s) external virtual override {
        require(deadline >= block.timestamp, "ERC20Permit: expired deadline");

        bytes32 hashStruct = keccak256(
            abi.encode(
                PERMIT_TYPEHASH,
                owner,
                spender,
                amount,
                nonces[owner]++,
                deadline
            )
        );

        bytes32 hash = keccak256(
            abi.encodePacked(
                "\x19\x01",
                block.chainid == deploymentChainId ? _DOMAIN_SEPARATOR : _calculateDomainSeparator(block.chainid),
                hashStruct
            )
        );

        address signer = ecrecover(hash, v, r, s);
        require(
            signer != address(0) && signer == owner,
            "ERC20Permit: invalid signature"
        );

        _setAllowance(owner, spender, amount);
    }
}

File 20 of 28 : IERC2612.sol
// SPDX-License-Identifier: MIT
// Code adapted from https://github.com/OpenZeppelin/openzeppelin-contracts/pull/2237/
pragma solidity ^0.8.0;

/**
 * @dev Interface of the ERC2612 standard as defined in the EIP.
 *
 * Adds the {permit} method, which can be used to change one's
 * {IERC20-allowance} without having to send a transaction, by signing a
 * message. This allows users to spend tokens without having to hold Ether.
 *
 * See https://eips.ethereum.org/EIPS/eip-2612.
 */
interface IERC2612 {
    /**
     * @dev Sets `amount` 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:
     *
     * - `owner` cannot be the zero address.
     * - `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 amount, uint256 deadline, uint8 v, bytes32 r, bytes32 s) external;

    /**
     * @dev Returns the current ERC2612 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);
}

File 21 of 28 : ERC20.sol
// SPDX-License-Identifier: MIT
// Inspired on token.sol from DappHub. Natspec adpated from OpenZeppelin.

pragma solidity ^0.8.0;
import "./IERC20Metadata.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}.
 *
 * We have followed general OpenZeppelin guidelines: functions revert instead
 * of returning `false` on failure. This behavior is nonetheless conventional
 * and does not conflict with the expectations of ERC20 applications.
 *
 * Additionally, an {Approval} event is emitted on calls to {transferFrom}.
 * This allows applications to reconstruct the allowance for all accounts just
 * by listening to said events. Other implementations of the EIP may not emit
 * these events, as it isn't required by the specification.
 * 
 * Calls to {transferFrom} do not check for allowance if the caller is the owner
 * of the funds. This allows to reduce the number of approvals that are necessary.
 *
 * Finally, {transferFrom} does not decrease the allowance if it is set to
 * type(uint256).max. This reduces the gas costs without any likely impact.
 */
contract ERC20 is IERC20Metadata {
    uint256                                           internal  _totalSupply;
    mapping (address => uint256)                      internal  _balanceOf;
    mapping (address => mapping (address => uint256)) internal  _allowance;
    string                                            public override name = "???";
    string                                            public override symbol = "???";
    uint8                                             public override decimals = 18;

    /**
     *  @dev Sets the values for {name}, {symbol} and {decimals}.
     */
    constructor(string memory name_, string memory symbol_, uint8 decimals_) {
        name = name_;
        symbol = symbol_;
        decimals = decimals_;
    }

    /**
     * @dev See {IERC20-totalSupply}.
     */
    function totalSupply() external view virtual override returns (uint256) {
        return _totalSupply;
    }

    /**
     * @dev See {IERC20-balanceOf}.
     */
    function balanceOf(address guy) external view virtual override returns (uint256) {
        return _balanceOf[guy];
    }

    /**
     * @dev See {IERC20-allowance}.
     */
    function allowance(address owner, address spender) external view virtual override returns (uint256) {
        return _allowance[owner][spender];
    }

    /**
     * @dev See {IERC20-approve}.
     */
    function approve(address spender, uint wad) external virtual override returns (bool) {
        return _setAllowance(msg.sender, spender, wad);
    }

    /**
     * @dev See {IERC20-transfer}.
     *
     * Requirements:
     *
     * - the caller must have a balance of at least `wad`.
     */
    function transfer(address dst, uint wad) external virtual override returns (bool) {
        return _transfer(msg.sender, dst, wad);
    }

    /**
     * @dev See {IERC20-transferFrom}.
     *
     * Emits an {Approval} event indicating the updated allowance. This is not
     * required by the EIP. See the note at the beginning of {ERC20}.
     *
     * Requirements:
     *
     * - `src` must have a balance of at least `wad`.
     * - the caller is not `src`, it must have allowance for ``src``'s tokens of at least
     * `wad`.
     */
    /// if_succeeds {:msg "TransferFrom - decrease allowance"} msg.sender != src ==> old(_allowance[src][msg.sender]) >= wad;
    function transferFrom(address src, address dst, uint wad) external virtual override returns (bool) {
        _decreaseAllowance(src, wad);

        return _transfer(src, dst, wad);
    }

    /**
     * @dev Moves tokens `wad` from `src` to `dst`.
     * 
     * Emits a {Transfer} event.
     *
     * Requirements:
     *
     * - `src` must have a balance of at least `amount`.
     */
    /// if_succeeds {:msg "Transfer - src decrease"} old(_balanceOf[src]) >= _balanceOf[src];
    /// if_succeeds {:msg "Transfer - dst increase"} _balanceOf[dst] >= old(_balanceOf[dst]);
    /// if_succeeds {:msg "Transfer - supply"} old(_balanceOf[src]) + old(_balanceOf[dst]) == _balanceOf[src] + _balanceOf[dst];
    function _transfer(address src, address dst, uint wad) internal virtual returns (bool) {
        require(_balanceOf[src] >= wad, "ERC20: Insufficient balance");
        unchecked { _balanceOf[src] = _balanceOf[src] - wad; }
        _balanceOf[dst] = _balanceOf[dst] + wad;

        emit Transfer(src, dst, wad);

        return true;
    }

    /**
     * @dev Sets the allowance granted to `spender` by `owner`.
     *
     * Emits an {Approval} event indicating the updated allowance.
     */
    function _setAllowance(address owner, address spender, uint wad) internal virtual returns (bool) {
        _allowance[owner][spender] = wad;
        emit Approval(owner, spender, wad);

        return true;
    }

    /**
     * @dev Decreases the allowance granted to the caller by `src`, unless src == msg.sender or _allowance[src][msg.sender] == MAX
     *
     * Emits an {Approval} event indicating the updated allowance, if the allowance is updated.
     *
     * Requirements:
     *
     * - `spender` must have allowance for the caller of at least
     * `wad`, unless src == msg.sender
     */
    /// if_succeeds {:msg "Decrease allowance - underflow"} old(_allowance[src][msg.sender]) <= _allowance[src][msg.sender];
    function _decreaseAllowance(address src, uint wad) internal virtual returns (bool) {
        if (src != msg.sender) {
            uint256 allowed = _allowance[src][msg.sender];
            if (allowed != type(uint).max) {
                require(allowed >= wad, "ERC20: Insufficient approval");
                unchecked { _setAllowance(src, msg.sender, allowed - wad); }
            }
        }

        return true;
    }

    /** @dev Creates `wad` tokens and assigns them to `dst`, increasing
     * the total supply.
     *
     * Emits a {Transfer} event with `from` set to the zero address.
     */
    /// if_succeeds {:msg "Mint - balance overflow"} old(_balanceOf[dst]) >= _balanceOf[dst];
    /// if_succeeds {:msg "Mint - supply overflow"} old(_totalSupply) >= _totalSupply;
    function _mint(address dst, uint wad) internal virtual returns (bool) {
        _balanceOf[dst] = _balanceOf[dst] + wad;
        _totalSupply = _totalSupply + wad;
        emit Transfer(address(0), dst, wad);

        return true;
    }

    /**
     * @dev Destroys `wad` tokens from `src`, reducing the
     * total supply.
     *
     * Emits a {Transfer} event with `to` set to the zero address.
     *
     * Requirements:
     *
     * - `src` must have at least `wad` tokens.
     */
    /// if_succeeds {:msg "Burn - balance underflow"} old(_balanceOf[src]) <= _balanceOf[src];
    /// if_succeeds {:msg "Burn - supply underflow"} old(_totalSupply) <= _totalSupply;
    function _burn(address src, uint wad) internal virtual returns (bool) {
        unchecked {
            require(_balanceOf[src] >= wad, "ERC20: Insufficient balance");
            _balanceOf[src] = _balanceOf[src] - wad;
            _totalSupply = _totalSupply - wad;
            emit Transfer(src, address(0), wad);
        }

        return true;
    }
}

File 22 of 28 : ICauldron.sol
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.0;
import "./IFYToken.sol";
import "./IOracle.sol";
import "./DataTypes.sol";

interface ICauldron {
    /// @dev Variable rate lending oracle for an underlying
    function lendingOracles(bytes6 baseId) external view returns (IOracle);

    /// @dev An user can own one or more Vaults, with each vault being able to borrow from a single series.
    function vaults(bytes12 vault)
        external
        view
        returns (DataTypes.Vault memory);

    /// @dev Series available in Cauldron.
    function series(bytes6 seriesId)
        external
        view
        returns (DataTypes.Series memory);

    /// @dev Assets available in Cauldron.
    function assets(bytes6 assetsId) external view returns (address);

    /// @dev Each vault records debt and collateral balances_.
    function balances(bytes12 vault)
        external
        view
        returns (DataTypes.Balances memory);


    // @dev Assets that are approved as collateral for a series
    function ilks(bytes6 seriesId, bytes6 assetId)
        external
        view
        returns (bool);

    /// @dev Max, min and sum of debt per underlying and collateral.
    function debt(bytes6 baseId, bytes6 ilkId)
        external
        view
        returns (DataTypes.Debt memory);

    // @dev Spot price oracle addresses and collateralization ratios
    function spotOracles(bytes6 baseId, bytes6 ilkId)
        external
        view
        returns (DataTypes.SpotOracle memory);

    /// @dev Create a new vault, linked to a series (and therefore underlying) and up to 5 collateral types
    function build(
        address owner,
        bytes12 vaultId,
        bytes6 seriesId,
        bytes6 ilkId
    ) external returns (DataTypes.Vault memory);

    /// @dev Destroy an empty vault. Used to recover gas costs.
    function destroy(bytes12 vault) external;

    /// @dev Change a vault series and/or collateral types.
    function tweak(
        bytes12 vaultId,
        bytes6 seriesId,
        bytes6 ilkId
    ) external returns (DataTypes.Vault memory);

    /// @dev Give a vault to another user.
    function give(bytes12 vaultId, address receiver)
        external
        returns (DataTypes.Vault memory);

    /// @dev Move collateral and debt between vaults.
    function stir(
        bytes12 from,
        bytes12 to,
        uint128 ink,
        uint128 art
    ) external returns (DataTypes.Balances memory, DataTypes.Balances memory);

    /// @dev Manipulate a vault debt and collateral.
    function pour(
        bytes12 vaultId,
        int128 ink,
        int128 art
    ) external returns (DataTypes.Balances memory);

    /// @dev Change series and debt of a vault.
    /// The module calling this function also needs to buy underlying in the pool for the new series, and sell it in pool for the old series.
    function roll(
        bytes12 vaultId,
        bytes6 seriesId,
        int128 art
    ) external returns (DataTypes.Vault memory, DataTypes.Balances memory);

    /// @dev Reduce debt and collateral from a vault, ignoring collateralization checks.
    function slurp(
        bytes12 vaultId,
        uint128 ink,
        uint128 art
    ) external returns (DataTypes.Balances memory);

    // ==== Helpers ====

    /// @dev Convert a debt amount for a series from base to fyToken terms.
    /// @notice Think about rounding if using, since we are dividing.
    function debtFromBase(bytes6 seriesId, uint128 base)
        external
        returns (uint128 art);

    /// @dev Convert a debt amount for a series from fyToken to base terms
    function debtToBase(bytes6 seriesId, uint128 art)
        external
        returns (uint128 base);

    // ==== Accounting ====

    /// @dev Record the borrowing rate at maturity for a series
    function mature(bytes6 seriesId) external;

    /// @dev Retrieve the rate accrual since maturity, maturing if necessary.
    function accrual(bytes6 seriesId) external returns (uint256);

    /// @dev Return the collateralization level of a vault. It will be negative if undercollateralized.
    function level(bytes12 vaultId) external returns (int256);
}

File 23 of 28 : ILadle.sol
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.0;
import "../Router.sol";
import "./IJoin.sol";
import "./ICauldron.sol";
import "./IFYToken.sol";
import "./IOracle.sol";
import "@yield-protocol/utils-v2/src/interfaces/IWETH9.sol";
import "@yield-protocol/yieldspace-tv/src/interfaces/IPool.sol";

interface ILadle {

    // ---- Storage ----

    function cauldron() 
        external view
         returns(ICauldron);
    
    function router() 
        external view
         returns(Router);
    
    function weth() 
        external view
         returns(IWETH9);
    
    function borrowingFee() 
        external view
         returns(uint256);

    function joins(bytes6) 
        external view 
        returns (IJoin);

    function pools(bytes6) 
        external view 
        returns (address);

    function modules(address) 
        external view 
        returns (bool);

    function integrations(address) 
        external view 
        returns (bool);

    function tokens(address) 
        external view 
        returns (bool);
    
    // ---- Administration ----

    /// @dev Add or remove an integration.
    function addIntegration(address integration, bool set)
        external;

    /// @dev Add or remove a token that the Ladle can call `transfer` or `permit` on.
    function addToken(address token, bool set)
        external;


    /// @dev Add a new Join for an Asset, or replace an existing one for a new one.
    /// There can be only one Join per Asset. Until a Join is added, no tokens of that Asset can be posted or withdrawn.
    function addJoin(bytes6 assetId, IJoin join)
        external;

    /// @dev Add a new Pool for a Series, or replace an existing one for a new one.
    /// There can be only one Pool per Series. Until a Pool is added, it is not possible to borrow Base.
    function addPool(bytes6 seriesId, IPool pool)
        external;

    /// @dev Add or remove a module.
    /// @notice Treat modules as you would Ladle upgrades. Modules have unrestricted access to the Ladle
    /// storage, and can wreak havoc easily.
    /// Modules must not do any changes to any vault (owner, seriesId, ilkId) because of vault caching.
    /// Modules must not be contracts that can self-destruct because of `moduleCall`.
    /// Modules can't use `msg.value` because of `batch`.
    function addModule(address module, bool set)
        external;

    /// @dev Set the fee parameter
    function setFee(uint256 fee)
        external;

    // ---- Call management ----

    /// @dev Allows batched call to self (this contract).
    /// @param calls An array of inputs for each call.
    function batch(bytes[] calldata calls)
        external payable
        returns(bytes[] memory results);

    /// @dev Allow users to route calls to a contract, to be used with batch
    function route(address integration, bytes calldata data)
        external payable
        returns (bytes memory result);

    /// @dev Allow users to use functionality coded in a module, to be used with batch
    function moduleCall(address module, bytes calldata data)
        external payable
        returns (bytes memory result);

    // ---- Token management ----

    /// @dev Execute an ERC2612 permit for the selected token
    function forwardPermit(IERC2612 token, address spender, uint256 amount, uint256 deadline, uint8 v, bytes32 r, bytes32 s)
        external payable;

    /// @dev Execute a Dai-style permit for the selected token
    function forwardDaiPermit(IERC20 token, address spender, uint256 nonce, uint256 deadline, bool allowed, uint8 v, bytes32 r, bytes32 s)
        external payable;

    /// @dev Allow users to trigger a token transfer from themselves to a receiver through the ladle, to be used with batch
    function transfer(IERC20 token, address receiver, uint128 wad)
        external payable;

    /// @dev Retrieve any token in the Ladle
    function retrieve(IERC20 token, address to) 
        external payable
        returns (uint256 amount);

    /// @dev Accept Ether, wrap it and forward it to the WethJoin
    /// This function should be called first in a batch, and the Join should keep track of stored reserves
    /// Passing the id for a join that doesn't link to a contract implemnting IWETH9 will fail
    function joinEther(bytes6 etherId)
        external payable
        returns (uint256 ethTransferred);

    /// @dev Unwrap Wrapped Ether held by this Ladle, and send the Ether
    /// This function should be called last in a batch, and the Ladle should have no reason to keep an WETH balance
    function exitEther(address to)
        external payable
        returns (uint256 ethTransferred);

    // ---- Vault management ----

    /// @dev Create a new vault, linked to a series (and therefore underlying) and a collateral
    function build(bytes6 seriesId, bytes6 ilkId, uint8 salt)
        external payable
        returns(bytes12, DataTypes.Vault memory);

    /// @dev Change a vault series or collateral.
    function tweak(bytes12 vaultId_, bytes6 seriesId, bytes6 ilkId)
        external payable
        returns(DataTypes.Vault memory vault);

    /// @dev Give a vault to another user.
    function give(bytes12 vaultId_, address receiver)
        external payable
        returns(DataTypes.Vault memory vault);

    /// @dev Destroy an empty vault. Used to recover gas costs.
    function destroy(bytes12 vaultId_)
        external payable;

    // ---- Asset and debt management ----

    /// @dev Move collateral and debt between vaults.
    function stir(bytes12 from, bytes12 to, uint128 ink, uint128 art)
        external payable;

    /// @dev Add collateral and borrow from vault, pull assets from and push borrowed asset to user
    /// Or, repay to vault and remove collateral, pull borrowed asset from and push assets to user
    /// Borrow only before maturity.
    function pour(bytes12 vaultId_, address to, int128 ink, int128 art)
        external payable;

    /// @dev Add collateral and borrow from vault, so that a precise amount of base is obtained by the user.
    /// The base is obtained by borrowing fyToken and buying base with it in a pool.
    /// Only before maturity.
    function serve(bytes12 vaultId_, address to, uint128 ink, uint128 base, uint128 max)
        external payable
        returns (uint128 art);

    /// @dev Repay vault debt using underlying token at a 1:1 exchange rate, without trading in a pool.
    /// It can add or remove collateral at the same time.
    /// The debt to repay is denominated in fyToken, even if the tokens pulled from the user are underlying.
    /// The debt to repay must be entered as a negative number, as with `pour`.
    /// Debt cannot be acquired with this function.
    function close(bytes12 vaultId_, address to, int128 ink, int128 art)
        external payable
        returns (uint128 base);

    /// @dev Repay debt by selling base in a pool and using the resulting fyToken
    /// The base tokens need to be already in the pool, unaccounted for.
    /// Only before maturity. After maturity use close.
    function repay(bytes12 vaultId_, address to, int128 ink, uint128 min)
        external payable
        returns (uint128 art);

    /// @dev Repay all debt in a vault by buying fyToken from a pool with base.
    /// The base tokens need to be already in the pool, unaccounted for. The surplus base will be returned to msg.sender.
    /// Only before maturity. After maturity use close.
    function repayVault(bytes12 vaultId_, address to, int128 ink, uint128 max)
        external payable
        returns (uint128 base);

    /// @dev Change series and debt of a vault.
    function roll(bytes12 vaultId_, bytes6 newSeriesId, uint8 loan, uint128 max)
        external payable
        returns (DataTypes.Vault memory vault, uint128 newDebt);

    // ---- Ladle as a token holder ----

    /// @dev Use fyToken in the Ladle to repay debt. Return unused fyToken to `to`.
    /// Return as much collateral as debt was repaid, as well. This function is only used when
    /// removing liquidity added with "Borrow and Pool", so it's safe to assume the exchange rate
    /// is 1:1. If used in other contexts, it might revert, which is fine.
    function repayFromLadle(bytes12 vaultId_, address to)
        external payable
        returns (uint256 repaid);

    /// @dev Use base in the Ladle to repay debt. Return unused base to `to`.
    /// Return as much collateral as debt was repaid, as well. This function is only used when
    /// removing liquidity added with "Borrow and Pool", so it's safe to assume the exchange rate
    /// is 1:1. If used in other contexts, it might revert, which is fine.
    function closeFromLadle(bytes12 vaultId_, address to)
        external payable
        returns (uint256 repaid);

    /// @dev Allow users to redeem fyToken, to be used with batch.
    /// If 0 is passed as the amount to redeem, it redeems the fyToken balance of the Ladle instead.
    function redeem(bytes6 seriesId, address to, uint256 wad)
        external payable
        returns (uint256);
}

File 24 of 28 : DataTypes.sol
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.0;
import "./IFYToken.sol";
import "./IOracle.sol";

library DataTypes {
    // ======== Cauldron data types ========
    struct Series {
        IFYToken fyToken; // Redeemable token for the series.
        bytes6 baseId; // Asset received on redemption.
        uint32 maturity; // Unix time at which redemption becomes possible.
        // bytes2 free
    }

    struct Debt {
        uint96 max; // Maximum debt accepted for a given underlying, across all series
        uint24 min; // Minimum debt accepted for a given underlying, across all series
        uint8 dec; // Multiplying factor (10**dec) for max and min
        uint128 sum; // Current debt for a given underlying, across all series
    }

    struct SpotOracle {
        IOracle oracle; // Address for the spot price oracle
        uint32 ratio; // Collateralization ratio to multiply the price for
        // bytes8 free
    }

    struct Vault {
        address owner;
        bytes6 seriesId; // Each vault is related to only one series, which also determines the underlying.
        bytes6 ilkId; // Asset accepted as collateral
    }

    struct Balances {
        uint128 art; // Debt amount
        uint128 ink; // Collateral amount
    }

    // ======== Witch data types ========
    struct Auction {
        address owner;
        uint32 start;
        bytes6 baseId; // We cache the baseId here
        uint128 ink;
        uint128 art;
        address auctioneer;
        bytes6 ilkId; // We cache the ilkId here
        bytes6 seriesId; // We cache the seriesId here
    }

    struct Line {
        uint32 duration; // Time that auctions take to go to minimal price and stay there
        uint64 vaultProportion; // Proportion of the vault that is available each auction (1e18 = 100%)
        uint64 collateralProportion; // Proportion of collateral that is sold at auction start (1e18 = 100%)
    }

    struct Limits {
        uint128 max; // Maximum concurrent auctioned collateral
        uint128 sum; // Current concurrent auctioned collateral
    }
}

library VRDataTypes {
    struct Vault {
        address owner;
        bytes6 baseId; // Each variable rate vault is related to only one base.
        bytes6 ilkId; // Asset accepted as collateral
    }
}

File 25 of 28 : IWETH9.sol
// SPDX-License-Identifier: MIT
import "../token/IERC20.sol";

pragma solidity ^0.8.0;


interface IWETH9 is IERC20 {
    event  Deposit(address indexed dst, uint wad);
    event  Withdrawal(address indexed src, uint wad);

    function deposit() external payable;
    function withdraw(uint wad) external;
}

File 26 of 28 : Router.sol
// SPDX-License-Identifier: BUSL-1.1
pragma solidity >=0.8.13;
import "@yield-protocol/utils-v2/src/utils/RevertMsgExtractor.sol";
import "@yield-protocol/utils-v2/src/utils/IsContract.sol";


/// @dev Router forwards calls between two contracts, so that any permissions
/// given to the original caller are stripped from the call.
/// This is useful when implementing generic call routing functions on contracts
/// that might have ERC20 approvals or AccessControl authorizations.
contract Router {
    using IsContract for address;

    address immutable public owner;

    constructor () {
        owner = msg.sender;
    }

    /// @dev Allow users to route calls, to be used with batch
    function route(address target, bytes calldata data)
        external payable
        returns (bytes memory result)
    {
        require(msg.sender == owner, "Only owner");
        require(target.isContract(), "Target is not a contract");
        bool success;
        (success, result) = target.call(data);
        if (!success) revert(RevertMsgExtractor.getRevertMsg(result));
    }
}

File 27 of 28 : IMaturingToken.sol
// SPDX-License-Identifier: MIT
pragma solidity >=0.8.15;
import "@yield-protocol/utils-v2/src/token/IERC20.sol";

interface IMaturingToken is IERC20 {
    function maturity() external view returns (uint256);
}

File 28 of 28 : IsContract.sol
// SPDX-License-Identifier: MIT
// Taken from Address.sol from OpenZeppelin.
pragma solidity ^0.8.0;


library IsContract {
  /// @dev Returns true if `account` is a contract.
  function isContract(address account) internal view returns (bool) {
      // This method relies on extcodesize, 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;
  }
}

Settings
{
  "optimizer": {
    "enabled": true,
    "runs": 100
  },
  "outputSelection": {
    "*": {
      "*": [
        "evm.bytecode",
        "evm.deployedBytecode",
        "devdoc",
        "userdoc",
        "metadata",
        "abi"
      ]
    }
  },
  "libraries": {
    "@yield-protocol/utils-v2/src/token/SafeERC20Namer.sol": {
      "SafeERC20Namer": "0x39bb9cBe0221D769E30bD08d185842065BcE1706"
    }
  }
}

Contract Security Audit

Contract ABI

[{"inputs":[{"internalType":"string","name":"name_","type":"string"},{"internalType":"string","name":"symbol_","type":"string"},{"internalType":"contract IFYToken","name":"fyToken_","type":"address"}],"stateMutability":"nonpayable","type":"constructor"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"owner","type":"address"},{"indexed":true,"internalType":"address","name":"spender","type":"address"},{"indexed":false,"internalType":"uint256","name":"value","type":"uint256"}],"name":"Approval","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"user","type":"address"},{"indexed":false,"internalType":"address","name":"receiver","type":"address"},{"indexed":false,"internalType":"uint256","name":"claimed","type":"uint256"}],"name":"Claimed","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"pool","type":"address"},{"indexed":false,"internalType":"uint256","name":"lpTokenDivested","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"baseObtained","type":"uint256"}],"name":"Divested","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"pool","type":"address"},{"indexed":false,"internalType":"uint256","name":"lpTokenDivested","type":"uint256"}],"name":"Drained","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"pool","type":"address"},{"indexed":false,"internalType":"uint256","name":"lpTokenDivested","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"baseObtained","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"fyTokenObtained","type":"uint256"}],"name":"Ejected","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"pool","type":"address"},{"indexed":false,"internalType":"uint256","name":"baseInvested","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"lpTokensObtained","type":"uint256"}],"name":"Invested","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"accumulated","type":"uint256"}],"name":"RewardsPerTokenUpdated","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint32","name":"start","type":"uint32"},{"indexed":false,"internalType":"uint32","name":"end","type":"uint32"},{"indexed":false,"internalType":"uint256","name":"rate","type":"uint256"}],"name":"RewardsSet","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"contract IERC20","name":"token","type":"address"}],"name":"RewardsTokenSet","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"bytes4","name":"role","type":"bytes4"},{"indexed":true,"internalType":"bytes4","name":"newAdminRole","type":"bytes4"}],"name":"RoleAdminChanged","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"bytes4","name":"role","type":"bytes4"},{"indexed":true,"internalType":"address","name":"account","type":"address"},{"indexed":true,"internalType":"address","name":"sender","type":"address"}],"name":"RoleGranted","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"bytes4","name":"role","type":"bytes4"},{"indexed":true,"internalType":"address","name":"account","type":"address"},{"indexed":true,"internalType":"address","name":"sender","type":"address"}],"name":"RoleRevoked","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"soldFYToken","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"returnedBase","type":"uint256"}],"name":"SoldFYToken","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"from","type":"address"},{"indexed":true,"internalType":"address","name":"to","type":"address"},{"indexed":false,"internalType":"uint256","name":"value","type":"uint256"}],"name":"Transfer","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"user","type":"address"},{"indexed":false,"internalType":"uint256","name":"userRewards","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"paidRewardPerToken","type":"uint256"}],"name":"UserRewardsUpdated","type":"event"},{"inputs":[],"name":"DOMAIN_SEPARATOR","outputs":[{"internalType":"bytes32","name":"","type":"bytes32"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"LOCK","outputs":[{"internalType":"bytes4","name":"","type":"bytes4"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"LOCK8605463013","outputs":[{"internalType":"bytes4","name":"","type":"bytes4"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"PERMIT_TYPEHASH","outputs":[{"internalType":"bytes32","name":"","type":"bytes32"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"ROOT","outputs":[{"internalType":"bytes4","name":"","type":"bytes4"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"ROOT4146650865","outputs":[{"internalType":"bytes4","name":"","type":"bytes4"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"owner","type":"address"},{"internalType":"address","name":"spender","type":"address"}],"name":"allowance","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"spender","type":"address"},{"internalType":"uint256","name":"wad","type":"uint256"}],"name":"approve","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"guy","type":"address"}],"name":"balanceOf","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"base","outputs":[{"internalType":"contract IERC20","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"baseCached","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"to","type":"address"}],"name":"burn","outputs":[{"internalType":"uint256","name":"poolTokensObtained","type":"uint256"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"},{"internalType":"address","name":"","type":"address"},{"internalType":"uint256","name":"","type":"uint256"},{"internalType":"uint256","name":"","type":"uint256"}],"name":"burn","outputs":[{"internalType":"uint256","name":"","type":"uint256"},{"internalType":"uint256","name":"","type":"uint256"},{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"to","type":"address"}],"name":"burnDivested","outputs":[{"internalType":"uint256","name":"baseObtained","type":"uint256"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"contract IPool","name":"pool_","type":"address"},{"internalType":"uint256","name":"poolTokens","type":"uint256"}],"name":"burnPoolTokens","outputs":[{"internalType":"uint256","name":"baseReceived","type":"uint256"},{"internalType":"uint256","name":"fyTokenReceived","type":"uint256"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"fyTokenTo","type":"address"},{"internalType":"address","name":"baseTo","type":"address"}],"name":"buyFYToken","outputs":[{"internalType":"uint256","name":"soldFYToken","type":"uint256"},{"internalType":"uint256","name":"returnedBase","type":"uint256"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"target","type":"address"},{"internalType":"bytes","name":"data","type":"bytes"}],"name":"call","outputs":[{"internalType":"bytes","name":"","type":"bytes"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"to","type":"address"}],"name":"claim","outputs":[{"internalType":"uint256","name":"claiming","type":"uint256"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"decimals","outputs":[{"internalType":"uint8","name":"","type":"uint8"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"deploymentChainId","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"divest","outputs":[{"internalType":"uint256","name":"baseObtained","type":"uint256"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"eject","outputs":[{"internalType":"uint256","name":"baseReceived","type":"uint256"},{"internalType":"uint256","name":"fyTokenReceived","type":"uint256"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"fyToken","outputs":[{"internalType":"contract IFYToken","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"fyTokenCached","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"bytes4","name":"role","type":"bytes4"}],"name":"getRoleAdmin","outputs":[{"internalType":"bytes4","name":"","type":"bytes4"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"bytes4","name":"role","type":"bytes4"},{"internalType":"address","name":"account","type":"address"}],"name":"grantRole","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"bytes4[]","name":"roles","type":"bytes4[]"},{"internalType":"address","name":"account","type":"address"}],"name":"grantRoles","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"bytes4","name":"role","type":"bytes4"},{"internalType":"address","name":"account","type":"address"}],"name":"hasRole","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"to","type":"address"}],"name":"init","outputs":[{"internalType":"uint256","name":"baseIn","type":"uint256"},{"internalType":"uint256","name":"fyTokenIn","type":"uint256"},{"internalType":"uint256","name":"minted","type":"uint256"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"contract IPool","name":"pool_","type":"address"}],"name":"invest","outputs":[{"internalType":"uint256","name":"poolTokensObtained","type":"uint256"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"bytes4","name":"role","type":"bytes4"}],"name":"lockRole","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"maturity","outputs":[{"internalType":"uint32","name":"","type":"uint32"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"to","type":"address"}],"name":"mint","outputs":[{"internalType":"uint256","name":"minted","type":"uint256"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"to","type":"address"}],"name":"mintDivested","outputs":[{"internalType":"uint256","name":"minted","type":"uint256"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"name","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"nonces","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"owner","type":"address"},{"internalType":"address","name":"spender","type":"address"},{"internalType":"uint256","name":"amount","type":"uint256"},{"internalType":"uint256","name":"deadline","type":"uint256"},{"internalType":"uint8","name":"v","type":"uint8"},{"internalType":"bytes32","name":"r","type":"bytes32"},{"internalType":"bytes32","name":"s","type":"bytes32"}],"name":"permit","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"pool","outputs":[{"internalType":"contract IPool","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"poolCached","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"user","type":"address"}],"name":"remit","outputs":[{"internalType":"uint256","name":"claiming","type":"uint256"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"bytes4","name":"role","type":"bytes4"},{"internalType":"address","name":"account","type":"address"}],"name":"renounceRole","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"restart","outputs":[{"internalType":"uint256","name":"baseIn","type":"uint256"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"bytes4","name":"role","type":"bytes4"},{"internalType":"address","name":"account","type":"address"}],"name":"revokeRole","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"bytes4[]","name":"roles","type":"bytes4[]"},{"internalType":"address","name":"account","type":"address"}],"name":"revokeRoles","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"rewards","outputs":[{"internalType":"uint128","name":"accumulated","type":"uint128"},{"internalType":"uint128","name":"checkpoint","type":"uint128"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"rewardsPerToken","outputs":[{"internalType":"uint128","name":"accumulated","type":"uint128"},{"internalType":"uint32","name":"lastUpdated","type":"uint32"},{"internalType":"uint96","name":"rate","type":"uint96"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"rewardsPeriod","outputs":[{"internalType":"uint32","name":"start","type":"uint32"},{"internalType":"uint32","name":"end","type":"uint32"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"rewardsToken","outputs":[{"internalType":"contract IERC20","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint32","name":"start","type":"uint32"},{"internalType":"uint32","name":"end","type":"uint32"},{"internalType":"uint96","name":"rate","type":"uint96"}],"name":"setRewards","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"contract IERC20","name":"rewardsToken_","type":"address"}],"name":"setRewardsToken","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"bytes4","name":"role","type":"bytes4"},{"internalType":"bytes4","name":"adminRole","type":"bytes4"}],"name":"setRoleAdmin","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"state","outputs":[{"internalType":"enum Strategy.State","name":"","type":"uint8"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"symbol","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"totalSupply","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"dst","type":"address"},{"internalType":"uint256","name":"wad","type":"uint256"}],"name":"transfer","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"src","type":"address"},{"internalType":"address","name":"dst","type":"address"},{"internalType":"uint256","name":"wad","type":"uint256"}],"name":"transferFrom","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"version","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"pure","type":"function"}]

6101406040526003610100908152623f3f3f60e81b61012052600490620000279082620004d0565b506040805180820190915260038152623f3f3f60e81b6020820152600590620000519082620004d0565b506006805460ff191660121790557f6e71edae12b1b97f4d1f60370fef10105fa2faae0126114a169c64845d6126c96080523480156200009057600080fd5b506040516200484938038062004849833981016040819052620000b3916200066c565b806001600160a01b0316636f307dc36040518163ffffffff1660e01b8152600401602060405180830381865afa158015620000f2573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190620001189190620006ec565b6040516323b95ceb60e21b81526001600160a01b03831660048201528290859085907339bb9cbe0221d769e30bd08d185842065bce170690638ee573ac90602401602060405180830381865af415801562000177573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906200019d919062000713565b828282828282620001b06000336200023f565b620001c46001600160e01b031980620002d7565b6004620001d28482620004d0565b506005620001e18382620004d0565b506006805460ff191660ff9290921691909117905550504660c0819052620002099062000375565b60a052505050506001600160a01b0393841660e0525050600c80546001600160a01b0319169190921617905550620007b6915050565b6001600160e01b031982166000908152602081815260408083206001600160a01b038516845290915290205460ff16620002d3576001600160e01b031982166000818152602081815260408083206001600160a01b0386168085529252808320805460ff1916600117905551339391927fe6231789d19137da31d0550f4ba9ee379020a8cfb64cb79bf1790c996d2e616591a45b5050565b6001600160e01b0319811662000309836001600160e01b03191660009081526020819052604090206001015460e01b90565b6001600160e01b03191614620002d3576001600160e01b0319828116600081815260208190526040808220600101805463ffffffff191660e087901c17905551928416927fd348e2220a50b4500ec353f6e802d2f14dd1b5d6786148fd1bbcc570bf92d4739190a35050565b60007f8b73c3c69bb8fe3d512ecc4cf759cc79239f7b179b0ffacaa9a75d522b39400f6004604051620003a9919062000738565b60408051918290038220828201825260018352603160f81b602093840152815180840194909452838201527fc89efdaa54c0f20c7adf612882df0950f5a951637e0307cdcb4c672f298b8bc6606084015260808301949094523060a0808401919091528451808403909101815260c09092019093528051920191909120919050565b634e487b7160e01b600052604160045260246000fd5b600181811c908216806200045657607f821691505b6020821081036200047757634e487b7160e01b600052602260045260246000fd5b50919050565b601f821115620004cb57600081815260208120601f850160051c81016020861015620004a65750805b601f850160051c820191505b81811015620004c757828155600101620004b2565b5050505b505050565b81516001600160401b03811115620004ec57620004ec6200042b565b6200050481620004fd845462000441565b846200047d565b602080601f8311600181146200053c5760008415620005235750858301515b600019600386901b1c1916600185901b178555620004c7565b600085815260208120601f198616915b828110156200056d578886015182559484019460019091019084016200054c565b50858210156200058c5787850151600019600388901b60f8161c191681555b5050505050600190811b01905550565b600082601f830112620005ae57600080fd5b81516001600160401b0380821115620005cb57620005cb6200042b565b604051601f8301601f19908116603f01168101908282118183101715620005f657620005f66200042b565b816040528381526020925086838588010111156200061357600080fd5b600091505b8382101562000637578582018301518183018401529082019062000618565b83821115620006495760008385830101525b9695505050505050565b6001600160a01b03811681146200066957600080fd5b50565b6000806000606084860312156200068257600080fd5b83516001600160401b03808211156200069a57600080fd5b620006a8878388016200059c565b94506020860151915080821115620006bf57600080fd5b50620006ce868287016200059c565b9250506040840151620006e18162000653565b809150509250925092565b600060208284031215620006ff57600080fd5b81516200070c8162000653565b9392505050565b6000602082840312156200072657600080fd5b815160ff811681146200070c57600080fd5b6000808354620007488162000441565b600182811680156200076357600181146200077957620007aa565b60ff1984168752821515830287019450620007aa565b8760005260208060002060005b85811015620007a15781548a82015290840190820162000786565b50505082870194505b50929695505050505050565b60805160a05160c05160e051613ff06200085960003960008181610554015281816109ee01528181610a6701528181610e79015281816111cc01528181611447015281816115d0015281816117e80152818161191901528181611ef7015281816122590152612b4c015260008181610790015281816112e8015261240c01526000818161131e01526124410152600081816104f101526123830152613ff06000f3fe608060405234801561001057600080fd5b50600436106102fe5760003560e01c8063687f0e4c1161019e578063ae93c1b5116100ef578063d7020d0a1161009d578063d7020d0a146107eb578063dc3bfba9146107f9578063dd62ed3e1461080c578063de02cde714610845578063de320cc114610858578063effae3531461086b578063f8f800171461087e578063ffffffff1461070f57600080fd5b8063ae93c1b514610744578063afeba11914610757578063c19d93fb1461076a578063cd0d00961461078b578063d1af0c7d146107b2578063d505accf146107c5578063d51c2828146107d857600080fd5b806395d89b411161014c57806395d89b41146106d857806397f980c3146106e05780639d5cf374146106f3578063a354f39e146106fc578063a4f0d7d01461070f578063a9059cbb1461071e578063ad82110f1461073157600080fd5b8063687f0e4c146105e85780636a627842146105fb57806370641a361461060e57806370a08231146106745780637ecebe001461069d57806382e94ac5146106bd57806389afcb44146106c557600080fd5b806323b872dd1161025857806354fd4d501161020657806354fd4d5014610576578063559742d91461057e57806358969ab9146105915780635909c12f14610303578063592db8b1146105b95780635ba5e9f0146105c25780635c859956146105d557600080fd5b806323b872dd146104a157806326ae2b78146104b457806330adf81f146104ec578063313ce567146105135780633644e5151461053257806344faded01461053a5780635001f3b51461054f57600080fd5b806316f0115b116102b557806316f0115b146103f157806318160ddd1461041157806319ab453c146104195780631b8b921d146104475780631e83409a1461045a5780631ef3755d1461046d578063204f83f91461047557600080fd5b801561030357806303f9c79314610329578063058aace11461034a57806306fdde03146103525780630700037d14610367578063095ea7b3146103bb57806310ab9432146103de575b600080fd5b61030b600081565b6040516001600160e01b031990911681526020015b60405180910390f35b61033c610337366004613641565b610887565b604051908152602001610320565b61033c610c22565b61035a610f45565b60405161032091906136ba565b61039b610375366004613641565b600b602052600090815260409020546001600160801b0380821691600160801b90041682565b604080516001600160801b03938416815292909116602083015201610320565b6103ce6103c93660046136dd565b610fd3565b6040519015158152602001610320565b6103ce6103ec366004613721565b610fe7565b600d54610404906001600160a01b031681565b6040516103209190613758565b60015461033c565b61042c610427366004613641565b610ff3565b60408051938452602084019290925290820152606001610320565b61035a61045536600461376c565b611041565b61033c610468366004613641565b61111d565b61033c61112f565b600c5461048c90600160a01b900463ffffffff1681565b60405163ffffffff9091168152602001610320565b6103ce6104af3660046137f1565b6112c4565b6009546104cf9063ffffffff80821691600160201b90041682565b6040805163ffffffff938416815292909116602083015201610320565b61033c7f000000000000000000000000000000000000000000000000000000000000000081565b6006546105209060ff1681565b60405160ff9091168152602001610320565b61033c6112e4565b61054d610548366004613721565b611340565b005b6104047f000000000000000000000000000000000000000000000000000000000000000081565b61035a61137e565b61054d61058c366004613832565b611399565b6105a461059f3660046136dd565b6113d8565b60408051928352602083019190915201610320565b61033c60105481565b61030b6105d0366004613832565b61175f565b6105a46105e336600461384d565b61176a565b61054d6105f6366004613721565b611985565b61033c610609366004613641565b6119e0565b600a54610641906001600160801b03811690600160801b810463ffffffff1690600160a01b90046001600160601b031683565b604080516001600160801b03909416845263ffffffff90921660208401526001600160601b031690820152606001610320565b61033c610682366004613641565b6001600160a01b031660009081526002602052604090205490565b61033c6106ab366004613641565b60076020526000908152604090205481565b6105a4611af7565b61033c6106d3366004613641565b611d7f565b61035a611e46565b61033c6106ee366004613641565b611e53565b61033c600f5481565b61054d61070a36600461387d565b611f26565b61030b6001600160e01b031981565b6103ce61072c3660046136dd565b612125565b61054d61073f36600461391b565b612132565b61054d6107523660046139da565b6121b4565b61033c610765366004613641565b6121e7565b600c5461077e90600160c01b900460ff1681565b6040516103209190613a23565b61033c7f000000000000000000000000000000000000000000000000000000000000000081565b600854610404906001600160a01b031681565b61054d6107d3366004613a4b565b612317565b61033c6107e6366004613641565b612586565b61042c6102fe366004613ac2565b600c54610404906001600160a01b031681565b61033c61081a36600461384d565b6001600160a01b03918216600090815260036020908152604080832093909416825291909152205490565b61054d610853366004613721565b612592565b61054d610866366004613641565b6125c5565b61054d61087936600461391b565b6126a2565b61033c600e5481565b600061089f6000356001600160e01b03191633612717565b6108c45760405162461bcd60e51b81526004016108bb90613b08565b60405180910390fd5b600c54600190600160c01b900460ff1660048111156108e5576108e5613a0d565b8160048111156108f7576108f7613a0d565b146109145760405162461bcd60e51b81526004016108bb90613b2f565b6000836001600160a01b031663dc3bfba96040518163ffffffff1660e01b8152600401602060405180830381865afa158015610954573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906109789190613b62565b90506000600e549050846001600160a01b0316635001f3b56040518163ffffffff1660e01b8152600401602060405180830381865afa1580156109bf573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906109e39190613b62565b6001600160a01b03167f00000000000000000000000000000000000000000000000000000000000000006001600160a01b031614610a555760405162461bcd60e51b815260206004820152600f60248201526e4d69736d617463686564206261736560881b60448201526064016108bb565b6000600e55610a8e6001600160a01b037f000000000000000000000000000000000000000000000000000000000000000016868361274c565b60405163066ad14f60e21b81526001600160a01b038616906319ab453c90610aba903090600401613758565b6060604051808303816000875af1158015610ad9573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610afd9190613b7f565b600f819055600c80546001600160a01b0319166001600160a01b03878116919091179091556040805163204f83f960e01b81529051929850908916935063204f83f992506004808201926020929091908290030181865afa158015610b66573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610b8a9190613bad565b600c805463ffffffff60a01b1916600160a01b63ffffffff9390931692909202919091179055600d80546001600160a01b0319166001600160a01b038716179055610bd660028661284b565b60408051828152602081018690526001600160a01b038716917f9e9d071824fd57d062ca63fd8b786d8da48a6807eebbcb2d83f9e8d21398e28c910160405180910390a2505050919050565b600c54600090600290600160c01b900460ff166004811115610c4657610c46613a0d565b816004811115610c5857610c58613a0d565b14610c755760405162461bcd60e51b81526004016108bb90613b2f565b600d54600c546001600160a01b039182169181169063ffffffff600160a01b9091048116429091161015610ce15760405162461bcd60e51b81526020600482015260136024820152724f6e6c79206166746572206d6174757269747960681b60448201526064016108bb565b6040516370a0823160e01b81526000906001600160a01b038416906370a0823190610d10903090600401613758565b602060405180830381865afa158015610d2d573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610d519190613bca565b6000600f559050610d6c6001600160a01b038416848361274c565b600080846001600160a01b031663d7020d0a303060006000196040518563ffffffff1660e01b8152600401610da49493929190613be3565b6060604051808303816000875af1158015610dc3573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610de79190613b7f565b92509250506000846001600160a01b0316631e9a695030846040518363ffffffff1660e01b8152600401610e1c929190613c0c565b6020604051808303816000875af1158015610e3b573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610e5f9190613bca565b6040516370a0823160e01b81529091506001600160a01b037f000000000000000000000000000000000000000000000000000000000000000016906370a0823190610eae903090600401613758565b602060405180830381865afa158015610ecb573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610eef9190613bca565b600e55610efd60018761284b565b6001600160a01b038616600080516020613f7b83398151915285610f218487613c3b565b6040805192835260208301829052909b500160405180910390a25050505050505090565b60048054610f5290613c53565b80601f0160208091040260200160405190810160405280929190818152602001828054610f7e90613c53565b8015610fcb5780601f10610fa057610100808354040283529160200191610fcb565b820191906000526020600020905b815481529060010190602001808311610fae57829003601f168201915b505050505081565b6000610fe0338484612a72565b9392505050565b6000610fe08383612717565b600080600061100e6000356001600160e01b03191633612717565b61102a5760405162461bcd60e51b81526004016108bb90613b08565b6000915061103784612adb565b9491935084925050565b60606110596000356001600160e01b03191633612717565b6110755760405162461bcd60e51b81526004016108bb90613b08565b600080856001600160a01b03168585604051611092929190613c87565b6000604051808303816000865af19150503d80600081146110cf576040519150601f19603f3d011682016040523d82523d6000602084013e6110d4565b606091505b5091509150816111145760405162461bcd60e51b815260206004820152600b60248201526a10d85b1b0819985a5b195960aa1b60448201526064016108bb565b95945050505050565b60006111293383612c2e565b92915050565b60006111476000356001600160e01b03191633612717565b6111635760405162461bcd60e51b81526004016108bb90613b08565b600c54600490600160c01b900460ff168181111561118357611183613a0d565b81600481111561119557611195613a0d565b146111b25760405162461bcd60e51b81526004016108bb90613b2f565b6040516370a0823160e01b81526000906001600160a01b037f000000000000000000000000000000000000000000000000000000000000000016906370a0823190611201903090600401613758565b602060405180830381865afa15801561121e573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906112429190613bca565b925082600e8190551161128c5760405162461bcd60e51b8152602060048201526012602482015271139bc818985cd9481d1bc81c995cdd185c9d60721b60448201526064016108bb565b6112966001612cd6565b6040805160008082526020820181905291600080516020613f7b833981519152910160405180910390a25090565b60006112d08483612d3b565b506112dc848484612de4565b949350505050565b60007f0000000000000000000000000000000000000000000000000000000000000000461461131b5761131646612e0d565b905090565b507f000000000000000000000000000000000000000000000000000000000000000090565b8161135361134d82612e9e565b33612717565b61136f5760405162461bcd60e51b81526004016108bb90613c97565b6113798383612ec0565b505050565b6040805180820190915260018152603160f81b602082015290565b806113a661134d82612e9e565b6113c25760405162461bcd60e51b81526004016108bb90613c97565b6113d4826001600160e01b0319612f2f565b5050565b6000803330146114195760405162461bcd60e51b815260206004820152600c60248201526b155b985d5d1a1bdc9a5e995960a21b60448201526064016108bb565b61142d6001600160a01b038516858561274c565b6040516370a0823160e01b81526000906001600160a01b037f000000000000000000000000000000000000000000000000000000000000000016906370a082319061147c903090600401613758565b602060405180830381865afa158015611499573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906114bd9190613bca565b600c546040516370a0823160e01b81529192506000916001600160a01b03909116906370a08231906114f3903090600401613758565b602060405180830381865afa158015611510573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906115349190613bca565b604051636b81068560e11b81529091506001600160a01b0387169063d7020d0a9061156c903090819060009060001990600401613be3565b6060604051808303816000875af115801561158b573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906115af9190613b7f565b6040516370a0823160e01b8152919650945085915083906001600160a01b037f000000000000000000000000000000000000000000000000000000000000000016906370a0823190611605903090600401613758565b602060405180830381865afa158015611622573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906116469190613bca565b6116509190613cbb565b146116925760405162461bcd60e51b81526020600482015260126024820152714275726e206661696c6564202d206261736560701b60448201526064016108bb565b600c546040516370a0823160e01b8152849183916001600160a01b03909116906370a08231906116c6903090600401613758565b602060405180830381865afa1580156116e3573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906117079190613bca565b6117119190613cbb565b146117565760405162461bcd60e51b8152602060048201526015602482015274213ab937103330b4b632b2101690333caa37b5b2b760591b60448201526064016108bb565b50509250929050565b600061112982612e9e565b6000806003600c60189054906101000a900460ff16600481111561179057611790613a0d565b8160048111156117a2576117a2613a0d565b146117bf5760405162461bcd60e51b81526004016108bb90613b2f565b600c54600e546010546040516370a0823160e01b81526001600160a01b039384169360009184917f000000000000000000000000000000000000000000000000000000000000000016906370a082319061181d903090600401613758565b602060405180830381865afa15801561183a573d6000803e3d6000fd5b505050506040513d601f19601f8201168201806040525081019061185e9190613bca565b6118689190613cbb565b905081811161187957806000611884565b816118848183613cbb565b90975095506118938784613c3b565b600e556118a08783613cbb565b6010819055915060008290036118e5576118ba6001612cd6565b6040805160008082526020820181905291600080516020613f7b833981519152910160405180910390a25b6118f96001600160a01b0385168a8961274c565b80871015611940576119408861190f8984613cbb565b6001600160a01b037f000000000000000000000000000000000000000000000000000000000000000016919061274c565b60408051888152602081018890527fc507be2ec3f5cf464d2bb8665d035a25c6156a546684fb273e1fa95906cd7fd1910160405180910390a150505050509250929050565b6001600160a01b03811633146119d65760405162461bcd60e51b81526020600482015260166024820152752932b737bab731b29037b7363c903337b91039b2b63360511b60448201526064016108bb565b6113d48282612ec0565b600c54600090600290600160c01b900460ff166004811115611a0457611a04613a0d565b816004811115611a1657611a16613a0d565b14611a335760405162461bcd60e51b81526004016108bb90613b2f565b600d54600f546040516370a0823160e01b81526001600160a01b0390921691600090829084906370a0823190611a6d903090600401613758565b602060405180830381865afa158015611a8a573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190611aae9190613bca565b611ab89190613cbb565b9050611ac48183613c3b565b600f556001548290611ad7908390613cd2565b611ae19190613cf1565b9450611aed8686612fae565b5050505050919050565b600080611b106000356001600160e01b03191633612717565b611b2c5760405162461bcd60e51b81526004016108bb90613b08565b600c54600290600160c01b900460ff166004811115611b4d57611b4d613a0d565b816004811115611b5f57611b5f613a0d565b14611b7c5760405162461bcd60e51b81526004016108bb90613b2f565b600d546040516370a0823160e01b81526001600160a01b039091169060009082906370a0823190611bb1903090600401613758565b602060405180830381865afa158015611bce573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190611bf29190613bca565b6000600f556040516358969ab960e01b815290915030906358969ab990611c1f9085908590600401613c0c565b60408051808303816000875af1925050508015611c59575060408051601f3d908101601f19168201909252611c5691810190613d13565b60015b611cc457611c716001600160a01b038316338361274c565b611c7c60048361284b565b816001600160a01b03167fb2559daa129ad136aac2133ac6a0c75920abbef7d6663a017a94e181b13786c382604051611cb791815260200190565b60405180910390a2611d78565b600e8290556010819055909550935084848015611d3757611ce660038561284b565b60408051848152602081018990529081018790526001600160a01b038516907fa4934663617b597bfe6baf57cd7218dfe65e68ad58d417b92ed0bce047cf562a9060600160405180910390a2611d75565b611d4260018561284b565b60408051848152602081018990526001600160a01b03861691600080516020613f7b833981519152910160405180910390a25b50505b5050509091565b600c54600090600290600160c01b900460ff166004811115611da357611da3613a0d565b816004811115611db557611db5613a0d565b14611dd25760405162461bcd60e51b81526004016108bb90613b2f565b600d54600f54600154306000818152600260205260409020546001600160a01b039094169390611e029082612fcc565b5081611e0e8285613cd2565b611e189190613cf1565b9550611e2e6001600160a01b038516888861274c565b611e388684613cbb565b600f55509395945050505050565b60058054610f5290613c53565b600c54600090600190600160c01b900460ff166004811115611e7757611e77613a0d565b816004811115611e8957611e89613a0d565b14611ea65760405162461bcd60e51b81526004016108bb90613b2f565b600e5430600090815260026020526040902054600154611ec68284613cd2565b611ed09190613cf1565b9350611edc8483613cbb565b600e55611ee93082612fcc565b50611f1e6001600160a01b037f000000000000000000000000000000000000000000000000000000000000000016868661274c565b505050919050565b611f3c6000356001600160e01b03191633612717565b611f585760405162461bcd60e51b81526004016108bb90613b08565b8163ffffffff168363ffffffff161115611fa65760405162461bcd60e51b815260206004820152600f60248201526e125b98dbdc9c9958dd081a5b9c1d5d608a1b60448201526064016108bb565b6008546001600160a01b0316611ff65760405162461bcd60e51b815260206004820152601560248201527414995dd85c991cc81d1bdad95b881b9bdd081cd95d605a1b60448201526064016108bb565b60095463ffffffff1661200842612fea565b63ffffffff1610806120365750600954600160201b900463ffffffff1661202e42612fea565b63ffffffff16115b6120745760405162461bcd60e51b815260206004820152600f60248201526e4f6e676f696e67207265776172647360881b60448201526064016108bb565b61207c613014565b6009805463ffffffff85811667ffffffffffffffff199092168217600160201b91861691820217909255600a80546001600160801b0316600160801b83026001600160a01b031617600160a01b6001600160601b03861690810291909117909155604080519283526020830193909352918101919091527f95efd8a2a0aa591f48fd9673cf5d13c2150ca7a1fe1cbe438dd3f0ae470646639060600160405180910390a1505050565b6000610fe0338484612de4565b60005b82518110156113795761216361134d84838151811061215657612156613d37565b6020026020010151612e9e565b61217f5760405162461bcd60e51b81526004016108bb90613c97565b6121a283828151811061219457612194613d37565b602002602001015183612ec0565b806121ac81613d4d565b915050612135565b816121c161134d82612e9e565b6121dd5760405162461bcd60e51b81526004016108bb90613c97565b6113798383612f2f565b600c54600090600190600160c01b900460ff16600481111561220b5761220b613a0d565b81600481111561221d5761221d613a0d565b1461223a5760405162461bcd60e51b81526004016108bb90613b2f565b600e546040516370a0823160e01b815260009082906001600160a01b037f000000000000000000000000000000000000000000000000000000000000000016906370a082319061228e903090600401613758565b602060405180830381865afa1580156122ab573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906122cf9190613bca565b6122d99190613cbb565b90506122e58183613c3b565b600e5560015482906122f8908390613cd2565b6123029190613cf1565b935061230e8585612fae565b50505050919050565b428410156123675760405162461bcd60e51b815260206004820152601d60248201527f45524332305065726d69743a206578706972656420646561646c696e6500000060448201526064016108bb565b6001600160a01b038716600090815260076020526040812080547f0000000000000000000000000000000000000000000000000000000000000000918a918a918a9190866123b483613d4d565b909155506040805160208101969096526001600160a01b0394851690860152929091166060840152608083015260a082015260c0810186905260e00160405160208183030381529060405280519060200120905060007f0000000000000000000000000000000000000000000000000000000000000000461461243f5761243a46612e0d565b612461565b7f00000000000000000000000000000000000000000000000000000000000000005b60405161190160f01b602082015260228101919091526042810183905260620160408051601f198184030181528282528051602091820120600080855291840180845281905260ff89169284019290925260608301879052608083018690529092509060019060a0016020604051602081039080840390855afa1580156124ec573d6000803e3d6000fd5b5050604051601f1901519150506001600160a01b038116158015906125225750896001600160a01b0316816001600160a01b0316145b61256e5760405162461bcd60e51b815260206004820152601e60248201527f45524332305065726d69743a20696e76616c6964207369676e6174757265000060448201526064016108bb565b6125798a8a8a612a72565b5050505050505050505050565b60006111298283612c2e565b8161259f61134d82612e9e565b6125bb5760405162461bcd60e51b81526004016108bb90613c97565b61137983836131ed565b6125db6000356001600160e01b03191633612717565b6125f75760405162461bcd60e51b81526004016108bb90613b08565b6008546001600160a01b03161561264c5760405162461bcd60e51b815260206004820152601960248201527814995dd85c991cc81d1bdad95b88185b1c9958591e481cd95d603a1b60448201526064016108bb565b600880546001600160a01b0319166001600160a01b0383161790556040517f45a4cbe003c343d60028e5acd63eecf588647b0eb369733afa2c0482180d4f7490612697908390613758565b60405180910390a150565b60005b8251811015611379576126c661134d84838151811061215657612156613d37565b6126e25760405162461bcd60e51b81526004016108bb90613c97565b6127058382815181106126f7576126f7613d37565b6020026020010151836131ed565b8061270f81613d4d565b9150506126a5565b6001600160e01b031982166000908152602081815260408083206001600160a01b038516845290915290205460ff1692915050565b600080846001600160a01b031663a9059cbb60e01b8585604051602401612774929190613c0c565b60408051601f198184030181529181526020820180516001600160e01b03166001600160e01b03199094169390931790925290516127b29190613d66565b6000604051808303816000865af19150503d80600081146127ef576040519150601f19603f3d011682016040523d82523d6000602084013e6127f4565b606091505b509150915081801561281e57508051158061281e57508080602001905181019061281e9190613d82565b6128445761282b8161325e565b60405162461bcd60e51b81526004016108bb91906136ba565b5050505050565b600282600481111561285f5761285f613a0d565b0361298e57600d80546001600160a01b0319166001600160a01b0383169081179091556040805163dc3bfba960e01b8152905163dc3bfba9916004808201926020929091908290030181865afa1580156128bd573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906128e19190613b62565b600c60006101000a8154816001600160a01b0302191690836001600160a01b03160217905550806001600160a01b031663204f83f96040518163ffffffff1660e01b8152600401602060405180830381865afa158015612945573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906129699190613bad565b600c60146101000a81548163ffffffff021916908363ffffffff160217905550612a44565b60018260048111156129a2576129a2613a0d565b036129cc57600c80546001600160c01b0319169055600d80546001600160a01b0319169055612a44565b60038260048111156129e0576129e0613a0d565b03612a0a57600c805463ffffffff60a01b19169055600d80546001600160a01b0319169055612a44565b6004826004811115612a1e57612a1e613a0d565b03612a4457600c805463ffffffff60a01b19169055600d80546001600160a01b03191690555b600c805483919060ff60c01b1916600160c01b836004811115612a6957612a69613a0d565b02179055505050565b6001600160a01b03838116600081815260036020908152604080832094871680845294825280832086905551858152919392917f8c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b92591015b60405180910390a35060019392505050565b600c546000908190600160c01b900460ff166004811115612afe57612afe613a0d565b816004811115612b1057612b10613a0d565b14612b2d5760405162461bcd60e51b81526004016108bb90613b2f565b600c80546001600160a01b03191690556040516370a0823160e01b81527f00000000000000000000000000000000000000000000000000000000000000006001600160a01b0316906370a0823190612b89903090600401613758565b602060405180830381865afa158015612ba6573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190612bca9190613bca565b600e819055915081612c135760405162461bcd60e51b81526020600482015260126024820152712737ba1032b737bab3b4103130b9b29034b760711b60448201526064016108bb565b612c1d8383612fae565b50612c286001612cd6565b50919050565b6000612c38613014565b612c41836132bd565b6001600160a01b038085166000908152600b6020526040902080546001600160801b03191690556008546001600160801b03929092169250612c859116838361274c565b604080516001600160a01b038086168252841660208201529081018290527ff7a40077ff7a04c7e61f6f26fb13774259ddf1b6bce9ecf26a8276cdd39926839060600160405180910390a192915050565b6002816004811115612cea57612cea613a0d565b03612d2d5760405162461bcd60e51b8152602060048201526013602482015272135d5cdd081c1c9bdd9a59194818481c1bdbdb606a1b60448201526064016108bb565b612d3881600061284b565b50565b60006001600160a01b0383163314612ddb576001600160a01b03831660009081526003602090815260408083203384529091529020546000198114612dd95782811015612dca5760405162461bcd60e51b815260206004820152601c60248201527f45524332303a20496e73756666696369656e7420617070726f76616c0000000060448201526064016108bb565b612dd78433858403612a72565b505b505b50600192915050565b6000612dee613014565b612df7846132bd565b50612e01836132bd565b506112dc84848461342d565b60007f8b73c3c69bb8fe3d512ecc4cf759cc79239f7b179b0ffacaa9a75d522b39400f6004604051612e3f9190613da4565b6040518091039020612e4f61137e565b80516020918201206040805192830194909452928101919091526060810191909152608081018390523060a082015260c001604051602081830303815290604052805190602001209050919050565b6001600160e01b03191660009081526020819052604090206001015460e01b90565b612eca8282612717565b156113d4576001600160e01b031982166000818152602081815260408083206001600160a01b0386168085529252808320805460ff1916905551339391927f4ddc7b757e7bdd7254a9cd39452d307a52761bc824625c6a33104a075d8099e691a45050565b6001600160e01b03198116612f4383612e9e565b6001600160e01b031916146113d4576001600160e01b0319828116600081815260208190526040808220600101805463ffffffff191660e087901c17905551928416927fd348e2220a50b4500ec353f6e802d2f14dd1b5d6786148fd1bbcc570bf92d4739190a35050565b6000612fb8613014565b612fc1836132bd565b50610fe083836134da565b6000612fd6613014565b612fdf836132bd565b50610fe08383613561565b600063ffffffff8211156130105760405162461bcd60e51b81526004016108bb90613e43565b5090565b60408051606081018252600a546001600160801b0381168252600160801b810463ffffffff908116602080850191909152600160a01b9092046001600160601b0316838501528351808501909452600954808216808652600160201b909104909116918401919091526001549192919061308d42612fea565b63ffffffff16101561309e57505050565b60006130b66130ac42612fea565b84602001516135e1565b905060008460200151826130ca9190613e6a565b63ffffffff169050806000036130e1575050505050565b82156131475761313b8386604001516001600160601b031683670de0b6b3a764000061310d9190613cd2565b6131179190613cd2565b6131219190613cf1565b865161313691906001600160801b0316613c3b565b613603565b6001600160801b031685525b63ffffffff8216602086018190528551600a80546040808a01516001600160601b0316600160a01b026001600160a01b03600160801b9096026001600160a01b03199093166001600160801b03861617929092179490941617905590517fe972555b20cae8150e291bb40efce3ef4e3ed6b6b2c39c029346600e95469d48916131de916001600160801b0391909116815260200190565b60405180910390a15050505050565b6131f78282612717565b6113d4576001600160e01b031982166000818152602081815260408083206001600160a01b0386168085529252808320805460ff1916600117905551339391927fe6231789d19137da31d0550f4ba9ee379020a8cfb64cb79bf1790c996d2e616591a45050565b60606044825110156132a357505060408051808201909152601d81527f5472616e73616374696f6e2072657665727465642073696c656e746c79000000602082015290565b600482019150818060200190518101906111299190613e8f565b6001600160a01b0381166000908152600b602090815260408083208151808301835290546001600160801b038082168352600160801b9182900481168386019081528451606081018652600a5492831680825293830463ffffffff1696810196909652600160a01b9091046001600160601b03169385019390935291519092916133a091670de0b6b3a7640000916133559190613f23565b6001600160a01b038716600090815260026020526040902054613381916001600160801b031690613cd2565b61338b9190613cf1565b835161313691906001600160801b0316613c3b565b6001600160801b0390811683528151811660208085019182526001600160a01b0387166000818152600b835260409081902087519451948616600160801b95909616948502861790558051918252918101939093528201527f5b9aaf4cc5141c090a75f8b8a627863eba92df81f0c83c096350da9b79aedd049060600160405180910390a1505192915050565b6001600160a01b0383166000908152600260205260408120548211156134655760405162461bcd60e51b81526004016108bb90613f43565b6001600160a01b038085166000908152600260205260408082208054869003905591851681522054613498908390613c3b565b6001600160a01b038085166000818152600260205260409081902093909355915190861690600080516020613f9b83398151915290612ac99086815260200190565b6001600160a01b0382166000908152600260205260408120546134fe908390613c3b565b6001600160a01b038416600090815260026020526040902055600154613525908390613c3b565b6001556040518281526001600160a01b03841690600090600080516020613f9b833981519152906020015b60405180910390a350600192915050565b6001600160a01b0382166000908152600260205260408120548211156135995760405162461bcd60e51b81526004016108bb90613f43565b6001600160a01b03831660008181526002602090815260408083208054879003905560018054879003905551858152919291600080516020613f9b8339815191529101613550565b60008163ffffffff168363ffffffff16106135fc5781610fe0565b5090919050565b60006001600160801b038211156130105760405162461bcd60e51b81526004016108bb90613e43565b6001600160a01b0381168114612d3857600080fd5b60006020828403121561365357600080fd5b8135610fe08161362c565b60005b83811015613679578181015183820152602001613661565b83811115613688576000848401525b50505050565b600081518084526136a681602086016020860161365e565b601f01601f19169290920160200192915050565b602081526000610fe0602083018461368e565b80356136d88161362c565b919050565b600080604083850312156136f057600080fd5b82356136fb8161362c565b946020939093013593505050565b80356001600160e01b0319811681146136d857600080fd5b6000806040838503121561373457600080fd5b61373d83613709565b9150602083013561374d8161362c565b809150509250929050565b6001600160a01b0391909116815260200190565b60008060006040848603121561378157600080fd5b833561378c8161362c565b9250602084013567ffffffffffffffff808211156137a957600080fd5b818601915086601f8301126137bd57600080fd5b8135818111156137cc57600080fd5b8760208285010111156137de57600080fd5b6020830194508093505050509250925092565b60008060006060848603121561380657600080fd5b83356138118161362c565b925060208401356138218161362c565b929592945050506040919091013590565b60006020828403121561384457600080fd5b610fe082613709565b6000806040838503121561386057600080fd5b823561373d8161362c565b63ffffffff81168114612d3857600080fd5b60008060006060848603121561389257600080fd5b833561389d8161386b565b925060208401356138ad8161386b565b915060408401356001600160601b03811681146138c957600080fd5b809150509250925092565b634e487b7160e01b600052604160045260246000fd5b604051601f8201601f1916810167ffffffffffffffff81118282101715613913576139136138d4565b604052919050565b6000806040838503121561392e57600080fd5b823567ffffffffffffffff8082111561394657600080fd5b818501915085601f83011261395a57600080fd5b813560208282111561396e5761396e6138d4565b8160051b925061397f8184016138ea565b828152928401810192818101908985111561399957600080fd5b948201945b848610156139be576139af86613709565b8252948201949082019061399e565b96506139cd90508782016136cd565b9450505050509250929050565b600080604083850312156139ed57600080fd5b6139f683613709565b9150613a0460208401613709565b90509250929050565b634e487b7160e01b600052602160045260246000fd5b6020810160058310613a4557634e487b7160e01b600052602160045260246000fd5b91905290565b600080600080600080600060e0888a031215613a6657600080fd5b8735613a718161362c565b96506020880135613a818161362c565b95506040880135945060608801359350608088013560ff81168114613aa557600080fd5b9699959850939692959460a0840135945060c09093013592915050565b60008060008060808587031215613ad857600080fd5b8435613ae38161362c565b93506020850135613af38161362c565b93969395505050506040820135916060013590565b6020808252600d908201526c1058d8d95cdcc819195b9a5959609a1b604082015260600190565b6020808252601990820152784e6f7420616c6c6f77656420696e207468697320737461746560381b604082015260600190565b600060208284031215613b7457600080fd5b8151610fe08161362c565b600080600060608486031215613b9457600080fd5b8351925060208401519150604084015190509250925092565b600060208284031215613bbf57600080fd5b8151610fe08161386b565b600060208284031215613bdc57600080fd5b5051919050565b6001600160a01b0394851681529290931660208301526040820152606081019190915260800190565b6001600160a01b03929092168252602082015260400190565b634e487b7160e01b600052601160045260246000fd5b60008219821115613c4e57613c4e613c25565b500190565b600181811c90821680613c6757607f821691505b602082108103612c2857634e487b7160e01b600052602260045260246000fd5b8183823760009101908152919050565b6020808252600a908201526927b7363c9030b236b4b760b11b604082015260600190565b600082821015613ccd57613ccd613c25565b500390565b6000816000190483118215151615613cec57613cec613c25565b500290565b600082613d0e57634e487b7160e01b600052601260045260246000fd5b500490565b60008060408385031215613d2657600080fd5b505080516020909101519092909150565b634e487b7160e01b600052603260045260246000fd5b600060018201613d5f57613d5f613c25565b5060010190565b60008251613d7881846020870161365e565b9190910192915050565b600060208284031215613d9457600080fd5b81518015158114610fe057600080fd5b600080835481600182811c915080831680613dc057607f831692505b60208084108203613ddf57634e487b7160e01b86526022600452602486fd5b818015613df35760018114613e0857613e35565b60ff1986168952841515850289019650613e35565b60008a81526020902060005b86811015613e2d5781548b820152908501908301613e14565b505084890196505b509498975050505050505050565b6020808252600d908201526c43617374206f766572666c6f7760981b604082015260600190565b600063ffffffff83811690831681811015613e8757613e87613c25565b039392505050565b600060208284031215613ea157600080fd5b815167ffffffffffffffff80821115613eb957600080fd5b818401915084601f830112613ecd57600080fd5b815181811115613edf57613edf6138d4565b613ef2601f8201601f19166020016138ea565b9150808252856020828501011115613f0957600080fd5b613f1a81602084016020860161365e565b50949350505050565b60006001600160801b0383811690831681811015613e8757613e87613c25565b6020808252601b908201527f45524332303a20496e73756666696369656e742062616c616e6365000000000060408201526060019056fef44b6ecb6421462dee6400bd4e3bb57864c0f428d0f7e7d49771f9fd7c30d4faddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3efa2646970667358221220b13b6f1a20a6b787e1b7c87a84f7f85eb6c255d3c2528550033e66796d4d042d64736f6c634300080f0033000000000000000000000000000000000000000000000000000000000000006000000000000000000000000000000000000000000000000000000000000000a0000000000000000000000000d28380de0e7093ac62bcb88610b9f4f4fb58be74000000000000000000000000000000000000000000000000000000000000001e5969656c64205374726174656779205553445420364d204a756e204465630000000000000000000000000000000000000000000000000000000000000000000a595355534454364d4a4400000000000000000000000000000000000000000000

Deployed Bytecode

0x608060405234801561001057600080fd5b50600436106102fe5760003560e01c8063687f0e4c1161019e578063ae93c1b5116100ef578063d7020d0a1161009d578063d7020d0a146107eb578063dc3bfba9146107f9578063dd62ed3e1461080c578063de02cde714610845578063de320cc114610858578063effae3531461086b578063f8f800171461087e578063ffffffff1461070f57600080fd5b8063ae93c1b514610744578063afeba11914610757578063c19d93fb1461076a578063cd0d00961461078b578063d1af0c7d146107b2578063d505accf146107c5578063d51c2828146107d857600080fd5b806395d89b411161014c57806395d89b41146106d857806397f980c3146106e05780639d5cf374146106f3578063a354f39e146106fc578063a4f0d7d01461070f578063a9059cbb1461071e578063ad82110f1461073157600080fd5b8063687f0e4c146105e85780636a627842146105fb57806370641a361461060e57806370a08231146106745780637ecebe001461069d57806382e94ac5146106bd57806389afcb44146106c557600080fd5b806323b872dd1161025857806354fd4d501161020657806354fd4d5014610576578063559742d91461057e57806358969ab9146105915780635909c12f14610303578063592db8b1146105b95780635ba5e9f0146105c25780635c859956146105d557600080fd5b806323b872dd146104a157806326ae2b78146104b457806330adf81f146104ec578063313ce567146105135780633644e5151461053257806344faded01461053a5780635001f3b51461054f57600080fd5b806316f0115b116102b557806316f0115b146103f157806318160ddd1461041157806319ab453c146104195780631b8b921d146104475780631e83409a1461045a5780631ef3755d1461046d578063204f83f91461047557600080fd5b801561030357806303f9c79314610329578063058aace11461034a57806306fdde03146103525780630700037d14610367578063095ea7b3146103bb57806310ab9432146103de575b600080fd5b61030b600081565b6040516001600160e01b031990911681526020015b60405180910390f35b61033c610337366004613641565b610887565b604051908152602001610320565b61033c610c22565b61035a610f45565b60405161032091906136ba565b61039b610375366004613641565b600b602052600090815260409020546001600160801b0380821691600160801b90041682565b604080516001600160801b03938416815292909116602083015201610320565b6103ce6103c93660046136dd565b610fd3565b6040519015158152602001610320565b6103ce6103ec366004613721565b610fe7565b600d54610404906001600160a01b031681565b6040516103209190613758565b60015461033c565b61042c610427366004613641565b610ff3565b60408051938452602084019290925290820152606001610320565b61035a61045536600461376c565b611041565b61033c610468366004613641565b61111d565b61033c61112f565b600c5461048c90600160a01b900463ffffffff1681565b60405163ffffffff9091168152602001610320565b6103ce6104af3660046137f1565b6112c4565b6009546104cf9063ffffffff80821691600160201b90041682565b6040805163ffffffff938416815292909116602083015201610320565b61033c7f6e71edae12b1b97f4d1f60370fef10105fa2faae0126114a169c64845d6126c981565b6006546105209060ff1681565b60405160ff9091168152602001610320565b61033c6112e4565b61054d610548366004613721565b611340565b005b6104047f000000000000000000000000dac17f958d2ee523a2206206994597c13d831ec781565b61035a61137e565b61054d61058c366004613832565b611399565b6105a461059f3660046136dd565b6113d8565b60408051928352602083019190915201610320565b61033c60105481565b61030b6105d0366004613832565b61175f565b6105a46105e336600461384d565b61176a565b61054d6105f6366004613721565b611985565b61033c610609366004613641565b6119e0565b600a54610641906001600160801b03811690600160801b810463ffffffff1690600160a01b90046001600160601b031683565b604080516001600160801b03909416845263ffffffff90921660208401526001600160601b031690820152606001610320565b61033c610682366004613641565b6001600160a01b031660009081526002602052604090205490565b61033c6106ab366004613641565b60076020526000908152604090205481565b6105a4611af7565b61033c6106d3366004613641565b611d7f565b61035a611e46565b61033c6106ee366004613641565b611e53565b61033c600f5481565b61054d61070a36600461387d565b611f26565b61030b6001600160e01b031981565b6103ce61072c3660046136dd565b612125565b61054d61073f36600461391b565b612132565b61054d6107523660046139da565b6121b4565b61033c610765366004613641565b6121e7565b600c5461077e90600160c01b900460ff1681565b6040516103209190613a23565b61033c7f000000000000000000000000000000000000000000000000000000000000000181565b600854610404906001600160a01b031681565b61054d6107d3366004613a4b565b612317565b61033c6107e6366004613641565b612586565b61042c6102fe366004613ac2565b600c54610404906001600160a01b031681565b61033c61081a36600461384d565b6001600160a01b03918216600090815260036020908152604080832093909416825291909152205490565b61054d610853366004613721565b612592565b61054d610866366004613641565b6125c5565b61054d61087936600461391b565b6126a2565b61033c600e5481565b600061089f6000356001600160e01b03191633612717565b6108c45760405162461bcd60e51b81526004016108bb90613b08565b60405180910390fd5b600c54600190600160c01b900460ff1660048111156108e5576108e5613a0d565b8160048111156108f7576108f7613a0d565b146109145760405162461bcd60e51b81526004016108bb90613b2f565b6000836001600160a01b031663dc3bfba96040518163ffffffff1660e01b8152600401602060405180830381865afa158015610954573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906109789190613b62565b90506000600e549050846001600160a01b0316635001f3b56040518163ffffffff1660e01b8152600401602060405180830381865afa1580156109bf573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906109e39190613b62565b6001600160a01b03167f000000000000000000000000dac17f958d2ee523a2206206994597c13d831ec76001600160a01b031614610a555760405162461bcd60e51b815260206004820152600f60248201526e4d69736d617463686564206261736560881b60448201526064016108bb565b6000600e55610a8e6001600160a01b037f000000000000000000000000dac17f958d2ee523a2206206994597c13d831ec716868361274c565b60405163066ad14f60e21b81526001600160a01b038616906319ab453c90610aba903090600401613758565b6060604051808303816000875af1158015610ad9573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610afd9190613b7f565b600f819055600c80546001600160a01b0319166001600160a01b03878116919091179091556040805163204f83f960e01b81529051929850908916935063204f83f992506004808201926020929091908290030181865afa158015610b66573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610b8a9190613bad565b600c805463ffffffff60a01b1916600160a01b63ffffffff9390931692909202919091179055600d80546001600160a01b0319166001600160a01b038716179055610bd660028661284b565b60408051828152602081018690526001600160a01b038716917f9e9d071824fd57d062ca63fd8b786d8da48a6807eebbcb2d83f9e8d21398e28c910160405180910390a2505050919050565b600c54600090600290600160c01b900460ff166004811115610c4657610c46613a0d565b816004811115610c5857610c58613a0d565b14610c755760405162461bcd60e51b81526004016108bb90613b2f565b600d54600c546001600160a01b039182169181169063ffffffff600160a01b9091048116429091161015610ce15760405162461bcd60e51b81526020600482015260136024820152724f6e6c79206166746572206d6174757269747960681b60448201526064016108bb565b6040516370a0823160e01b81526000906001600160a01b038416906370a0823190610d10903090600401613758565b602060405180830381865afa158015610d2d573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610d519190613bca565b6000600f559050610d6c6001600160a01b038416848361274c565b600080846001600160a01b031663d7020d0a303060006000196040518563ffffffff1660e01b8152600401610da49493929190613be3565b6060604051808303816000875af1158015610dc3573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610de79190613b7f565b92509250506000846001600160a01b0316631e9a695030846040518363ffffffff1660e01b8152600401610e1c929190613c0c565b6020604051808303816000875af1158015610e3b573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610e5f9190613bca565b6040516370a0823160e01b81529091506001600160a01b037f000000000000000000000000dac17f958d2ee523a2206206994597c13d831ec716906370a0823190610eae903090600401613758565b602060405180830381865afa158015610ecb573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190610eef9190613bca565b600e55610efd60018761284b565b6001600160a01b038616600080516020613f7b83398151915285610f218487613c3b565b6040805192835260208301829052909b500160405180910390a25050505050505090565b60048054610f5290613c53565b80601f0160208091040260200160405190810160405280929190818152602001828054610f7e90613c53565b8015610fcb5780601f10610fa057610100808354040283529160200191610fcb565b820191906000526020600020905b815481529060010190602001808311610fae57829003601f168201915b505050505081565b6000610fe0338484612a72565b9392505050565b6000610fe08383612717565b600080600061100e6000356001600160e01b03191633612717565b61102a5760405162461bcd60e51b81526004016108bb90613b08565b6000915061103784612adb565b9491935084925050565b60606110596000356001600160e01b03191633612717565b6110755760405162461bcd60e51b81526004016108bb90613b08565b600080856001600160a01b03168585604051611092929190613c87565b6000604051808303816000865af19150503d80600081146110cf576040519150601f19603f3d011682016040523d82523d6000602084013e6110d4565b606091505b5091509150816111145760405162461bcd60e51b815260206004820152600b60248201526a10d85b1b0819985a5b195960aa1b60448201526064016108bb565b95945050505050565b60006111293383612c2e565b92915050565b60006111476000356001600160e01b03191633612717565b6111635760405162461bcd60e51b81526004016108bb90613b08565b600c54600490600160c01b900460ff168181111561118357611183613a0d565b81600481111561119557611195613a0d565b146111b25760405162461bcd60e51b81526004016108bb90613b2f565b6040516370a0823160e01b81526000906001600160a01b037f000000000000000000000000dac17f958d2ee523a2206206994597c13d831ec716906370a0823190611201903090600401613758565b602060405180830381865afa15801561121e573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906112429190613bca565b925082600e8190551161128c5760405162461bcd60e51b8152602060048201526012602482015271139bc818985cd9481d1bc81c995cdd185c9d60721b60448201526064016108bb565b6112966001612cd6565b6040805160008082526020820181905291600080516020613f7b833981519152910160405180910390a25090565b60006112d08483612d3b565b506112dc848484612de4565b949350505050565b60007f0000000000000000000000000000000000000000000000000000000000000001461461131b5761131646612e0d565b905090565b507fb383455bfd8948f5c13ac779f791cd99802412369ed342e263373eb86619319490565b8161135361134d82612e9e565b33612717565b61136f5760405162461bcd60e51b81526004016108bb90613c97565b6113798383612ec0565b505050565b6040805180820190915260018152603160f81b602082015290565b806113a661134d82612e9e565b6113c25760405162461bcd60e51b81526004016108bb90613c97565b6113d4826001600160e01b0319612f2f565b5050565b6000803330146114195760405162461bcd60e51b815260206004820152600c60248201526b155b985d5d1a1bdc9a5e995960a21b60448201526064016108bb565b61142d6001600160a01b038516858561274c565b6040516370a0823160e01b81526000906001600160a01b037f000000000000000000000000dac17f958d2ee523a2206206994597c13d831ec716906370a082319061147c903090600401613758565b602060405180830381865afa158015611499573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906114bd9190613bca565b600c546040516370a0823160e01b81529192506000916001600160a01b03909116906370a08231906114f3903090600401613758565b602060405180830381865afa158015611510573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906115349190613bca565b604051636b81068560e11b81529091506001600160a01b0387169063d7020d0a9061156c903090819060009060001990600401613be3565b6060604051808303816000875af115801561158b573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906115af9190613b7f565b6040516370a0823160e01b8152919650945085915083906001600160a01b037f000000000000000000000000dac17f958d2ee523a2206206994597c13d831ec716906370a0823190611605903090600401613758565b602060405180830381865afa158015611622573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906116469190613bca565b6116509190613cbb565b146116925760405162461bcd60e51b81526020600482015260126024820152714275726e206661696c6564202d206261736560701b60448201526064016108bb565b600c546040516370a0823160e01b8152849183916001600160a01b03909116906370a08231906116c6903090600401613758565b602060405180830381865afa1580156116e3573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906117079190613bca565b6117119190613cbb565b146117565760405162461bcd60e51b8152602060048201526015602482015274213ab937103330b4b632b2101690333caa37b5b2b760591b60448201526064016108bb565b50509250929050565b600061112982612e9e565b6000806003600c60189054906101000a900460ff16600481111561179057611790613a0d565b8160048111156117a2576117a2613a0d565b146117bf5760405162461bcd60e51b81526004016108bb90613b2f565b600c54600e546010546040516370a0823160e01b81526001600160a01b039384169360009184917f000000000000000000000000dac17f958d2ee523a2206206994597c13d831ec716906370a082319061181d903090600401613758565b602060405180830381865afa15801561183a573d6000803e3d6000fd5b505050506040513d601f19601f8201168201806040525081019061185e9190613bca565b6118689190613cbb565b905081811161187957806000611884565b816118848183613cbb565b90975095506118938784613c3b565b600e556118a08783613cbb565b6010819055915060008290036118e5576118ba6001612cd6565b6040805160008082526020820181905291600080516020613f7b833981519152910160405180910390a25b6118f96001600160a01b0385168a8961274c565b80871015611940576119408861190f8984613cbb565b6001600160a01b037f000000000000000000000000dac17f958d2ee523a2206206994597c13d831ec716919061274c565b60408051888152602081018890527fc507be2ec3f5cf464d2bb8665d035a25c6156a546684fb273e1fa95906cd7fd1910160405180910390a150505050509250929050565b6001600160a01b03811633146119d65760405162461bcd60e51b81526020600482015260166024820152752932b737bab731b29037b7363c903337b91039b2b63360511b60448201526064016108bb565b6113d48282612ec0565b600c54600090600290600160c01b900460ff166004811115611a0457611a04613a0d565b816004811115611a1657611a16613a0d565b14611a335760405162461bcd60e51b81526004016108bb90613b2f565b600d54600f546040516370a0823160e01b81526001600160a01b0390921691600090829084906370a0823190611a6d903090600401613758565b602060405180830381865afa158015611a8a573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190611aae9190613bca565b611ab89190613cbb565b9050611ac48183613c3b565b600f556001548290611ad7908390613cd2565b611ae19190613cf1565b9450611aed8686612fae565b5050505050919050565b600080611b106000356001600160e01b03191633612717565b611b2c5760405162461bcd60e51b81526004016108bb90613b08565b600c54600290600160c01b900460ff166004811115611b4d57611b4d613a0d565b816004811115611b5f57611b5f613a0d565b14611b7c5760405162461bcd60e51b81526004016108bb90613b2f565b600d546040516370a0823160e01b81526001600160a01b039091169060009082906370a0823190611bb1903090600401613758565b602060405180830381865afa158015611bce573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190611bf29190613bca565b6000600f556040516358969ab960e01b815290915030906358969ab990611c1f9085908590600401613c0c565b60408051808303816000875af1925050508015611c59575060408051601f3d908101601f19168201909252611c5691810190613d13565b60015b611cc457611c716001600160a01b038316338361274c565b611c7c60048361284b565b816001600160a01b03167fb2559daa129ad136aac2133ac6a0c75920abbef7d6663a017a94e181b13786c382604051611cb791815260200190565b60405180910390a2611d78565b600e8290556010819055909550935084848015611d3757611ce660038561284b565b60408051848152602081018990529081018790526001600160a01b038516907fa4934663617b597bfe6baf57cd7218dfe65e68ad58d417b92ed0bce047cf562a9060600160405180910390a2611d75565b611d4260018561284b565b60408051848152602081018990526001600160a01b03861691600080516020613f7b833981519152910160405180910390a25b50505b5050509091565b600c54600090600290600160c01b900460ff166004811115611da357611da3613a0d565b816004811115611db557611db5613a0d565b14611dd25760405162461bcd60e51b81526004016108bb90613b2f565b600d54600f54600154306000818152600260205260409020546001600160a01b039094169390611e029082612fcc565b5081611e0e8285613cd2565b611e189190613cf1565b9550611e2e6001600160a01b038516888861274c565b611e388684613cbb565b600f55509395945050505050565b60058054610f5290613c53565b600c54600090600190600160c01b900460ff166004811115611e7757611e77613a0d565b816004811115611e8957611e89613a0d565b14611ea65760405162461bcd60e51b81526004016108bb90613b2f565b600e5430600090815260026020526040902054600154611ec68284613cd2565b611ed09190613cf1565b9350611edc8483613cbb565b600e55611ee93082612fcc565b50611f1e6001600160a01b037f000000000000000000000000dac17f958d2ee523a2206206994597c13d831ec716868661274c565b505050919050565b611f3c6000356001600160e01b03191633612717565b611f585760405162461bcd60e51b81526004016108bb90613b08565b8163ffffffff168363ffffffff161115611fa65760405162461bcd60e51b815260206004820152600f60248201526e125b98dbdc9c9958dd081a5b9c1d5d608a1b60448201526064016108bb565b6008546001600160a01b0316611ff65760405162461bcd60e51b815260206004820152601560248201527414995dd85c991cc81d1bdad95b881b9bdd081cd95d605a1b60448201526064016108bb565b60095463ffffffff1661200842612fea565b63ffffffff1610806120365750600954600160201b900463ffffffff1661202e42612fea565b63ffffffff16115b6120745760405162461bcd60e51b815260206004820152600f60248201526e4f6e676f696e67207265776172647360881b60448201526064016108bb565b61207c613014565b6009805463ffffffff85811667ffffffffffffffff199092168217600160201b91861691820217909255600a80546001600160801b0316600160801b83026001600160a01b031617600160a01b6001600160601b03861690810291909117909155604080519283526020830193909352918101919091527f95efd8a2a0aa591f48fd9673cf5d13c2150ca7a1fe1cbe438dd3f0ae470646639060600160405180910390a1505050565b6000610fe0338484612de4565b60005b82518110156113795761216361134d84838151811061215657612156613d37565b6020026020010151612e9e565b61217f5760405162461bcd60e51b81526004016108bb90613c97565b6121a283828151811061219457612194613d37565b602002602001015183612ec0565b806121ac81613d4d565b915050612135565b816121c161134d82612e9e565b6121dd5760405162461bcd60e51b81526004016108bb90613c97565b6113798383612f2f565b600c54600090600190600160c01b900460ff16600481111561220b5761220b613a0d565b81600481111561221d5761221d613a0d565b1461223a5760405162461bcd60e51b81526004016108bb90613b2f565b600e546040516370a0823160e01b815260009082906001600160a01b037f000000000000000000000000dac17f958d2ee523a2206206994597c13d831ec716906370a082319061228e903090600401613758565b602060405180830381865afa1580156122ab573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906122cf9190613bca565b6122d99190613cbb565b90506122e58183613c3b565b600e5560015482906122f8908390613cd2565b6123029190613cf1565b935061230e8585612fae565b50505050919050565b428410156123675760405162461bcd60e51b815260206004820152601d60248201527f45524332305065726d69743a206578706972656420646561646c696e6500000060448201526064016108bb565b6001600160a01b038716600090815260076020526040812080547f6e71edae12b1b97f4d1f60370fef10105fa2faae0126114a169c64845d6126c9918a918a918a9190866123b483613d4d565b909155506040805160208101969096526001600160a01b0394851690860152929091166060840152608083015260a082015260c0810186905260e00160405160208183030381529060405280519060200120905060007f0000000000000000000000000000000000000000000000000000000000000001461461243f5761243a46612e0d565b612461565b7fb383455bfd8948f5c13ac779f791cd99802412369ed342e263373eb8661931945b60405161190160f01b602082015260228101919091526042810183905260620160408051601f198184030181528282528051602091820120600080855291840180845281905260ff89169284019290925260608301879052608083018690529092509060019060a0016020604051602081039080840390855afa1580156124ec573d6000803e3d6000fd5b5050604051601f1901519150506001600160a01b038116158015906125225750896001600160a01b0316816001600160a01b0316145b61256e5760405162461bcd60e51b815260206004820152601e60248201527f45524332305065726d69743a20696e76616c6964207369676e6174757265000060448201526064016108bb565b6125798a8a8a612a72565b5050505050505050505050565b60006111298283612c2e565b8161259f61134d82612e9e565b6125bb5760405162461bcd60e51b81526004016108bb90613c97565b61137983836131ed565b6125db6000356001600160e01b03191633612717565b6125f75760405162461bcd60e51b81526004016108bb90613b08565b6008546001600160a01b03161561264c5760405162461bcd60e51b815260206004820152601960248201527814995dd85c991cc81d1bdad95b88185b1c9958591e481cd95d603a1b60448201526064016108bb565b600880546001600160a01b0319166001600160a01b0383161790556040517f45a4cbe003c343d60028e5acd63eecf588647b0eb369733afa2c0482180d4f7490612697908390613758565b60405180910390a150565b60005b8251811015611379576126c661134d84838151811061215657612156613d37565b6126e25760405162461bcd60e51b81526004016108bb90613c97565b6127058382815181106126f7576126f7613d37565b6020026020010151836131ed565b8061270f81613d4d565b9150506126a5565b6001600160e01b031982166000908152602081815260408083206001600160a01b038516845290915290205460ff1692915050565b600080846001600160a01b031663a9059cbb60e01b8585604051602401612774929190613c0c565b60408051601f198184030181529181526020820180516001600160e01b03166001600160e01b03199094169390931790925290516127b29190613d66565b6000604051808303816000865af19150503d80600081146127ef576040519150601f19603f3d011682016040523d82523d6000602084013e6127f4565b606091505b509150915081801561281e57508051158061281e57508080602001905181019061281e9190613d82565b6128445761282b8161325e565b60405162461bcd60e51b81526004016108bb91906136ba565b5050505050565b600282600481111561285f5761285f613a0d565b0361298e57600d80546001600160a01b0319166001600160a01b0383169081179091556040805163dc3bfba960e01b8152905163dc3bfba9916004808201926020929091908290030181865afa1580156128bd573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906128e19190613b62565b600c60006101000a8154816001600160a01b0302191690836001600160a01b03160217905550806001600160a01b031663204f83f96040518163ffffffff1660e01b8152600401602060405180830381865afa158015612945573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906129699190613bad565b600c60146101000a81548163ffffffff021916908363ffffffff160217905550612a44565b60018260048111156129a2576129a2613a0d565b036129cc57600c80546001600160c01b0319169055600d80546001600160a01b0319169055612a44565b60038260048111156129e0576129e0613a0d565b03612a0a57600c805463ffffffff60a01b19169055600d80546001600160a01b0319169055612a44565b6004826004811115612a1e57612a1e613a0d565b03612a4457600c805463ffffffff60a01b19169055600d80546001600160a01b03191690555b600c805483919060ff60c01b1916600160c01b836004811115612a6957612a69613a0d565b02179055505050565b6001600160a01b03838116600081815260036020908152604080832094871680845294825280832086905551858152919392917f8c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b92591015b60405180910390a35060019392505050565b600c546000908190600160c01b900460ff166004811115612afe57612afe613a0d565b816004811115612b1057612b10613a0d565b14612b2d5760405162461bcd60e51b81526004016108bb90613b2f565b600c80546001600160a01b03191690556040516370a0823160e01b81527f000000000000000000000000dac17f958d2ee523a2206206994597c13d831ec76001600160a01b0316906370a0823190612b89903090600401613758565b602060405180830381865afa158015612ba6573d6000803e3d6000fd5b505050506040513d601f19601f82011682018060405250810190612bca9190613bca565b600e819055915081612c135760405162461bcd60e51b81526020600482015260126024820152712737ba1032b737bab3b4103130b9b29034b760711b60448201526064016108bb565b612c1d8383612fae565b50612c286001612cd6565b50919050565b6000612c38613014565b612c41836132bd565b6001600160a01b038085166000908152600b6020526040902080546001600160801b03191690556008546001600160801b03929092169250612c859116838361274c565b604080516001600160a01b038086168252841660208201529081018290527ff7a40077ff7a04c7e61f6f26fb13774259ddf1b6bce9ecf26a8276cdd39926839060600160405180910390a192915050565b6002816004811115612cea57612cea613a0d565b03612d2d5760405162461bcd60e51b8152602060048201526013602482015272135d5cdd081c1c9bdd9a59194818481c1bdbdb606a1b60448201526064016108bb565b612d3881600061284b565b50565b60006001600160a01b0383163314612ddb576001600160a01b03831660009081526003602090815260408083203384529091529020546000198114612dd95782811015612dca5760405162461bcd60e51b815260206004820152601c60248201527f45524332303a20496e73756666696369656e7420617070726f76616c0000000060448201526064016108bb565b612dd78433858403612a72565b505b505b50600192915050565b6000612dee613014565b612df7846132bd565b50612e01836132bd565b506112dc84848461342d565b60007f8b73c3c69bb8fe3d512ecc4cf759cc79239f7b179b0ffacaa9a75d522b39400f6004604051612e3f9190613da4565b6040518091039020612e4f61137e565b80516020918201206040805192830194909452928101919091526060810191909152608081018390523060a082015260c001604051602081830303815290604052805190602001209050919050565b6001600160e01b03191660009081526020819052604090206001015460e01b90565b612eca8282612717565b156113d4576001600160e01b031982166000818152602081815260408083206001600160a01b0386168085529252808320805460ff1916905551339391927f4ddc7b757e7bdd7254a9cd39452d307a52761bc824625c6a33104a075d8099e691a45050565b6001600160e01b03198116612f4383612e9e565b6001600160e01b031916146113d4576001600160e01b0319828116600081815260208190526040808220600101805463ffffffff191660e087901c17905551928416927fd348e2220a50b4500ec353f6e802d2f14dd1b5d6786148fd1bbcc570bf92d4739190a35050565b6000612fb8613014565b612fc1836132bd565b50610fe083836134da565b6000612fd6613014565b612fdf836132bd565b50610fe08383613561565b600063ffffffff8211156130105760405162461bcd60e51b81526004016108bb90613e43565b5090565b60408051606081018252600a546001600160801b0381168252600160801b810463ffffffff908116602080850191909152600160a01b9092046001600160601b0316838501528351808501909452600954808216808652600160201b909104909116918401919091526001549192919061308d42612fea565b63ffffffff16101561309e57505050565b60006130b66130ac42612fea565b84602001516135e1565b905060008460200151826130ca9190613e6a565b63ffffffff169050806000036130e1575050505050565b82156131475761313b8386604001516001600160601b031683670de0b6b3a764000061310d9190613cd2565b6131179190613cd2565b6131219190613cf1565b865161313691906001600160801b0316613c3b565b613603565b6001600160801b031685525b63ffffffff8216602086018190528551600a80546040808a01516001600160601b0316600160a01b026001600160a01b03600160801b9096026001600160a01b03199093166001600160801b03861617929092179490941617905590517fe972555b20cae8150e291bb40efce3ef4e3ed6b6b2c39c029346600e95469d48916131de916001600160801b0391909116815260200190565b60405180910390a15050505050565b6131f78282612717565b6113d4576001600160e01b031982166000818152602081815260408083206001600160a01b0386168085529252808320805460ff1916600117905551339391927fe6231789d19137da31d0550f4ba9ee379020a8cfb64cb79bf1790c996d2e616591a45050565b60606044825110156132a357505060408051808201909152601d81527f5472616e73616374696f6e2072657665727465642073696c656e746c79000000602082015290565b600482019150818060200190518101906111299190613e8f565b6001600160a01b0381166000908152600b602090815260408083208151808301835290546001600160801b038082168352600160801b9182900481168386019081528451606081018652600a5492831680825293830463ffffffff1696810196909652600160a01b9091046001600160601b03169385019390935291519092916133a091670de0b6b3a7640000916133559190613f23565b6001600160a01b038716600090815260026020526040902054613381916001600160801b031690613cd2565b61338b9190613cf1565b835161313691906001600160801b0316613c3b565b6001600160801b0390811683528151811660208085019182526001600160a01b0387166000818152600b835260409081902087519451948616600160801b95909616948502861790558051918252918101939093528201527f5b9aaf4cc5141c090a75f8b8a627863eba92df81f0c83c096350da9b79aedd049060600160405180910390a1505192915050565b6001600160a01b0383166000908152600260205260408120548211156134655760405162461bcd60e51b81526004016108bb90613f43565b6001600160a01b038085166000908152600260205260408082208054869003905591851681522054613498908390613c3b565b6001600160a01b038085166000818152600260205260409081902093909355915190861690600080516020613f9b83398151915290612ac99086815260200190565b6001600160a01b0382166000908152600260205260408120546134fe908390613c3b565b6001600160a01b038416600090815260026020526040902055600154613525908390613c3b565b6001556040518281526001600160a01b03841690600090600080516020613f9b833981519152906020015b60405180910390a350600192915050565b6001600160a01b0382166000908152600260205260408120548211156135995760405162461bcd60e51b81526004016108bb90613f43565b6001600160a01b03831660008181526002602090815260408083208054879003905560018054879003905551858152919291600080516020613f9b8339815191529101613550565b60008163ffffffff168363ffffffff16106135fc5781610fe0565b5090919050565b60006001600160801b038211156130105760405162461bcd60e51b81526004016108bb90613e43565b6001600160a01b0381168114612d3857600080fd5b60006020828403121561365357600080fd5b8135610fe08161362c565b60005b83811015613679578181015183820152602001613661565b83811115613688576000848401525b50505050565b600081518084526136a681602086016020860161365e565b601f01601f19169290920160200192915050565b602081526000610fe0602083018461368e565b80356136d88161362c565b919050565b600080604083850312156136f057600080fd5b82356136fb8161362c565b946020939093013593505050565b80356001600160e01b0319811681146136d857600080fd5b6000806040838503121561373457600080fd5b61373d83613709565b9150602083013561374d8161362c565b809150509250929050565b6001600160a01b0391909116815260200190565b60008060006040848603121561378157600080fd5b833561378c8161362c565b9250602084013567ffffffffffffffff808211156137a957600080fd5b818601915086601f8301126137bd57600080fd5b8135818111156137cc57600080fd5b8760208285010111156137de57600080fd5b6020830194508093505050509250925092565b60008060006060848603121561380657600080fd5b83356138118161362c565b925060208401356138218161362c565b929592945050506040919091013590565b60006020828403121561384457600080fd5b610fe082613709565b6000806040838503121561386057600080fd5b823561373d8161362c565b63ffffffff81168114612d3857600080fd5b60008060006060848603121561389257600080fd5b833561389d8161386b565b925060208401356138ad8161386b565b915060408401356001600160601b03811681146138c957600080fd5b809150509250925092565b634e487b7160e01b600052604160045260246000fd5b604051601f8201601f1916810167ffffffffffffffff81118282101715613913576139136138d4565b604052919050565b6000806040838503121561392e57600080fd5b823567ffffffffffffffff8082111561394657600080fd5b818501915085601f83011261395a57600080fd5b813560208282111561396e5761396e6138d4565b8160051b925061397f8184016138ea565b828152928401810192818101908985111561399957600080fd5b948201945b848610156139be576139af86613709565b8252948201949082019061399e565b96506139cd90508782016136cd565b9450505050509250929050565b600080604083850312156139ed57600080fd5b6139f683613709565b9150613a0460208401613709565b90509250929050565b634e487b7160e01b600052602160045260246000fd5b6020810160058310613a4557634e487b7160e01b600052602160045260246000fd5b91905290565b600080600080600080600060e0888a031215613a6657600080fd5b8735613a718161362c565b96506020880135613a818161362c565b95506040880135945060608801359350608088013560ff81168114613aa557600080fd5b9699959850939692959460a0840135945060c09093013592915050565b60008060008060808587031215613ad857600080fd5b8435613ae38161362c565b93506020850135613af38161362c565b93969395505050506040820135916060013590565b6020808252600d908201526c1058d8d95cdcc819195b9a5959609a1b604082015260600190565b6020808252601990820152784e6f7420616c6c6f77656420696e207468697320737461746560381b604082015260600190565b600060208284031215613b7457600080fd5b8151610fe08161362c565b600080600060608486031215613b9457600080fd5b8351925060208401519150604084015190509250925092565b600060208284031215613bbf57600080fd5b8151610fe08161386b565b600060208284031215613bdc57600080fd5b5051919050565b6001600160a01b0394851681529290931660208301526040820152606081019190915260800190565b6001600160a01b03929092168252602082015260400190565b634e487b7160e01b600052601160045260246000fd5b60008219821115613c4e57613c4e613c25565b500190565b600181811c90821680613c6757607f821691505b602082108103612c2857634e487b7160e01b600052602260045260246000fd5b8183823760009101908152919050565b6020808252600a908201526927b7363c9030b236b4b760b11b604082015260600190565b600082821015613ccd57613ccd613c25565b500390565b6000816000190483118215151615613cec57613cec613c25565b500290565b600082613d0e57634e487b7160e01b600052601260045260246000fd5b500490565b60008060408385031215613d2657600080fd5b505080516020909101519092909150565b634e487b7160e01b600052603260045260246000fd5b600060018201613d5f57613d5f613c25565b5060010190565b60008251613d7881846020870161365e565b9190910192915050565b600060208284031215613d9457600080fd5b81518015158114610fe057600080fd5b600080835481600182811c915080831680613dc057607f831692505b60208084108203613ddf57634e487b7160e01b86526022600452602486fd5b818015613df35760018114613e0857613e35565b60ff1986168952841515850289019650613e35565b60008a81526020902060005b86811015613e2d5781548b820152908501908301613e14565b505084890196505b509498975050505050505050565b6020808252600d908201526c43617374206f766572666c6f7760981b604082015260600190565b600063ffffffff83811690831681811015613e8757613e87613c25565b039392505050565b600060208284031215613ea157600080fd5b815167ffffffffffffffff80821115613eb957600080fd5b818401915084601f830112613ecd57600080fd5b815181811115613edf57613edf6138d4565b613ef2601f8201601f19166020016138ea565b9150808252856020828501011115613f0957600080fd5b613f1a81602084016020860161365e565b50949350505050565b60006001600160801b0383811690831681811015613e8757613e87613c25565b6020808252601b908201527f45524332303a20496e73756666696369656e742062616c616e6365000000000060408201526060019056fef44b6ecb6421462dee6400bd4e3bb57864c0f428d0f7e7d49771f9fd7c30d4faddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3efa2646970667358221220b13b6f1a20a6b787e1b7c87a84f7f85eb6c255d3c2528550033e66796d4d042d64736f6c634300080f0033

Constructor Arguments (ABI-Encoded and is the last bytes of the Contract Creation Code above)

000000000000000000000000000000000000000000000000000000000000006000000000000000000000000000000000000000000000000000000000000000a0000000000000000000000000d28380de0e7093ac62bcb88610b9f4f4fb58be74000000000000000000000000000000000000000000000000000000000000001e5969656c64205374726174656779205553445420364d204a756e204465630000000000000000000000000000000000000000000000000000000000000000000a595355534454364d4a4400000000000000000000000000000000000000000000

-----Decoded View---------------
Arg [0] : name_ (string): Yield Strategy USDT 6M Jun Dec
Arg [1] : symbol_ (string): YSUSDT6MJD
Arg [2] : fyToken_ (address): 0xD28380De0e7093AC62bCb88610b9f4f4Fb58Be74

-----Encoded View---------------
7 Constructor Arguments found :
Arg [0] : 0000000000000000000000000000000000000000000000000000000000000060
Arg [1] : 00000000000000000000000000000000000000000000000000000000000000a0
Arg [2] : 000000000000000000000000d28380de0e7093ac62bcb88610b9f4f4fb58be74
Arg [3] : 000000000000000000000000000000000000000000000000000000000000001e
Arg [4] : 5969656c64205374726174656779205553445420364d204a756e204465630000
Arg [5] : 000000000000000000000000000000000000000000000000000000000000000a
Arg [6] : 595355534454364d4a4400000000000000000000000000000000000000000000


Block Transaction Difficulty Gas Used Reward
View All Blocks Produced

Block Uncle Number Difficulty Gas Used Reward
View All Uncles
Loading...
Loading
Loading...
Loading

Validator Index Block Amount
View All Withdrawals

Transaction Hash Block Value Eth2 PubKey Valid
View All Deposits
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.