ETH Price: $3,670.21 (+0.92%)
 
Transaction Hash
Method
Block
From
To
Withdraw Locked209494472024-10-12 12:13:3585 days ago1728735215IN
0x39cd4db6...B7Ca4D5F6
0 ETH0.0120833111.40948329
Sync206021312024-08-25 0:48:59134 days ago1724546939IN
0x39cd4db6...B7Ca4D5F6
0 ETH0.000606221.36394537
Withdraw Locked205654832024-08-19 21:53:35139 days ago1724104415IN
0x39cd4db6...B7Ca4D5F6
0 ETH0.001024171.69710259
Get Reward2205654812024-08-19 21:53:11139 days ago1724104391IN
0x39cd4db6...B7Ca4D5F6
0 ETH0.000634321.54787051
Sync205305692024-08-15 0:52:11144 days ago1723683131IN
0x39cd4db6...B7Ca4D5F6
0 ETH0.000707721.43715622
Sync197563822024-04-28 21:05:35252 days ago1714338335IN
0x39cd4db6...B7Ca4D5F6
0 ETH0.000734636.17961862
Sync190734222024-01-24 2:07:59348 days ago1706062079IN
0x39cd4db6...B7Ca4D5F6
0 ETH0.0011329310.25463749
Sync187519942023-12-09 23:39:23393 days ago1702165163IN
0x39cd4db6...B7Ca4D5F6
0 ETH0.0064876546.11971214
Withdraw Locked184094552023-10-23 0:37:59441 days ago1698021479IN
0x39cd4db6...B7Ca4D5F6
0 ETH0.004691617.20849289
Get Reward2183834902023-10-19 9:29:23444 days ago1697707763IN
0x39cd4db6...B7Ca4D5F6
0 ETH0.004756947.77521727
Get Reward2182075862023-09-24 18:57:35469 days ago1695581855IN
0x39cd4db6...B7Ca4D5F6
0 ETH0.0038990710.2082476
Sync181926352023-09-22 16:42:11471 days ago1695400931IN
0x39cd4db6...B7Ca4D5F6
0 ETH0.0047360110.11034617
Sync181141282023-09-11 15:53:35482 days ago1694447615IN
0x39cd4db6...B7Ca4D5F6
0 ETH0.0153262734.4825345
Withdraw Locked177850372023-07-27 14:36:11528 days ago1690468571IN
0x39cd4db6...B7Ca4D5F6
0 ETH0.0288664443.39722081
Withdraw Locked177848432023-07-27 13:57:23528 days ago1690466243IN
0x39cd4db6...B7Ca4D5F6
0 ETH0.0311966437.80014415
Sync177785682023-07-26 16:52:47529 days ago1690390367IN
0x39cd4db6...B7Ca4D5F6
0 ETH0.0046260732.88600796
Withdraw Locked177784302023-07-26 16:25:11529 days ago1690388711IN
0x39cd4db6...B7Ca4D5F6
0 ETH0.0197026831.47232696
Get Reward177585622023-07-23 21:41:59532 days ago1690148519IN
0x39cd4db6...B7Ca4D5F6
0 ETH0.0084671117.90390266
Get Reward177585302023-07-23 21:35:35532 days ago1690148135IN
0x39cd4db6...B7Ca4D5F6
0 ETH0.0095829417.88030071
Get Reward176943992023-07-14 21:35:11541 days ago1689370511IN
0x39cd4db6...B7Ca4D5F6
0 ETH0.0059676813.23197248
Get Reward176851422023-07-13 14:22:35542 days ago1689258155IN
0x39cd4db6...B7Ca4D5F6
0 ETH0.0083188324.44709845
Stake Locked176851392023-07-13 14:21:59542 days ago1689258119IN
0x39cd4db6...B7Ca4D5F6
0 ETH0.0173024924.23987742
Get Reward176703702023-07-11 12:28:11544 days ago1689078491IN
0x39cd4db6...B7Ca4D5F6
0 ETH0.0070165616.40189284
Get Reward176703662023-07-11 12:27:23544 days ago1689078443IN
0x39cd4db6...B7Ca4D5F6
0 ETH0.0069966516.70790964
Stake Locked176703632023-07-11 12:26:47544 days ago1689078407IN
0x39cd4db6...B7Ca4D5F6
0 ETH0.0108513416.1155565
View all transactions

View more zero value Internal Transactions in Advanced View mode

Advanced mode:
Loading...
Loading

Similar Match Source Code
This contract matches the deployed Bytecode of the Source Code for Contract 0x94c491e2...2c59FAbc1
The constructor portion of the code might be different and could alter the actual behaviour of the contract

Contract Name:
FraxUnifiedFarm_ERC20_Convex_FRAXBP_Volatile

Compiler Version
v0.8.17+commit.8df45f5f

Optimization Enabled:
Yes with 100000 runs

Other Settings:
default evmVersion, GNU GPLv2 license

Contract Source Code (Solidity)

/**
 *Submitted for verification at Etherscan.io on 2023-05-13
*/

// SPDX-License-Identifier: GPL-2.0-or-later
pragma solidity >=0.8.0;

// Sources flattened with hardhat v2.14.0 https://hardhat.org

// File contracts/Common/Context.sol


/*
 * @dev Provides information about the current execution context, including the
 * sender of the transaction and its data. While these are generally available
 * via msg.sender and msg.data, they should not be accessed in such a direct
 * manner, since when dealing with GSN meta-transactions the account sending and
 * paying for execution may not be the actual sender (as far as an application
 * is concerned).
 *
 * This contract is only required for intermediate, library-like contracts.
 */
abstract contract Context {
    function _msgSender() internal view virtual returns (address payable) {
        return payable(msg.sender);
    }

    function _msgData() internal view virtual returns (bytes memory) {
        this; // silence state mutability warning without generating bytecode - see https://github.com/ethereum/solidity/issues/2691
        return msg.data;
    }
}


// File contracts/Math/SafeMath.sol


/**
 * @dev Wrappers over Solidity's arithmetic operations with added overflow
 * checks.
 *
 * Arithmetic operations in Solidity wrap on overflow. This can easily result
 * in bugs, because programmers usually assume that an overflow raises an
 * error, which is the standard behavior in high level programming languages.
 * `SafeMath` restores this intuition by reverting the transaction when an
 * operation overflows.
 *
 * Using this library instead of the unchecked operations eliminates an entire
 * class of bugs, so it's recommended to use it always.
 */
library SafeMath {
    /**
     * @dev Returns the addition of two unsigned integers, reverting on
     * overflow.
     *
     * Counterpart to Solidity's `+` operator.
     *
     * Requirements:
     * - Addition cannot overflow.
     */
    function add(uint256 a, uint256 b) internal pure returns (uint256) {
        uint256 c = a + b;
        require(c >= a, "SafeMath: addition overflow");

        return c;
    }

    /**
     * @dev Returns the subtraction of two unsigned integers, reverting on
     * overflow (when the result is negative).
     *
     * Counterpart to Solidity's `-` operator.
     *
     * Requirements:
     * - Subtraction cannot overflow.
     */
    function sub(uint256 a, uint256 b) internal pure returns (uint256) {
        return sub(a, b, "SafeMath: subtraction overflow");
    }

    /**
     * @dev Returns the subtraction of two unsigned integers, reverting with custom message on
     * overflow (when the result is negative).
     *
     * Counterpart to Solidity's `-` operator.
     *
     * Requirements:
     * - Subtraction cannot overflow.
     *
     * _Available since v2.4.0._
     */
    function sub(uint256 a, uint256 b, string memory errorMessage) internal pure returns (uint256) {
        require(b <= a, errorMessage);
        uint256 c = a - b;

        return c;
    }

    /**
     * @dev Returns the multiplication of two unsigned integers, reverting on
     * overflow.
     *
     * Counterpart to Solidity's `*` operator.
     *
     * Requirements:
     * - Multiplication cannot overflow.
     */
    function mul(uint256 a, uint256 b) internal pure returns (uint256) {
        // Gas optimization: this is cheaper than requiring 'a' not being zero, but the
        // benefit is lost if 'b' is also tested.
        // See: https://github.com/OpenZeppelin/openzeppelin-contracts/pull/522
        if (a == 0) {
            return 0;
        }

        uint256 c = a * b;
        require(c / a == b, "SafeMath: multiplication overflow");

        return c;
    }

    /**
     * @dev Returns the integer division of two unsigned integers. Reverts on
     * division by zero. The result is rounded towards zero.
     *
     * Counterpart to Solidity's `/` operator. Note: this function uses a
     * `revert` opcode (which leaves remaining gas untouched) while Solidity
     * uses an invalid opcode to revert (consuming all remaining gas).
     *
     * Requirements:
     * - The divisor cannot be zero.
     */
    function div(uint256 a, uint256 b) internal pure returns (uint256) {
        return div(a, b, "SafeMath: division by zero");
    }

    /**
     * @dev Returns the integer division of two unsigned integers. Reverts with custom message on
     * division by zero. The result is rounded towards zero.
     *
     * Counterpart to Solidity's `/` operator. Note: this function uses a
     * `revert` opcode (which leaves remaining gas untouched) while Solidity
     * uses an invalid opcode to revert (consuming all remaining gas).
     *
     * Requirements:
     * - The divisor cannot be zero.
     *
     * _Available since v2.4.0._
     */
    function div(uint256 a, uint256 b, string memory errorMessage) internal pure returns (uint256) {
        // Solidity only automatically asserts when dividing by 0
        require(b > 0, errorMessage);
        uint256 c = a / b;
        // assert(a == b * c + a % b); // There is no case in which this doesn't hold

        return c;
    }

    /**
     * @dev Returns the remainder of dividing two unsigned integers. (unsigned integer modulo),
     * Reverts when dividing by zero.
     *
     * Counterpart to Solidity's `%` operator. This function uses a `revert`
     * opcode (which leaves remaining gas untouched) while Solidity uses an
     * invalid opcode to revert (consuming all remaining gas).
     *
     * Requirements:
     * - The divisor cannot be zero.
     */
    function mod(uint256 a, uint256 b) internal pure returns (uint256) {
        return mod(a, b, "SafeMath: modulo by zero");
    }

    /**
     * @dev Returns the remainder of dividing two unsigned integers. (unsigned integer modulo),
     * Reverts with custom message when dividing by zero.
     *
     * Counterpart to Solidity's `%` operator. This function uses a `revert`
     * opcode (which leaves remaining gas untouched) while Solidity uses an
     * invalid opcode to revert (consuming all remaining gas).
     *
     * Requirements:
     * - The divisor cannot be zero.
     *
     * _Available since v2.4.0._
     */
    function mod(uint256 a, uint256 b, string memory errorMessage) internal pure returns (uint256) {
        require(b != 0, errorMessage);
        return a % b;
    }
}


// File contracts/ERC20/IERC20.sol



/**
 * @dev Interface of the ERC20 standard as defined in the EIP. Does not include
 * the optional functions; to access them see {ERC20Detailed}.
 */
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 contracts/Uniswap_V3/pool/IUniswapV3PoolActions.sol


/// @title Permissionless pool actions
/// @notice Contains pool methods that can be called by anyone
interface IUniswapV3PoolActions {
    /// @notice Sets the initial price for the pool
    /// @dev Price is represented as a sqrt(amountToken1/amountToken0) Q64.96 value
    /// @param sqrtPriceX96 the initial sqrt price of the pool as a Q64.96
    function initialize(uint160 sqrtPriceX96) external;

    /// @notice Adds liquidity for the given recipient/tickLower/tickUpper position
    /// @dev The caller of this method receives a callback in the form of IUniswapV3MintCallback#uniswapV3MintCallback
    /// in which they must pay any token0 or token1 owed for the liquidity. The amount of token0/token1 due depends
    /// on tickLower, tickUpper, the amount of liquidity, and the current price.
    /// @param recipient The address for which the liquidity will be created
    /// @param tickLower The lower tick of the position in which to add liquidity
    /// @param tickUpper The upper tick of the position in which to add liquidity
    /// @param amount The amount of liquidity to mint
    /// @param data Any data that should be passed through to the callback
    /// @return amount0 The amount of token0 that was paid to mint the given amount of liquidity. Matches the value in the callback
    /// @return amount1 The amount of token1 that was paid to mint the given amount of liquidity. Matches the value in the callback
    function mint(
        address recipient,
        int24 tickLower,
        int24 tickUpper,
        uint128 amount,
        bytes calldata data
    ) external returns (uint256 amount0, uint256 amount1);

    /// @notice Collects tokens owed to a position
    /// @dev Does not recompute fees earned, which must be done either via mint or burn of any amount of liquidity.
    /// Collect must be called by the position owner. To withdraw only token0 or only token1, amount0Requested or
    /// amount1Requested may be set to zero. To withdraw all tokens owed, caller may pass any value greater than the
    /// actual tokens owed, e.g. type(uint128).max. Tokens owed may be from accumulated swap fees or burned liquidity.
    /// @param recipient The address which should receive the fees collected
    /// @param tickLower The lower tick of the position for which to collect fees
    /// @param tickUpper The upper tick of the position for which to collect fees
    /// @param amount0Requested How much token0 should be withdrawn from the fees owed
    /// @param amount1Requested How much token1 should be withdrawn from the fees owed
    /// @return amount0 The amount of fees collected in token0
    /// @return amount1 The amount of fees collected in token1
    function collect(
        address recipient,
        int24 tickLower,
        int24 tickUpper,
        uint128 amount0Requested,
        uint128 amount1Requested
    ) external returns (uint128 amount0, uint128 amount1);

    /// @notice Burn liquidity from the sender and account tokens owed for the liquidity to the position
    /// @dev Can be used to trigger a recalculation of fees owed to a position by calling with an amount of 0
    /// @dev Fees must be collected separately via a call to #collect
    /// @param tickLower The lower tick of the position for which to burn liquidity
    /// @param tickUpper The upper tick of the position for which to burn liquidity
    /// @param amount How much liquidity to burn
    /// @return amount0 The amount of token0 sent to the recipient
    /// @return amount1 The amount of token1 sent to the recipient
    function burn(
        int24 tickLower,
        int24 tickUpper,
        uint128 amount
    ) external returns (uint256 amount0, uint256 amount1);

    /// @notice Swap token0 for token1, or token1 for token0
    /// @dev The caller of this method receives a callback in the form of IUniswapV3SwapCallback#uniswapV3SwapCallback
    /// @param recipient The address to receive the output of the swap
    /// @param zeroForOne The direction of the swap, true for token0 to token1, false for token1 to token0
    /// @param amountSpecified The amount of the swap, which implicitly configures the swap as exact input (positive), or exact output (negative)
    /// @param sqrtPriceLimitX96 The Q64.96 sqrt price limit. If zero for one, the price cannot be less than this
    /// value after the swap. If one for zero, the price cannot be greater than this value after the swap
    /// @param data Any data to be passed through to the callback
    /// @return amount0 The delta of the balance of token0 of the pool, exact when negative, minimum when positive
    /// @return amount1 The delta of the balance of token1 of the pool, exact when negative, minimum when positive
    function swap(
        address recipient,
        bool zeroForOne,
        int256 amountSpecified,
        uint160 sqrtPriceLimitX96,
        bytes calldata data
    ) external returns (int256 amount0, int256 amount1);

    /// @notice Receive token0 and/or token1 and pay it back, plus a fee, in the callback
    /// @dev The caller of this method receives a callback in the form of IUniswapV3FlashCallback#uniswapV3FlashCallback
    /// @dev Can be used to donate underlying tokens pro-rata to currently in-range liquidity providers by calling
    /// with 0 amount{0,1} and sending the donation amount(s) from the callback
    /// @param recipient The address which will receive the token0 and token1 amounts
    /// @param amount0 The amount of token0 to send
    /// @param amount1 The amount of token1 to send
    /// @param data Any data to be passed through to the callback
    function flash(
        address recipient,
        uint256 amount0,
        uint256 amount1,
        bytes calldata data
    ) external;

    /// @notice Increase the maximum number of price and liquidity observations that this pool will store
    /// @dev This method is no-op if the pool already has an observationCardinalityNext greater than or equal to
    /// the input observationCardinalityNext.
    /// @param observationCardinalityNext The desired minimum number of observations for the pool to store
    function increaseObservationCardinalityNext(uint16 observationCardinalityNext) external;
}


// File contracts/Uniswap_V3/pool/IUniswapV3PoolDerivedState.sol


/// @title Pool state that is not stored
/// @notice Contains view functions to provide information about the pool that is computed rather than stored on the
/// blockchain. The functions here may have variable gas costs.
interface IUniswapV3PoolDerivedState {
    /// @notice Returns the cumulative tick and liquidity as of each timestamp `secondsAgo` from the current block timestamp
    /// @dev To get a time weighted average tick or liquidity-in-range, you must call this with two values, one representing
    /// the beginning of the period and another for the end of the period. E.g., to get the last hour time-weighted average tick,
    /// you must call it with secondsAgos = [3600, 0].
    /// @dev The time weighted average tick represents the geometric time weighted average price of the pool, in
    /// log base sqrt(1.0001) of token1 / token0. The TickMath library can be used to go from a tick value to a ratio.
    /// @param secondsAgos From how long ago each cumulative tick and liquidity value should be returned
    /// @return tickCumulatives Cumulative tick values as of each `secondsAgos` from the current block timestamp
    /// @return secondsPerLiquidityCumulativeX128s Cumulative seconds per liquidity-in-range value as of each `secondsAgos` from the current block
    /// timestamp
    function observe(uint32[] calldata secondsAgos)
        external
        view
        returns (int56[] memory tickCumulatives, uint160[] memory secondsPerLiquidityCumulativeX128s);

    /// @notice Returns a snapshot of the tick cumulative, seconds per liquidity and seconds inside a tick range
    /// @dev Snapshots must only be compared to other snapshots, taken over a period for which a position existed.
    /// I.e., snapshots cannot be compared if a position is not held for the entire period between when the first
    /// snapshot is taken and the second snapshot is taken.
    /// @param tickLower The lower tick of the range
    /// @param tickUpper The upper tick of the range
    /// @return tickCumulativeInside The snapshot of the tick accumulator for the range
    /// @return secondsPerLiquidityInsideX128 The snapshot of seconds per liquidity for the range
    /// @return secondsInside The snapshot of seconds per liquidity for the range
    function snapshotCumulativesInside(int24 tickLower, int24 tickUpper)
        external
        view
        returns (
            int56 tickCumulativeInside,
            uint160 secondsPerLiquidityInsideX128,
            uint32 secondsInside
        );
}


// File contracts/Uniswap_V3/pool/IUniswapV3PoolEvents.sol


/// @title Events emitted by a pool
/// @notice Contains all events emitted by the pool
interface IUniswapV3PoolEvents {
    /// @notice Emitted exactly once by a pool when #initialize is first called on the pool
    /// @dev Mint/Burn/Swap cannot be emitted by the pool before Initialize
    /// @param sqrtPriceX96 The initial sqrt price of the pool, as a Q64.96
    /// @param tick The initial tick of the pool, i.e. log base 1.0001 of the starting price of the pool
    event Initialize(uint160 sqrtPriceX96, int24 tick);

    /// @notice Emitted when liquidity is minted for a given position
    /// @param sender The address that minted the liquidity
    /// @param owner The owner of the position and recipient of any minted liquidity
    /// @param tickLower The lower tick of the position
    /// @param tickUpper The upper tick of the position
    /// @param amount The amount of liquidity minted to the position range
    /// @param amount0 How much token0 was required for the minted liquidity
    /// @param amount1 How much token1 was required for the minted liquidity
    event Mint(
        address sender,
        address indexed owner,
        int24 indexed tickLower,
        int24 indexed tickUpper,
        uint128 amount,
        uint256 amount0,
        uint256 amount1
    );

    /// @notice Emitted when fees are collected by the owner of a position
    /// @dev Collect events may be emitted with zero amount0 and amount1 when the caller chooses not to collect fees
    /// @param owner The owner of the position for which fees are collected
    /// @param tickLower The lower tick of the position
    /// @param tickUpper The upper tick of the position
    /// @param amount0 The amount of token0 fees collected
    /// @param amount1 The amount of token1 fees collected
    event Collect(
        address indexed owner,
        address recipient,
        int24 indexed tickLower,
        int24 indexed tickUpper,
        uint128 amount0,
        uint128 amount1
    );

    /// @notice Emitted when a position's liquidity is removed
    /// @dev Does not withdraw any fees earned by the liquidity position, which must be withdrawn via #collect
    /// @param owner The owner of the position for which liquidity is removed
    /// @param tickLower The lower tick of the position
    /// @param tickUpper The upper tick of the position
    /// @param amount The amount of liquidity to remove
    /// @param amount0 The amount of token0 withdrawn
    /// @param amount1 The amount of token1 withdrawn
    event Burn(
        address indexed owner,
        int24 indexed tickLower,
        int24 indexed tickUpper,
        uint128 amount,
        uint256 amount0,
        uint256 amount1
    );

    /// @notice Emitted by the pool for any swaps between token0 and token1
    /// @param sender The address that initiated the swap call, and that received the callback
    /// @param recipient The address that received the output of the swap
    /// @param amount0 The delta of the token0 balance of the pool
    /// @param amount1 The delta of the token1 balance of the pool
    /// @param sqrtPriceX96 The sqrt(price) of the pool after the swap, as a Q64.96
    /// @param liquidity The liquidity of the pool after the swap
    /// @param tick The log base 1.0001 of price of the pool after the swap
    event Swap(
        address indexed sender,
        address indexed recipient,
        int256 amount0,
        int256 amount1,
        uint160 sqrtPriceX96,
        uint128 liquidity,
        int24 tick
    );

    /// @notice Emitted by the pool for any flashes of token0/token1
    /// @param sender The address that initiated the swap call, and that received the callback
    /// @param recipient The address that received the tokens from flash
    /// @param amount0 The amount of token0 that was flashed
    /// @param amount1 The amount of token1 that was flashed
    /// @param paid0 The amount of token0 paid for the flash, which can exceed the amount0 plus the fee
    /// @param paid1 The amount of token1 paid for the flash, which can exceed the amount1 plus the fee
    event Flash(
        address indexed sender,
        address indexed recipient,
        uint256 amount0,
        uint256 amount1,
        uint256 paid0,
        uint256 paid1
    );

    /// @notice Emitted by the pool for increases to the number of observations that can be stored
    /// @dev observationCardinalityNext is not the observation cardinality until an observation is written at the index
    /// just before a mint/swap/burn.
    /// @param observationCardinalityNextOld The previous value of the next observation cardinality
    /// @param observationCardinalityNextNew The updated value of the next observation cardinality
    event IncreaseObservationCardinalityNext(
        uint16 observationCardinalityNextOld,
        uint16 observationCardinalityNextNew
    );

    /// @notice Emitted when the protocol fee is changed by the pool
    /// @param feeProtocol0Old The previous value of the token0 protocol fee
    /// @param feeProtocol1Old The previous value of the token1 protocol fee
    /// @param feeProtocol0New The updated value of the token0 protocol fee
    /// @param feeProtocol1New The updated value of the token1 protocol fee
    event SetFeeProtocol(uint8 feeProtocol0Old, uint8 feeProtocol1Old, uint8 feeProtocol0New, uint8 feeProtocol1New);

    /// @notice Emitted when the collected protocol fees are withdrawn by the factory owner
    /// @param sender The address that collects the protocol fees
    /// @param recipient The address that receives the collected protocol fees
    /// @param amount0 The amount of token0 protocol fees that is withdrawn
    /// @param amount0 The amount of token1 protocol fees that is withdrawn
    event CollectProtocol(address indexed sender, address indexed recipient, uint128 amount0, uint128 amount1);
}


// File contracts/Uniswap_V3/pool/IUniswapV3PoolImmutables.sol


/// @title Pool state that never changes
/// @notice These parameters are fixed for a pool forever, i.e., the methods will always return the same values
interface IUniswapV3PoolImmutables {
    /// @notice The contract that deployed the pool, which must adhere to the IUniswapV3Factory interface
    /// @return The contract address
    function factory() external view returns (address);

    /// @notice The first of the two tokens of the pool, sorted by address
    /// @return The token contract address
    function token0() external view returns (address);

    /// @notice The second of the two tokens of the pool, sorted by address
    /// @return The token contract address
    function token1() external view returns (address);

    /// @notice The pool's fee in hundredths of a bip, i.e. 1e-6
    /// @return The fee
    function fee() external view returns (uint24);

    /// @notice The pool tick spacing
    /// @dev Ticks can only be used at multiples of this value, minimum of 1 and always positive
    /// e.g.: a tickSpacing of 3 means ticks can be initialized every 3rd tick, i.e., ..., -6, -3, 0, 3, 6, ...
    /// This value is an int24 to avoid casting even though it is always positive.
    /// @return The tick spacing
    function tickSpacing() external view returns (int24);

    /// @notice The maximum amount of position liquidity that can use any tick in the range
    /// @dev This parameter is enforced per tick to prevent liquidity from overflowing a uint128 at any point, and
    /// also prevents out-of-range liquidity from being used to prevent adding in-range liquidity to a pool
    /// @return The max amount of liquidity per tick
    function maxLiquidityPerTick() external view returns (uint128);
}


// File contracts/Uniswap_V3/pool/IUniswapV3PoolOwnerActions.sol


/// @title Permissioned pool actions
/// @notice Contains pool methods that may only be called by the factory owner
interface IUniswapV3PoolOwnerActions {
    /// @notice Set the denominator of the protocol's % share of the fees
    /// @param feeProtocol0 new protocol fee for token0 of the pool
    /// @param feeProtocol1 new protocol fee for token1 of the pool
    function setFeeProtocol(uint8 feeProtocol0, uint8 feeProtocol1) external;

    /// @notice Collect the protocol fee accrued to the pool
    /// @param recipient The address to which collected protocol fees should be sent
    /// @param amount0Requested The maximum amount of token0 to send, can be 0 to collect fees in only token1
    /// @param amount1Requested The maximum amount of token1 to send, can be 0 to collect fees in only token0
    /// @return amount0 The protocol fee collected in token0
    /// @return amount1 The protocol fee collected in token1
    function collectProtocol(
        address recipient,
        uint128 amount0Requested,
        uint128 amount1Requested
    ) external returns (uint128 amount0, uint128 amount1);
}


// File contracts/Uniswap_V3/pool/IUniswapV3PoolState.sol


/// @title Pool state that can change
/// @notice These methods compose the pool's state, and can change with any frequency including multiple times
/// per transaction
interface IUniswapV3PoolState {
    /// @notice The 0th storage slot in the pool stores many values, and is exposed as a single method to save gas
    /// when accessed externally.
    /// @return sqrtPriceX96 The current price of the pool as a sqrt(token1/token0) Q64.96 value
    /// tick The current tick of the pool, i.e. according to the last tick transition that was run.
    /// This value may not always be equal to SqrtTickMath.getTickAtSqrtRatio(sqrtPriceX96) if the price is on a tick
    /// boundary.
    /// observationIndex The index of the last oracle observation that was written,
    /// observationCardinality The current maximum number of observations stored in the pool,
    /// observationCardinalityNext The next maximum number of observations, to be updated when the observation.
    /// feeProtocol The protocol fee for both tokens of the pool.
    /// Encoded as two 4 bit values, where the protocol fee of token1 is shifted 4 bits and the protocol fee of token0
    /// is the lower 4 bits. Used as the denominator of a fraction of the swap fee, e.g. 4 means 1/4th of the swap fee.
    /// unlocked Whether the pool is currently locked to reentrancy
    function slot0()
        external
        view
        returns (
            uint160 sqrtPriceX96,
            int24 tick,
            uint16 observationIndex,
            uint16 observationCardinality,
            uint16 observationCardinalityNext,
            uint8 feeProtocol,
            bool unlocked
        );

    /// @notice The fee growth as a Q128.128 fees of token0 collected per unit of liquidity for the entire life of the pool
    /// @dev This value can overflow the uint256
    function feeGrowthGlobal0X128() external view returns (uint256);

    /// @notice The fee growth as a Q128.128 fees of token1 collected per unit of liquidity for the entire life of the pool
    /// @dev This value can overflow the uint256
    function feeGrowthGlobal1X128() external view returns (uint256);

    /// @notice The amounts of token0 and token1 that are owed to the protocol
    /// @dev Protocol fees will never exceed uint128 max in either token
    function protocolFees() external view returns (uint128 token0, uint128 token1);

    /// @notice The currently in range liquidity available to the pool
    /// @dev This value has no relationship to the total liquidity across all ticks
    function liquidity() external view returns (uint128);

    /// @notice Look up information about a specific tick in the pool
    /// @param tick The tick to look up
    /// @return liquidityGross the total amount of position liquidity that uses the pool either as tick lower or
    /// tick upper,
    /// liquidityNet how much liquidity changes when the pool price crosses the tick,
    /// feeGrowthOutside0X128 the fee growth on the other side of the tick from the current tick in token0,
    /// feeGrowthOutside1X128 the fee growth on the other side of the tick from the current tick in token1,
    /// tickCumulativeOutside the cumulative tick value on the other side of the tick from the current tick
    /// secondsPerLiquidityOutsideX128 the seconds spent per liquidity on the other side of the tick from the current tick,
    /// secondsOutside the seconds spent on the other side of the tick from the current tick,
    /// initialized Set to true if the tick is initialized, i.e. liquidityGross is greater than 0, otherwise equal to false.
    /// Outside values can only be used if the tick is initialized, i.e. if liquidityGross is greater than 0.
    /// In addition, these values are only relative and must be used only in comparison to previous snapshots for
    /// a specific position.
    function ticks(int24 tick)
        external
        view
        returns (
            uint128 liquidityGross,
            int128 liquidityNet,
            uint256 feeGrowthOutside0X128,
            uint256 feeGrowthOutside1X128,
            int56 tickCumulativeOutside,
            uint160 secondsPerLiquidityOutsideX128,
            uint32 secondsOutside,
            bool initialized
        );

    /// @notice Returns 256 packed tick initialized boolean values. See TickBitmap for more information
    function tickBitmap(int16 wordPosition) external view returns (uint256);

    /// @notice Returns the information about a position by the position's key
    /// @param key The position's key is a hash of a preimage composed by the owner, tickLower and tickUpper
    /// @return _liquidity The amount of liquidity in the position,
    /// Returns feeGrowthInside0LastX128 fee growth of token0 inside the tick range as of the last mint/burn/poke,
    /// Returns feeGrowthInside1LastX128 fee growth of token1 inside the tick range as of the last mint/burn/poke,
    /// Returns tokensOwed0 the computed amount of token0 owed to the position as of the last mint/burn/poke,
    /// Returns tokensOwed1 the computed amount of token1 owed to the position as of the last mint/burn/poke
    function positions(bytes32 key)
        external
        view
        returns (
            uint128 _liquidity,
            uint256 feeGrowthInside0LastX128,
            uint256 feeGrowthInside1LastX128,
            uint128 tokensOwed0,
            uint128 tokensOwed1
        );

    /// @notice Returns data about a specific observation index
    /// @param index The element of the observations array to fetch
    /// @dev You most likely want to use #observe() instead of this method to get an observation as of some amount of time
    /// ago, rather than at a specific index in the array.
    /// @return blockTimestamp The timestamp of the observation,
    /// Returns tickCumulative the tick multiplied by seconds elapsed for the life of the pool as of the observation timestamp,
    /// Returns secondsPerLiquidityCumulativeX128 the seconds per in range liquidity for the life of the pool as of the observation timestamp,
    /// Returns initialized whether the observation has been initialized and the values are safe to use
    function observations(uint256 index)
        external
        view
        returns (
            uint32 blockTimestamp,
            int56 tickCumulative,
            uint160 secondsPerLiquidityCumulativeX128,
            bool initialized
        );
}


// File contracts/Uniswap_V3/IUniswapV3Pool.sol







/// @title The interface for a Uniswap V3 Pool
/// @notice A Uniswap pool facilitates swapping and automated market making between any two assets that strictly conform
/// to the ERC20 specification
/// @dev The pool interface is broken up into many smaller pieces
interface IUniswapV3Pool is
    IUniswapV3PoolImmutables,
    IUniswapV3PoolState,
    IUniswapV3PoolDerivedState,
    IUniswapV3PoolActions,
    IUniswapV3PoolOwnerActions,
    IUniswapV3PoolEvents
{

}


// File contracts/Misc_AMOs/bunni/IBunniLens.sol


interface IBunniLens {
    struct BunniKey {
        IUniswapV3Pool pool;
        int24 tickLower;
        int24 tickUpper;
    }

    function getReserves (BunniKey calldata key) external view returns (uint112 reserve0, uint112 reserve1);
    function hub () external view returns (address);
    function pricePerFullShare (BunniKey calldata key) external view returns (uint128 liquidity, uint256 amount0, uint256 amount1);
}


// File contracts/Curve/ICurvefrxETHETHPool.sol


interface ICurvefrxETHETHPool {
  function A() external view returns (uint256);
  function A_precise() external view returns (uint256);
  function get_p() external view returns (uint256);
  function price_oracle() external view returns (uint256);
  function get_virtual_price() external view returns (uint256);
  function calc_token_amount(uint256[2] memory _amounts, bool _is_deposit) external view returns (uint256);
  function add_liquidity(uint256[2] memory _amounts, uint256 _min_mint_amount) external returns (uint256);
  function get_dy(int128 i, int128 j, uint256 _dx) external view returns (uint256);
  function exchange(int128 i, int128 j, uint256 _dx, uint256 _min_dy) external payable returns (uint256);
  function remove_liquidity(uint256 _amount, uint256[2] memory _min_amounts) external returns (uint256[2] memory);
  function remove_liquidity_imbalance(uint256[2] memory _amounts, uint256 _max_burn_amount) external returns (uint256);
  function calc_withdraw_one_coin(uint256 _token_amount, int128 i) external view returns (uint256);
  function remove_liquidity_one_coin(uint256 _token_amount, int128 i, uint256 _min_amount) external returns (uint256);
  function ramp_A(uint256 _future_A, uint256 _future_time) external;
  function stop_ramp_A() external;
  function commit_new_fee(uint256 _new_fee, uint256 _new_admin_fee) external;
  function apply_new_fee() external;
  function revert_new_parameters() external;
  function set_ma_exp_time(uint256 _ma_exp_time) external;
  function commit_transfer_ownership(address _owner) external;
  function apply_transfer_ownership() external;
  function revert_transfer_ownership() external;
  function admin_balances(uint256 i) external view returns (uint256);
  function withdraw_admin_fees() external;
  function donate_admin_fees() external;
  function kill_me() external;
  function unkill_me() external;
  function coins(uint256 arg0) external view returns (address);
  function balances(uint256 arg0) external view returns (uint256);
  function fee() external view returns (uint256);
  function admin_fee() external view returns (uint256);
  function owner() external view returns (address);
  function lp_token() external view returns (address);
  function initial_A() external view returns (uint256);
  function future_A() external view returns (uint256);
  function initial_A_time() external view returns (uint256);
  function future_A_time() external view returns (uint256);
  function admin_actions_deadline() external view returns (uint256);
  function transfer_ownership_deadline() external view returns (uint256);
  function future_fee() external view returns (uint256);
  function future_admin_fee() external view returns (uint256);
  function future_owner() external view returns (address);
  function ma_exp_time() external view returns (uint256);
  function ma_last_time() external view returns (uint256);
}


// File contracts/Misc_AMOs/bunni/IBunniGauge.sol


interface IBunniGauge {
  function deposit ( uint256 _value ) external;
  function deposit ( uint256 _value, address _addr ) external;
  function deposit ( uint256 _value, address _addr, bool _claim_rewards ) external;
  function withdraw ( uint256 _value ) external;
  function withdraw ( uint256 _value, bool _claim_rewards ) external;
  function claim_rewards (  ) external;
  function claim_rewards ( address _addr ) external;
  function claim_rewards ( address _addr, address _receiver ) external;
  function transferFrom ( address _from, address _to, uint256 _value ) external returns ( bool );
  function transfer ( address _to, uint256 _value ) external returns ( bool );
  function approve ( address _spender, uint256 _value ) external returns ( bool );
  function permit ( address _owner, address _spender, uint256 _value, uint256 _deadline, uint8 _v, bytes32 _r, bytes32 _s ) external returns ( bool );
  function increaseAllowance ( address _spender, uint256 _added_value ) external returns ( bool );
  function decreaseAllowance ( address _spender, uint256 _subtracted_value ) external returns ( bool );
  function user_checkpoint ( address addr ) external returns ( bool );
  function set_rewards_receiver ( address _receiver ) external;
  function kick ( address addr ) external;
  function deposit_reward_token ( address _reward_token, uint256 _amount ) external;
  function add_reward ( address _reward_token, address _distributor ) external;
  function set_reward_distributor ( address _reward_token, address _distributor ) external;
  function makeGaugePermissionless (  ) external;
  function killGauge (  ) external;
  function unkillGauge (  ) external;
  function change_pending_admin ( address new_pending_admin ) external;
  function claim_admin (  ) external;
  function set_tokenless_production ( uint8 new_tokenless_production ) external;
  function claimed_reward ( address _addr, address _token ) external view returns ( uint256 );
  function claimable_reward ( address _user, address _reward_token ) external view returns ( uint256 );
  function claimable_tokens ( address addr ) external returns ( uint256 );
  function integrate_checkpoint (  ) external view returns ( uint256 );
  function future_epoch_time (  ) external view returns ( uint256 );
  function inflation_rate (  ) external view returns ( uint256 );
  function decimals (  ) external view returns ( uint256 );
  function version (  ) external view returns ( string memory );
  function allowance ( address owner, address spender ) external view returns ( uint256 );
  function is_killed (  ) external view returns ( bool );
  function initialize ( address _lp_token, uint256 relative_weight_cap, address _voting_escrow_delegation, address _admin, bytes32 _position_key ) external;
  function setRelativeWeightCap ( uint256 relative_weight_cap ) external;
  function getRelativeWeightCap (  ) external view returns ( uint256 );
  function getCappedRelativeWeight ( uint256 time ) external view returns ( uint256 );
  function getMaxRelativeWeightCap (  ) external pure returns ( uint256 );
  function tokenless_production (  ) external view returns ( uint8 );
  function pending_admin (  ) external view returns ( address );
  function admin (  ) external view returns ( address );
  function voting_escrow_delegation (  ) external view returns ( address );
  function balanceOf ( address arg0 ) external view returns ( uint256 );
  function totalSupply (  ) external view returns ( uint256 );
  function name (  ) external view returns ( string memory );
  function symbol (  ) external view returns ( string memory );
  function DOMAIN_SEPARATOR (  ) external view returns ( bytes32 );
  function nonces ( address arg0 ) external view returns ( uint256 );
  function lp_token (  ) external view returns ( address );
  function gauge_state (  ) external view returns ( uint8 );
  function position_key (  ) external view returns ( bytes32 );
  function reward_count (  ) external view returns ( uint256 );
//   function reward_data ( address arg0 ) external view returns ( tuple );
  function rewards_receiver ( address arg0 ) external view returns ( address );
  function reward_integral_for ( address arg0, address arg1 ) external view returns ( uint256 );
  function working_balances ( address arg0 ) external view returns ( uint256 );
  function working_supply (  ) external view returns ( uint256 );
  function integrate_inv_supply_of ( address arg0 ) external view returns ( uint256 );
  function integrate_checkpoint_of ( address arg0 ) external view returns ( uint256 );
  function integrate_fraction ( address arg0 ) external view returns ( uint256 );
  function period (  ) external view returns ( int128 );
  function reward_tokens ( uint256 arg0 ) external view returns ( address );
  function period_timestamp ( uint256 arg0 ) external view returns ( uint256 );
  function integrate_inv_supply ( uint256 arg0 ) external view returns ( uint256 );
}


// File contracts/Misc_AMOs/convex/IConvexStakingWrapperFrax.sol


interface IConvexStakingWrapperFrax {
  function addRewards() external;
  function allowance(address owner, address spender) external view returns (uint256);
  function approve(address spender, uint256 amount) external returns (bool);
  function balanceOf(address account) external view returns (uint256);
  function collateralVault() external view returns (address);
  function convexBooster() external view returns (address);
  function convexPool() external view returns (address);
  function convexPoolId() external view returns (uint256);
  function convexToken() external view returns (address);
  function crv() external view returns (address);
  function curveToken() external view returns (address);
  function cvx() external view returns (address);
  function decimals() external view returns (uint8);
  function decreaseAllowance(address spender, uint256 subtractedValue) external returns (bool);
  function deposit(uint256 _amount, address _to) external;
//   function earned(address _account) external view returns (tuple[] memory claimable);
  function getReward(address _account, address _forwardTo) external;
  function getReward(address _account) external;
  function increaseAllowance(address spender, uint256 addedValue) external returns (bool);
  function initialize(address _curveToken, address _convexToken, address _convexPool, uint256 _poolId, address _vault) external;
  function isInit() external view returns (bool);
  function isShutdown() external view returns (bool);
  function name() external view returns (string memory);
  function owner() external view returns (address);
  function registeredRewards(address) external view returns (uint256);
  function renounceOwnership() external;
  function rewardLength() external view returns (uint256);
  function rewards(uint256) external view returns (address reward_token, address reward_pool, uint128 reward_integral, uint128 reward_remaining);
  function setApprovals() external;
  function setVault(address _vault) external;
  function shutdown() external;
  function stake(uint256 _amount, address _to) external;
  function symbol() external view returns (string memory);
  function totalBalanceOf(address _account) external view returns (uint256);
  function totalSupply() external view returns (uint256);
  function transfer(address recipient, uint256 amount) external returns (bool);
  function transferFrom(address sender, address recipient, uint256 amount) external returns (bool);
  function transferOwnership(address newOwner) external;
  function user_checkpoint(address _account) external returns (bool);
  function user_checkpoint(address[2] memory _accounts) external returns (bool);
  function withdraw(uint256 _amount) external;
  function withdrawAndUnwrap(uint256 _amount) external;
}


// File contracts/Misc_AMOs/convex/IDepositToken.sol


interface IDepositToken {
  function allowance(address owner, address spender) external view returns (uint256);
  function approve(address spender, uint256 amount) external returns (bool);
  function balanceOf(address account) external view returns (uint256);
  function burn(address _from, uint256 _amount) external;
  function decimals() external view returns (uint8);
  function decreaseAllowance(address spender, uint256 subtractedValue) external returns (bool);
  function increaseAllowance(address spender, uint256 addedValue) external returns (bool);
  function mint(address _to, uint256 _amount) external;
  function name() external view returns (string memory);
  function operator() external view returns (address);
  function symbol() external view returns (string memory);
  function totalSupply() external view returns (uint256);
  function transfer(address recipient, uint256 amount) external returns (bool);
  function transferFrom(address sender, address recipient, uint256 amount) external returns (bool);
}


// File contracts/Misc_AMOs/curve/I2pool.sol


interface I2pool {
    function decimals() external view returns (uint256);
    function transfer(address _to, uint256 _value) external returns (bool);
    function transferFrom(address _from, address _to, uint256 _value) external returns (bool);
    function approve(address _spender, uint256 _value) external returns (bool);
    function A() external view returns (uint256);
    function A_precise() external view returns (uint256);
    function get_virtual_price() external view returns (uint256);
    function lp_price() external view returns (uint256);
    function price_oracle() external view returns (uint256);
    function calc_token_amount(uint256[2] memory _amounts, bool _is_deposit) external view returns (uint256);
    function add_liquidity(uint256[2] memory _amounts, uint256 _min_mint_amount) external returns (uint256);
    function get_dy(int128 i, int128 j, uint256 _dx) external view returns (uint256);
    function exchange(int128 i, int128 j, uint256 _dx, uint256 _min_dy) external returns (uint256);
    function remove_liquidity(uint256 _amount, uint256[2] memory _min_amounts) external returns (uint256[2] memory);
    function remove_liquidity_imbalance(uint256[2] memory _amounts, uint256 _max_burn_amount) external returns (uint256);
    function calc_withdraw_one_coin(uint256 _token_amount, int128 i) external view returns (uint256);
    function remove_liquidity_one_coin(uint256 _token_amount, int128 i, uint256 _min_amount) external returns (uint256);
    function ramp_A(uint256 _future_A, uint256 _future_time) external;
    function stop_ramp_A() external;
    function commit_new_fee(uint256 _new_fee, uint256 _new_admin_fee) external;
    function apply_new_fee() external;
    function revert_new_parameters() external;
    function commit_transfer_ownership(address _owner) external;
    function apply_transfer_ownership() external;
    function revert_transfer_ownership() external;
    function admin_balances(uint256 i) external view returns (uint256);
    function withdraw_admin_fees() external;
    function donate_admin_fees() external;
    function kill_me() external;
    function unkill_me() external;
    function coins(uint256 arg0) external view returns (address);
    function balances(uint256 arg0) external view returns (uint256);
    function fee() external view returns (uint256);
    function admin_fee() external view returns (uint256);
    function owner() external view returns (address);
    function initial_A() external view returns (uint256);
    function future_A() external view returns (uint256);
    function initial_A_time() external view returns (uint256);
    function future_A_time() external view returns (uint256);
    function admin_actions_deadline() external view returns (uint256);
    function transfer_ownership_deadline() external view returns (uint256);
    function future_fee() external view returns (uint256);
    function future_admin_fee() external view returns (uint256);
    function future_owner() external view returns (address);
    function name() external view returns (string memory);
    function symbol() external view returns (string memory);
    function balanceOf(address arg0) external view returns (uint256);
    function allowance(address arg0, address arg1) external view returns (uint256);
    function totalSupply() external view returns (uint256);
}


// File contracts/Misc_AMOs/curve/I2poolToken.sol


interface I2poolToken {
  function decimals() external view returns (uint256);
  function transfer(address _to, uint256 _value) external returns (bool);
  function transferFrom(address _from, address _to, uint256 _value) external returns (bool);
  function approve(address _spender, uint256 _value) external returns (bool);
  function increaseAllowance(address _spender, uint256 _added_value) external returns (bool);
  function decreaseAllowance(address _spender, uint256 _subtracted_value) external returns (bool);
  function mint(address _to, uint256 _value) external returns (bool);
  function burnFrom(address _to, uint256 _value) external returns (bool);
  function set_minter(address _minter) external;
  function set_name(string memory _name, string memory _symbol) external;
  function name() external view returns (string memory);
  function symbol() external view returns (string memory);
  function balanceOf(address arg0) external view returns (uint256);
  function allowance(address arg0, address arg1) external view returns (uint256);
  function totalSupply() external view returns (uint256);
  function minter() external view returns (address);
}


// File contracts/Curve/IFraxGaugeController.sol


// https://github.com/swervefi/swerve/edit/master/packages/swerve-contracts/interfaces/IGaugeController.sol

interface IFraxGaugeController {
    struct Point {
        uint256 bias;
        uint256 slope;
    }

    struct VotedSlope {
        uint256 slope;
        uint256 power;
        uint256 end;
    }

    // Public variables
    function admin() external view returns (address);
    function future_admin() external view returns (address);
    function token() external view returns (address);
    function voting_escrow() external view returns (address);
    function n_gauge_types() external view returns (int128);
    function n_gauges() external view returns (int128);
    function gauge_type_names(int128) external view returns (string memory);
    function gauges(uint256) external view returns (address);
    function vote_user_slopes(address, address)
        external
        view
        returns (VotedSlope memory);
    function vote_user_power(address) external view returns (uint256);
    function last_user_vote(address, address) external view returns (uint256);
    function points_weight(address, uint256)
        external
        view
        returns (Point memory);
    function time_weight(address) external view returns (uint256);
    function points_sum(int128, uint256) external view returns (Point memory);
    function time_sum(uint256) external view returns (uint256);
    function points_total(uint256) external view returns (uint256);
    function time_total() external view returns (uint256);
    function points_type_weight(int128, uint256)
        external
        view
        returns (uint256);
    function time_type_weight(uint256) external view returns (uint256);

    // Getter functions
    function gauge_types(address) external view returns (int128);
    function gauge_relative_weight(address) external view returns (uint256);
    function gauge_relative_weight(address, uint256) external view returns (uint256);
    function get_gauge_weight(address) external view returns (uint256);
    function get_type_weight(int128) external view returns (uint256);
    function get_total_weight() external view returns (uint256);
    function get_weights_sum_per_type(int128) external view returns (uint256);

    // External functions
    function commit_transfer_ownership(address) external;
    function apply_transfer_ownership() external;
    function add_gauge(
        address,
        int128,
        uint256
    ) external;
    function checkpoint() external;
    function checkpoint_gauge(address) external;
    function global_emission_rate() external view returns (uint256);
    function gauge_relative_weight_write(address)
        external
        returns (uint256);
    function gauge_relative_weight_write(address, uint256)
        external
        returns (uint256);
    function add_type(string memory, uint256) external;
    function change_type_weight(int128, uint256) external;
    function change_gauge_weight(address, uint256) external;
    function change_global_emission_rate(uint256) external;
    function vote_for_gauge_weights(address, uint256) external;
}


// File contracts/Curve/IFraxGaugeFXSRewardsDistributor.sol


interface IFraxGaugeFXSRewardsDistributor {
  function acceptOwnership() external;
  function curator_address() external view returns(address);
  function currentReward(address gauge_address) external view returns(uint256 reward_amount);
  function distributeReward(address gauge_address) external returns(uint256 weeks_elapsed, uint256 reward_tally);
  function distributionsOn() external view returns(bool);
  function gauge_whitelist(address) external view returns(bool);
  function is_middleman(address) external view returns(bool);
  function last_time_gauge_paid(address) external view returns(uint256);
  function nominateNewOwner(address _owner) external;
  function nominatedOwner() external view returns(address);
  function owner() external view returns(address);
  function recoverERC20(address tokenAddress, uint256 tokenAmount) external;
  function setCurator(address _new_curator_address) external;
  function setGaugeController(address _gauge_controller_address) external;
  function setGaugeState(address _gauge_address, bool _is_middleman, bool _is_active) external;
  function setTimelock(address _new_timelock) external;
  function timelock_address() external view returns(address);
  function toggleDistributions() external;
}


// File contracts/Curve/IveFXS.sol


interface IveFXS {

    struct LockedBalance {
        int128 amount;
        uint256 end;
    }

    function commit_transfer_ownership(address addr) external;
    function apply_transfer_ownership() external;
    function commit_smart_wallet_checker(address addr) external;
    function apply_smart_wallet_checker() external;
    function toggleEmergencyUnlock() external;
    function recoverERC20(address token_addr, uint256 amount) external;
    function get_last_user_slope(address addr) external view returns (int128);
    function user_point_history__ts(address _addr, uint256 _idx) external view returns (uint256);
    function locked__end(address _addr) external view returns (uint256);
    function checkpoint() external;
    function deposit_for(address _addr, uint256 _value) external;
    function create_lock(uint256 _value, uint256 _unlock_time) external;
    function increase_amount(uint256 _value) external;
    function increase_unlock_time(uint256 _unlock_time) external;
    function withdraw() external;
    function balanceOf(address addr) external view returns (uint256);
    function balanceOf(address addr, uint256 _t) external view returns (uint256);
    function balanceOfAt(address addr, uint256 _block) external view returns (uint256);
    function totalSupply() external view returns (uint256);
    function totalSupply(uint256 t) external view returns (uint256);
    function totalSupplyAt(uint256 _block) external view returns (uint256);
    function totalFXSSupply() external view returns (uint256);
    function totalFXSSupplyAt(uint256 _block) external view returns (uint256);
    function changeController(address _newController) external;
    function token() external view returns (address);
    function supply() external view returns (uint256);
    function locked(address addr) external view returns (LockedBalance memory);
    function epoch() external view returns (uint256);
    function point_history(uint256 arg0) external view returns (int128 bias, int128 slope, uint256 ts, uint256 blk, uint256 fxs_amt);
    function user_point_history(address arg0, uint256 arg1) external view returns (int128 bias, int128 slope, uint256 ts, uint256 blk, uint256 fxs_amt);
    function user_point_epoch(address arg0) external view returns (uint256);
    function slope_changes(uint256 arg0) external view returns (int128);
    function controller() external view returns (address);
    function transfersEnabled() external view returns (bool);
    function emergencyUnlockActive() external view returns (bool);
    function name() external view returns (string memory);
    function symbol() external view returns (string memory);
    function version() external view returns (string memory);
    function decimals() external view returns (uint256);
    function future_smart_wallet_checker() external view returns (address);
    function smart_wallet_checker() external view returns (address);
    function admin() external view returns (address);
    function future_admin() external view returns (address);
}


// File contracts/Math/Math.sol


/**
 * @dev Standard math utilities missing in the Solidity language.
 */
library Math {
    /**
     * @dev Returns the largest of two numbers.
     */
    function max(uint256 a, uint256 b) internal pure returns (uint256) {
        return a >= b ? a : b;
    }

    /**
     * @dev Returns the smallest of two numbers.
     */
    function min(uint256 a, uint256 b) internal pure returns (uint256) {
        return a < b ? a : b;
    }

    /**
     * @dev Returns the average of two numbers. The result is rounded towards
     * zero.
     */
    function average(uint256 a, uint256 b) internal pure returns (uint256) {
        // (a + b) / 2 can overflow, so we distribute
        return (a / 2) + (b / 2) + ((a % 2 + b % 2) / 2);
    }

    // babylonian method (https://en.wikipedia.org/wiki/Methods_of_computing_square_roots#Babylonian_method)
    function sqrt(uint y) internal pure returns (uint z) {
        if (y > 3) {
            z = y;
            uint x = y / 2 + 1;
            while (x < z) {
                z = x;
                x = (y / x + x) / 2;
            }
        } else if (y != 0) {
            z = 1;
        }
    }
}


// File contracts/Misc_AMOs/bunni/IBunniMinter.sol


interface IBunniMinter {
  function allowed_to_mint_for ( address minter, address user ) external view returns ( bool );
  function getGaugeController (  ) external view returns ( address );
  function getMinterApproval ( address minter, address user ) external view returns ( bool );
  function getToken (  ) external view returns ( address );
  function getTokenAdmin (  ) external view returns ( address );
  function mint ( address gauge ) external returns ( uint256 );
  function mintFor ( address gauge, address user ) external returns ( uint256 );
  function mintMany ( address[] memory gauges ) external returns ( uint256 );
  function mintManyFor ( address[] memory gauges, address user ) external returns ( uint256 );
  function mint_for ( address gauge, address user ) external;
  function mint_many ( address[8] memory gauges ) external;
  function minted ( address user, address gauge ) external view returns ( uint256 );
  function setMinterApproval ( address minter, bool approval ) external;
  function toggle_approve_mint ( address minter ) external;
}


// File contracts/Staking/Owned.sol


// https://docs.synthetix.io/contracts/Owned
contract Owned {
    address public owner;
    address public nominatedOwner;

    constructor (address _owner) public {
        require(_owner != address(0), "Owner address cannot be 0");
        owner = _owner;
        emit OwnerChanged(address(0), _owner);
    }

    function nominateNewOwner(address _owner) external onlyOwner {
        nominatedOwner = _owner;
        emit OwnerNominated(_owner);
    }

    function acceptOwnership() external {
        require(msg.sender == nominatedOwner, "You must be nominated before you can accept ownership");
        emit OwnerChanged(owner, nominatedOwner);
        owner = nominatedOwner;
        nominatedOwner = address(0);
    }

    modifier onlyOwner {
        require(msg.sender == owner, "Only the contract owner may perform this action");
        _;
    }

    event OwnerNominated(address newOwner);
    event OwnerChanged(address oldOwner, address newOwner);
}


// File contracts/Uniswap/TransferHelper.sol


// helper methods for interacting with ERC20 tokens and sending ETH that do not consistently return true/false
library TransferHelper {
    function safeApprove(address token, address to, uint value) internal {
        // bytes4(keccak256(bytes('approve(address,uint256)')));
        (bool success, bytes memory data) = token.call(abi.encodeWithSelector(0x095ea7b3, to, value));
        require(success && (data.length == 0 || abi.decode(data, (bool))), 'TransferHelper: APPROVE_FAILED');
    }

    function safeTransfer(address token, address to, uint value) internal {
        // bytes4(keccak256(bytes('transfer(address,uint256)')));
        (bool success, bytes memory data) = token.call(abi.encodeWithSelector(0xa9059cbb, to, value));
        require(success && (data.length == 0 || abi.decode(data, (bool))), 'TransferHelper: TRANSFER_FAILED');
    }

    function safeTransferFrom(address token, address from, address to, uint value) internal {
        // bytes4(keccak256(bytes('transferFrom(address,address,uint256)')));
        (bool success, bytes memory data) = token.call(abi.encodeWithSelector(0x23b872dd, from, to, value));
        require(success && (data.length == 0 || abi.decode(data, (bool))), 'TransferHelper: TRANSFER_FROM_FAILED');
    }

    function safeTransferETH(address to, uint value) internal {
        (bool success,) = to.call{value:value}(new bytes(0));
        require(success, 'TransferHelper: ETH_TRANSFER_FAILED');
    }
}


// File contracts/Utils/ReentrancyGuard.sol


/**
 * @dev Contract module that helps prevent reentrant calls to a function.
 *
 * Inheriting from `ReentrancyGuard` will make the {nonReentrant} modifier
 * available, which can be applied to functions to make sure there are no nested
 * (reentrant) calls to them.
 *
 * Note that because there is a single `nonReentrant` guard, functions marked as
 * `nonReentrant` may not call one another. This can be worked around by making
 * those functions `private`, and then adding `external` `nonReentrant` entry
 * points to them.
 *
 * TIP: If you would like to learn more about reentrancy and alternative ways
 * to protect against it, check out our blog post
 * https://blog.openzeppelin.com/reentrancy-after-istanbul/[Reentrancy After Istanbul].
 */
abstract contract ReentrancyGuard {
    // Booleans are more expensive than uint256 or any type that takes up a full
    // word because each write operation emits an extra SLOAD to first read the
    // slot's contents, replace the bits taken up by the boolean, and then write
    // back. This is the compiler's defense against contract upgrades and
    // pointer aliasing, and it cannot be disabled.

    // The values being non-zero value makes deployment a bit more expensive,
    // but in exchange the refund on every call to nonReentrant will be lower in
    // amount. Since refunds are capped to a percentage of the total
    // transaction's gas, it is best to keep them low in cases like this one, to
    // increase the likelihood of the full refund coming into effect.
    uint256 private constant _NOT_ENTERED = 1;
    uint256 private constant _ENTERED = 2;

    uint256 private _status;

    constructor () internal {
        _status = _NOT_ENTERED;
    }

    /**
     * @dev Prevents a contract from calling itself, directly or indirectly.
     * Calling a `nonReentrant` function from another `nonReentrant`
     * function is not supported. It is possible to prevent this from happening
     * by making the `nonReentrant` function external, and make it call a
     * `private` function that does the actual work.
     */
    modifier nonReentrant() {
        // On the first call to nonReentrant, _notEntered will be true
        require(_status != _ENTERED, "ReentrancyGuard: reentrant call");

        // Any calls to nonReentrant after this point will fail
        _status = _ENTERED;

        _;

        // By storing the original value once again, a refund is triggered (see
        // https://eips.ethereum.org/EIPS/eip-2200)
        _status = _NOT_ENTERED;
    }
}


// File contracts/Staking/FraxUnifiedFarmTemplate.sol


// ====================================================================
// |     ______                   _______                             |
// |    / _____________ __  __   / ____(_____  ____ _____  ________   |
// |   / /_  / ___/ __ `| |/_/  / /_  / / __ \/ __ `/ __ \/ ___/ _ \  |
// |  / __/ / /  / /_/ _>  <   / __/ / / / / / /_/ / / / / /__/  __/  |
// | /_/   /_/   \__,_/_/|_|  /_/   /_/_/ /_/\__,_/_/ /_/\___/\___/   |
// |                                                                  |
// ====================================================================
// ====================== FraxUnifiedFarmTemplate =====================
// ====================================================================
// Farming contract that accounts for veFXS
// Overrideable for UniV3, ERC20s, etc
// New for V2
//      - Multiple reward tokens possible
//      - Can add to existing locked stakes
//      - Contract is aware of proxied veFXS
//      - veFXS multiplier formula changed
// Apes together strong

// Frax Finance: https://github.com/FraxFinance

// Primary Author(s)
// Travis Moore: https://github.com/FortisFortuna

// Reviewer(s) / Contributor(s)
// Jason Huan: https://github.com/jasonhuan
// Sam Kazemian: https://github.com/samkazemian
// Dennis: github.com/denett

// Originally inspired by Synthetix.io, but heavily modified by the Frax team
// (Locked, veFXS, and UniV3 portions are new)
// https://raw.githubusercontent.com/Synthetixio/synthetix/develop/contracts/StakingRewards.sol








// Extra rewards



// import "../Misc_AMOs/convex/IConvexBaseRewardPool.sol";

contract FraxUnifiedFarmTemplate is Owned, ReentrancyGuard {


    // -------------------- VARIES --------------------

    // // Bunni
    // IBunniGauge public stakingToken;
    // IBunniLens public lens = IBunniLens(0xb73F303472C4fD4FF3B9f59ce0F9b13E47fbfD19);
    // IBunniMinter public minter = IBunniMinter(0xF087521Ffca0Fa8A43F5C445773aB37C5f574DA0);

    /* ========== STATE VARIABLES ========== */

    // Instances
    IveFXS private constant veFXS = IveFXS(0xc8418aF6358FFddA74e09Ca9CC3Fe03Ca6aDC5b0);
    
    // Frax related
    address internal constant frax_address = 0x853d955aCEf822Db058eb8505911ED77F175b99e;
    /// @notice fraxPerLPToken is a public view function, although doesn't show the stored value
    uint256 public fraxPerLPStored;

    // Constant for various precisions
    uint256 internal constant MULTIPLIER_PRECISION = 1e18;

    // Time tracking
    /// @notice Ending timestamp for the current period
    uint256 public periodFinish;
    /// @notice Timestamp of the last update - when this period started
    uint256 public lastUpdateTime;

    // Lock time and multiplier settings
    uint256 public lock_max_multiplier = uint256(2e18); // E18. 1x = e18
    uint256 public lock_time_for_max_multiplier = 1 * 1095 * 86400; // 3 years
    // uint256 public lock_time_for_max_multiplier = 2 * 86400; // 2 days
    uint256 public lock_time_min = 594000; // 6.875 * 86400 (~7 day)

    // veFXS related
    uint256 public vefxs_boost_scale_factor = uint256(4e18); // E18. 4x = 4e18; 100 / scale_factor = % vefxs supply needed for max boost
    uint256 public vefxs_max_multiplier = uint256(2e18); // E18. 1x = 1e18
    uint256 public vefxs_per_frax_for_max_boost = uint256(4e18); // E18. 2e18 means 2 veFXS must be held by the staker per 1 FRAX
    mapping(address => uint256) internal _vefxsMultiplierStored;
    mapping(address => bool) internal valid_vefxs_proxies;
    mapping(address => mapping(address => bool)) internal proxy_allowed_stakers;

    // Reward addresses, gauge addresses, reward rates, and reward managers
    /// @notice token addr -> manager addr
    mapping(address => address) public rewardManagers; 
    address[] internal rewardTokens;
    address[] internal gaugeControllers;
    address[] internal rewardDistributors;
    uint256[] internal rewardRatesManual;
    mapping(address => bool) internal isRewardToken;
    /// @notice token addr -> token index
    mapping(address => uint256) public rewardTokenAddrToIdx;
    
    // Reward period
    uint256 public constant rewardsDuration = 604800; // 7 * 86400  (7 days)

    // Reward tracking
    uint256[] private rewardsPerTokenStored;
    mapping(address => mapping(uint256 => uint256)) private userRewardsPerTokenPaid; // staker addr -> token id -> paid amount
    mapping(address => mapping(uint256 => uint256)) private rewards; // staker addr -> token id -> reward amount
    mapping(address => uint256) public lastRewardClaimTime; // staker addr -> timestamp
    
    // Gauge tracking
    uint256[] private last_gauge_relative_weights;
    uint256[] private last_gauge_time_totals;

    // Balance tracking
    uint256 internal _total_liquidity_locked;
    uint256 internal _total_combined_weight;
    mapping(address => uint256) internal _locked_liquidity;
    mapping(address => uint256) internal _combined_weights;
    /// @notice Keeps track of LP balances proxy-wide. Needed to make sure the proxy boost is kept in line
    mapping(address => uint256) public proxy_lp_balances; 


    /// @notice Stakers set which proxy(s) they want to use
    /// @dev Keep public so users can see on the frontend if they have a proxy
    mapping(address => address) public staker_designated_proxies;

    // Admin booleans for emergencies and overrides
    bool public stakesUnlocked; // Release locked stakes in case of emergency
    bool internal withdrawalsPaused; // For emergencies
    bool internal rewardsCollectionPaused; // For emergencies
    bool internal stakingPaused; // For emergencies
    bool internal collectRewardsOnWithdrawalPaused; // For emergencies if a token is overemitted

    /* ========== STRUCTS ========== */
    // In children...


    /* ========== MODIFIERS ========== */

    modifier onlyByOwnGov() {
        require(msg.sender == owner || msg.sender == 0x8412ebf45bAC1B340BbE8F318b928C466c4E39CA, "Not owner or timelock");
        _;
    }

    modifier onlyTknMgrs(address reward_token_address) {
        require(msg.sender == owner || isTokenManagerFor(msg.sender, reward_token_address), "Not owner or tkn mgr");
        _;
    }

    modifier updateRewardAndBalanceMdf(address account, bool sync_too) {
        _updateRewardAndBalance(account, sync_too, false);
        _;
    }

    /* ========== CONSTRUCTOR ========== */

    constructor (
        address _owner,
        address[] memory _rewardTokens,
        address[] memory _rewardManagers,
        uint256[] memory _rewardRatesManual,
        address[] memory _gaugeControllers,
        address[] memory _rewardDistributors
    ) Owned(_owner) {

        // Address arrays
        rewardTokens = _rewardTokens;
        gaugeControllers = _gaugeControllers;
        rewardDistributors = _rewardDistributors;
        rewardRatesManual = _rewardRatesManual;

        for (uint256 i = 0; i < _rewardTokens.length; i++){ 
            // For fast token address -> token ID lookups later
            rewardTokenAddrToIdx[_rewardTokens[i]] = i;

            // Add to the mapping
            isRewardToken[_rewardTokens[i]] = true;

            // Initialize the stored rewards
            rewardsPerTokenStored.push(0);

            // Initialize the reward managers
            rewardManagers[_rewardTokens[i]] = _rewardManagers[i];

            // Push in empty relative weights to initialize the array
            last_gauge_relative_weights.push(0);

            // Push in empty time totals to initialize the array
            last_gauge_time_totals.push(0);
        }

        // Other booleans
        stakesUnlocked = false;

        // Initialization
        lastUpdateTime = block.timestamp;

        // Sync the first period finish here with the gauge's 
        // periodFinish = IFraxGaugeController(gaugeControllers[0]).time_total();
        periodFinish = IFraxGaugeController(0x3669C421b77340B2979d1A00a792CC2ee0FcE737).time_total();
        
    }

    /* ============= VIEWS ============= */

    // ------ REWARD RELATED ------

    /// @notice Checks if the caller is a manager for the reward token
    /// @param caller_addr The address of the caller
    /// @param reward_token_addr The address of the reward token
    /// @return bool True if the caller is a manager for the reward token
    function isTokenManagerFor(address caller_addr, address reward_token_addr) public view returns (bool){
        if (!isRewardToken[reward_token_addr]) return false;
        else if (caller_addr == address(0) || reward_token_addr == address(0)) return false;
        else if (caller_addr == owner) return true; // Contract owner
        else if (rewardManagers[reward_token_addr] == caller_addr) return true; // Reward manager
        return false; 
    }

    /// @notice Gets all the reward tokens this contract handles
    /// @return rewardTokens_ The reward tokens array
    function getAllRewardTokens() external view returns (address[] memory) {
        return rewardTokens;
    }

    // Last time the reward was applicable
    function lastTimeRewardApplicable() internal view returns (uint256) {
        return Math.min(block.timestamp, periodFinish);
    }

    /// @notice The amount of reward tokens being paid out per second this period
    /// @param token_idx The index of the reward token
    /// @return rwd_rate The reward rate
    function rewardRates(uint256 token_idx) public view returns (uint256 rwd_rate) {
        address gauge_controller_address = gaugeControllers[token_idx];
        if (gauge_controller_address != address(0)) {
            rwd_rate = (IFraxGaugeController(gauge_controller_address).global_emission_rate() * last_gauge_relative_weights[token_idx]) / 1e18;
        }
        else {
            rwd_rate = rewardRatesManual[token_idx];
        }
    }

    // Amount of reward tokens per LP token / liquidity unit
    function rewardsPerToken() public view returns (uint256[] memory newRewardsPerTokenStored) {
        if (_total_liquidity_locked == 0 || _total_combined_weight == 0) {
            return rewardsPerTokenStored;
        }
        else {
            newRewardsPerTokenStored = new uint256[](rewardTokens.length);
            for (uint256 i = 0; i < rewardsPerTokenStored.length; i++){ 
                newRewardsPerTokenStored[i] = rewardsPerTokenStored[i] + (
                    ((lastTimeRewardApplicable() - lastUpdateTime) * rewardRates(i) * 1e18) / _total_combined_weight
                );
            }
            return newRewardsPerTokenStored;
        }
    }

    /// @notice The amount of reward tokens an account has earned / accrued
    /// @dev In the edge-case of one of the account's stake expiring since the last claim, this will
    /// @param account The account to check
    /// @return new_earned Array of reward token amounts earned by the account
    function earned(address account) public view returns (uint256[] memory new_earned) {
        uint256[] memory reward_arr = rewardsPerToken();
        new_earned = new uint256[](rewardTokens.length);

        if (_combined_weights[account] > 0){
            for (uint256 i = 0; i < rewardTokens.length; i++){ 
                new_earned[i] = ((_combined_weights[account] * (reward_arr[i] - userRewardsPerTokenPaid[account][i])) / 1e18)
                                + rewards[account][i];
            }
        }
    }

    /// @notice The total reward tokens emitted in the given period
    /// @return rewards_per_duration_arr Array of reward token amounts emitted in the current period
    function getRewardForDuration() external view returns (uint256[] memory rewards_per_duration_arr) {
        rewards_per_duration_arr = new uint256[](rewardRatesManual.length);

        for (uint256 i = 0; i < rewardRatesManual.length; i++){ 
            rewards_per_duration_arr[i] = rewardRates(i) * rewardsDuration;
        }
    }


    // ------ LIQUIDITY AND WEIGHTS ------

    /// @notice The farm's total locked liquidity / LP tokens
    /// @return The total locked liquidity
    function totalLiquidityLocked() external view returns (uint256) {
        return _total_liquidity_locked;
    }

    /// @notice A user's locked liquidity / LP tokens
    /// @param account The address of the account
    /// @return The locked liquidity
    function lockedLiquidityOf(address account) external view returns (uint256) {
        return _locked_liquidity[account];
    }

    /// @notice The farm's total combined weight of all users
    /// @return The total combined weight
    function totalCombinedWeight() external view returns (uint256) {
        return _total_combined_weight;
    }

    /// @notice Total 'balance' used for calculating the percent of the pool the account owns
    /// @notice Takes into account the locked stake time multiplier and veFXS multiplier
    /// @param account The address of the account
    /// @return The combined weight
    function combinedWeightOf(address account) external view returns (uint256) {
        return _combined_weights[account];
    }

    /// @notice Calculates the combined weight for an account
    /// @notice Must be overriden by the child contract
    /// @dev account The address of the account
    function calcCurCombinedWeight(address account) public virtual view 
        returns (
            uint256 old_combined_weight,
            uint256 new_vefxs_multiplier,
            uint256 new_combined_weight
        )
    {
        revert("Need cCCW logic");
    }

    // ------ LOCK RELATED ------

    /// @notice Reads the lock boost multiplier for a given duration
    /// @param secs The duration of the lock in seconds
    /// @return The multiplier amount
    function lockMultiplier(uint256 secs) public view returns (uint256) {
        return Math.min(
            lock_max_multiplier,
            (secs * lock_max_multiplier) / lock_time_for_max_multiplier
        ) ;
    }

    // ------ FRAX RELATED ------

    /// @notice The amount of FRAX denominated value being boosted that an address has staked
    /// @param account The address to check
    /// @return The amount of FRAX value boosted
    function userStakedFrax(address account) public view returns (uint256) {
        return (fraxPerLPStored * _locked_liquidity[account]) / MULTIPLIER_PRECISION;
    }

    /// @notice The amount of FRAX denominated value being boosted that a proxy address has staked
    /// @param proxy_address The address to check
    /// @return The amount of FRAX value boosted
    function proxyStakedFrax(address proxy_address) public view returns (uint256) {
        return (fraxPerLPStored * proxy_lp_balances[proxy_address]) / MULTIPLIER_PRECISION;
    }

    /// @notice The maximum LP that can get max veFXS boosted for a given address at its current veFXS balance
    /// @param account The address to check
    /// @return The maximum LP that can get max veFXS boosted for a given address at its current veFXS balance
    function maxLPForMaxBoost(address account) external view returns (uint256) {
        return (veFXS.balanceOf(account) * MULTIPLIER_PRECISION * MULTIPLIER_PRECISION) / (vefxs_per_frax_for_max_boost * fraxPerLPStored);
    }

    /// @notice Must be overriden to return the current FRAX per LP token
    /// @return The current number of FRAX per LP token
    function fraxPerLPToken() public virtual view returns (uint256) {
        revert("Need fPLPT logic");
    }

    // ------ veFXS RELATED ------

    /// @notice The minimum veFXS required to get max boost for a given address
    /// @param account The address to check
    /// @return The minimum veFXS required to get max boost
    function minVeFXSForMaxBoost(address account) public view returns (uint256) {
        return (userStakedFrax(account) * vefxs_per_frax_for_max_boost) / MULTIPLIER_PRECISION;
    }

    /// @notice The minimum veFXS required to get max boost for a given proxy
    /// @param proxy_address The proxy address
    /// @return The minimum veFXS required to get max boost
    function minVeFXSForMaxBoostProxy(address proxy_address) public view returns (uint256) {
        return (proxyStakedFrax(proxy_address) * vefxs_per_frax_for_max_boost) / MULTIPLIER_PRECISION;
    }

    /// @notice Looks up a staker's proxy
    /// @param addr The address to check
    /// @return the_proxy The proxy address, or address(0)
    function getProxyFor(address addr) public view returns (address){
        if (valid_vefxs_proxies[addr]) {
            // If addr itself is a proxy, return that.
            // If it farms itself directly, it should use the shared LP tally in proxyStakedFrax
            return addr;
        }
        else {
            // Otherwise, return the proxy, or address(0)
            return staker_designated_proxies[addr];
        }
    }

    /// @notice The multiplier for a given account, based on veFXS
    /// @param account The account to check
    /// @return vefxs_multiplier The multiplier boost for the account
    function veFXSMultiplier(address account) public view returns (uint256 vefxs_multiplier) {
        // Use either the user's or their proxy's veFXS balance
        uint256 vefxs_bal_to_use = 0;
        address the_proxy = getProxyFor(account);
        vefxs_bal_to_use = (the_proxy == address(0)) ? veFXS.balanceOf(account) : veFXS.balanceOf(the_proxy);

        // First option based on fraction of total veFXS supply, with an added scale factor
        uint256 mult_optn_1 = (vefxs_bal_to_use * vefxs_max_multiplier * vefxs_boost_scale_factor) 
                            / (veFXS.totalSupply() * MULTIPLIER_PRECISION);
        
        // Second based on old method, where the amount of FRAX staked comes into play
        uint256 mult_optn_2;
        {
            uint256 veFXS_needed_for_max_boost;

            // Need to use proxy-wide FRAX balance if applicable, to prevent exploiting
            veFXS_needed_for_max_boost = (the_proxy == address(0)) ? minVeFXSForMaxBoost(account) : minVeFXSForMaxBoostProxy(the_proxy);

            if (veFXS_needed_for_max_boost > 0){ 
                uint256 user_vefxs_fraction = (vefxs_bal_to_use * MULTIPLIER_PRECISION) / veFXS_needed_for_max_boost;
                
                mult_optn_2 = (user_vefxs_fraction * vefxs_max_multiplier) / MULTIPLIER_PRECISION;
            }
            else mult_optn_2 = 0; // This will happen with the first stake, when user_staked_frax is 0
        }

        // Select the higher of the two
        vefxs_multiplier = (mult_optn_1 > mult_optn_2 ? mult_optn_1 : mult_optn_2);

        // Cap the boost to the vefxs_max_multiplier
        if (vefxs_multiplier > vefxs_max_multiplier) vefxs_multiplier = vefxs_max_multiplier;
    }

    /* =============== MUTATIVE FUNCTIONS =============== */

    /// @notice Toggle whether a staker can use the proxy's veFXS balance to boost yields
    /// @notice Proxy must call this first, then the staker must call stakerSetVeFXSProxy
    function proxyToggleStaker(address staker_address) external {
        require(valid_vefxs_proxies[msg.sender], "Invalid proxy");
        proxy_allowed_stakers[msg.sender][staker_address] = !proxy_allowed_stakers[msg.sender][staker_address]; 

        // Disable the staker's set proxy if it was the toggler and is currently on
        if (staker_designated_proxies[staker_address] == msg.sender){
            staker_designated_proxies[staker_address] = address(0); 

            // Remove the LP as well
            proxy_lp_balances[msg.sender] -= _locked_liquidity[staker_address];
        }
    }

    /// @notice After proxy toggles staker to true, staker must call and confirm this
    /// @param proxy_address The address of the veFXS proxy
    function stakerSetVeFXSProxy(address proxy_address) external {
        require(valid_vefxs_proxies[proxy_address], "Invalid proxy");
        require(proxy_allowed_stakers[proxy_address][msg.sender], "Proxy has not allowed you yet");
        
        // Corner case sanity check to make sure LP isn't double counted
        address old_proxy_addr = staker_designated_proxies[msg.sender];
        if (old_proxy_addr != address(0)) {
            // Remove the LP count from the old proxy
            proxy_lp_balances[old_proxy_addr] -= _locked_liquidity[msg.sender];
        }

        // Set the new proxy
        staker_designated_proxies[msg.sender] = proxy_address; 

        // Add the the LP as well
        proxy_lp_balances[proxy_address] += _locked_liquidity[msg.sender];
    }

    // ------ STAKING ------
    // In children...


    // ------ WITHDRAWING ------
    // In children...


    // ------ REWARDS SYNCING ------

    function _updateRewardAndBalance(address account, bool sync_too) internal {
        _updateRewardAndBalance(account, sync_too, false);
    }

    function _updateRewardAndBalance(address account, bool sync_too, bool pre_sync_vemxstored) internal {
        // Need to retro-adjust some things if the period hasn't been renewed, then start a new one
        if (sync_too){
            sync();
        }

        // Used to make sure the veFXS multiplier is correct if a stake is increased, before calcCurCombinedWeight
        if (pre_sync_vemxstored){
            _vefxsMultiplierStored[account] = veFXSMultiplier(account);
        }
        
        if (account != address(0)) {
            // To keep the math correct, the user's combined weight must be recomputed to account for their
            // ever-changing veFXS balance.
            (   
                uint256 old_combined_weight,
                uint256 new_vefxs_multiplier,
                uint256 new_combined_weight
            ) = calcCurCombinedWeight(account);

            // Calculate the earnings first
            _syncEarned(account);

            // Update the user's stored veFXS multipliers
            _vefxsMultiplierStored[account] = new_vefxs_multiplier;

            // Update the user's and the global combined weights
            if (new_combined_weight >= old_combined_weight) {
                uint256 weight_diff = new_combined_weight - old_combined_weight;
                _total_combined_weight = _total_combined_weight + weight_diff;
                _combined_weights[account] = old_combined_weight + weight_diff;
            } else {
                uint256 weight_diff = old_combined_weight - new_combined_weight;
                _total_combined_weight = _total_combined_weight - weight_diff;
                _combined_weights[account] = old_combined_weight - weight_diff;
            }

        }
    }

    function _syncEarned(address account) internal {
        if (account != address(0)) {
            // Calculate the earnings
            uint256[] memory earned_arr = earned(account);

            // Update the rewards array
            for (uint256 i = 0; i < earned_arr.length; i++){ 
                rewards[account][i] = earned_arr[i];
            }

            // Update the rewards paid array
            for (uint256 i = 0; i < earned_arr.length; i++){ 
                userRewardsPerTokenPaid[account][i] = rewardsPerTokenStored[i];
            }
        }
    }


    // ------ REWARDS CLAIMING ------

    /// @notice A function that can be overridden to add extra logic to the getReward function
    /// @param destination_address The address to send the rewards to
    function getRewardExtraLogic(address destination_address) public nonReentrant {
        require(rewardsCollectionPaused == false, "Rewards collection paused");
        return _getRewardExtraLogic(msg.sender, destination_address);
    }

    function _getRewardExtraLogic(address rewardee, address destination_address) internal virtual {
        revert("Need gREL logic");
    }

    // Two different getReward functions are needed because of delegateCall and msg.sender issues
    // For backwards-compatibility
    /// @notice Claims rewards to destination address
    /// @param destination_address The address to send the rewards to
    /// @return rewards_before The rewards available before the claim
    function getReward(address destination_address) external nonReentrant returns (uint256[] memory) {
        return _getReward(msg.sender, destination_address, true);
    }

    /// @notice Claims rewards to destination address & wether to do extra logic
    /// @param destination_address The address to send the rewards to
    /// @param claim_extra_too Whether to do extra logic
    /// @return rewards_before The rewards available before the claim
    function getReward2(address destination_address, bool claim_extra_too) external nonReentrant returns (uint256[] memory) {
        return _getReward(msg.sender, destination_address, claim_extra_too);
    }

    // No withdrawer == msg.sender check needed since this is only internally callable
    function _getReward(address rewardee, address destination_address, bool do_extra_logic) internal updateRewardAndBalanceMdf(rewardee, true) returns (uint256[] memory rewards_before) {
        // Update the last reward claim time first, as an extra reentrancy safeguard
        lastRewardClaimTime[rewardee] = block.timestamp;
        
        // Make sure rewards collection isn't paused
        require(rewardsCollectionPaused == false, "Rewards collection paused");
        
        // Update the rewards array and distribute rewards
        rewards_before = new uint256[](rewardTokens.length);

        for (uint256 i = 0; i < rewardTokens.length; i++){ 
            rewards_before[i] = rewards[rewardee][i];
            rewards[rewardee][i] = 0;
            if (rewards_before[i] > 0) {
                TransferHelper.safeTransfer(rewardTokens[i], destination_address, rewards_before[i]);

                emit RewardPaid(rewardee, rewards_before[i], rewardTokens[i], destination_address);
            }
        }

        // Handle additional reward logic
        if (do_extra_logic) {
            _getRewardExtraLogic(rewardee, destination_address);
        }
    }


    // ------ FARM SYNCING ------

    // If the period expired, renew it
    function retroCatchUp() internal {
        // Catch up the old rewards first
        _updateStoredRewardsAndTime();

        // Pull in rewards from the rewards distributor, if applicable
        for (uint256 i = 0; i < rewardDistributors.length; i++){ 
            address reward_distributor_address = rewardDistributors[i];
            if (reward_distributor_address != address(0)) {
                IFraxGaugeFXSRewardsDistributor(reward_distributor_address).distributeReward(address(this));
            }
        }

        // Ensure the provided reward amount is not more than the balance in the contract.
        // This keeps the reward rate in the right range, preventing overflows due to
        // very high values of rewardRate in the earned and rewardsPerToken functions;
        // Reward + leftover must be less than 2^256 / 10^18 to avoid overflow.
        uint256 num_periods_elapsed = uint256(block.timestamp - periodFinish) / rewardsDuration; // Floor division to the nearest period
        
        // Make sure there are enough tokens to renew the reward period
        for (uint256 i = 0; i < rewardTokens.length; i++){ 
            require((rewardRates(i) * rewardsDuration * (num_periods_elapsed + 1)) <= IERC20(rewardTokens[i]).balanceOf(address(this)), string(abi.encodePacked("Not enough reward tokens available: ", rewardTokens[i])) );
        }
        
        // uint256 old_lastUpdateTime = lastUpdateTime;
        // uint256 new_lastUpdateTime = block.timestamp;

        // lastUpdateTime = periodFinish;
        periodFinish = periodFinish + ((num_periods_elapsed + 1) * rewardsDuration);

        // // Bunni oLIT rewards
        // // ==========================================
        // // Pull in rewards and set the reward rate for one week, based off of that
        // // If the rewards get messed up for some reason, set this to 0 and it will skip
        // // Should only be called once per week max
        // if (rewardRatesManual[1] != 0) {
        //     // oLIT
        //     // ====================================
        //     uint256 olit_before = IERC20(rewardTokens[1]).balanceOf(address(this));
        //     minter.mint(address(stakingToken));
        //     uint256 olit_after = IERC20(rewardTokens[1]).balanceOf(address(this));

        //     // Set the new reward rate
        //     rewardRatesManual[1] = (olit_after - olit_before) / rewardsDuration;
        // }

        // CONVEX EXTRA REWARDS (OLD METHOD)
        // ==========================================
        // Pull in rewards and set the reward rate for one week, based off of that
        // If the rewards get messed up for some reason, set this to 0 and it will skip
        // if (rewardRatesManual[1] != 0 && rewardRatesManual[2] != 0) {
        //     // CRV & CVX
        //     // ====================================
        //     uint256 crv_before = ERC20(rewardTokens[1]).balanceOf(address(this));
        //     uint256 cvx_before = ERC20(rewardTokens[2]).balanceOf(address(this));
        //     IConvexBaseRewardPool(0x329cb014b562d5d42927cfF0dEdF4c13ab0442EF).getReward(
        //         address(this),
        //         true
        //     );
        //     uint256 crv_after = ERC20(rewardTokens[1]).balanceOf(address(this));
        //     uint256 cvx_after = ERC20(rewardTokens[2]).balanceOf(address(this));

        //     // Set the new reward rate
        //     rewardRatesManual[1] = (crv_after - crv_before) / rewardsDuration;
        //     rewardRatesManual[2] = (cvx_after - cvx_before) / rewardsDuration;
        // }

        // Make sure everything is caught up again
        _updateStoredRewardsAndTime();
    }

    function _updateStoredRewardsAndTime() internal {
        // Get the rewards
        uint256[] memory rewards_per_token = rewardsPerToken();

        // Update the rewardsPerTokenStored
        for (uint256 i = 0; i < rewardsPerTokenStored.length; i++){ 
            rewardsPerTokenStored[i] = rewards_per_token[i];
        }

        // Update the last stored time
        lastUpdateTime = lastTimeRewardApplicable();
    }

    /// @notice Updates the gauge weights, if applicable
    /// @param force_update If true, will update the weights even if the time hasn't elapsed
    function sync_gauge_weights(bool force_update) public {
        // Loop through the gauge controllers
        for (uint256 i = 0; i < gaugeControllers.length; i++){ 
            address gauge_controller_address = gaugeControllers[i];
            if (gauge_controller_address != address(0)) {
                if (force_update || (block.timestamp > last_gauge_time_totals[i])){
                    // Update the gauge_relative_weight
                    last_gauge_relative_weights[i] = IFraxGaugeController(gauge_controller_address).gauge_relative_weight_write(address(this), block.timestamp);
                    last_gauge_time_totals[i] = IFraxGaugeController(gauge_controller_address).time_total();
                }
            }
        }
    }

    /// @notice Updates gauge weights, fraxPerLP, pulls in new rewards or updates rewards
    function sync() public {
        // Sync the gauge weight, if applicable
        sync_gauge_weights(false);

        // Update the fraxPerLPStored
        fraxPerLPStored = fraxPerLPToken();

        if (block.timestamp >= periodFinish) {
            retroCatchUp();
        }
        else {
            _updateStoredRewardsAndTime();
        }
    }

    /* ========== RESTRICTED FUNCTIONS - Curator callable ========== */
    
    // ------ FARM SYNCING ------
    // In children...

    // ------ PAUSES ------

    /// @notice Owner or governance can pause/unpause staking, withdrawals, rewards collection, and collectRewardsOnWithdrawal
    /// @param _stakingPaused Whether staking is paused
    /// @param _withdrawalsPaused Whether withdrawals are paused
    /// @param _rewardsCollectionPaused Whether rewards collection is paused
    /// @param _collectRewardsOnWithdrawalPaused Whether collectRewardsOnWithdrawal is paused
    function setPauses(
        bool _stakingPaused,
        bool _withdrawalsPaused,
        bool _rewardsCollectionPaused,
        bool _collectRewardsOnWithdrawalPaused
    ) external onlyByOwnGov {
        stakingPaused = _stakingPaused;
        withdrawalsPaused = _withdrawalsPaused;
        rewardsCollectionPaused = _rewardsCollectionPaused;
        collectRewardsOnWithdrawalPaused = _collectRewardsOnWithdrawalPaused;
    }

    /* ========== RESTRICTED FUNCTIONS - Owner or timelock only ========== */
    
    /// @notice Owner or governance can unlock stakes - irreversible!
    function unlockStakes() external onlyByOwnGov {
        stakesUnlocked = !stakesUnlocked;
    }

    /// @notice Owner or governance sets whether an address is a valid veFXS proxy
    /// @param _proxy_addr The address to set
    function toggleValidVeFXSProxy(address _proxy_addr) external onlyByOwnGov {
        valid_vefxs_proxies[_proxy_addr] = !valid_vefxs_proxies[_proxy_addr];
    }

    /// @notice Allows owner to recover any ERC20 or token manager to recover their reward token.
    /// @param tokenAddress The address of the token to recover
    /// @param tokenAmount The amount of the token to recover
    function recoverERC20(address tokenAddress, uint256 tokenAmount) external onlyTknMgrs(tokenAddress) {
        // Check if the desired token is a reward token
        bool isRewTkn = isRewardToken[tokenAddress];

        // Only the reward managers can take back their reward tokens
        // Also, other tokens, like the staking token, airdrops, or accidental deposits, can be withdrawn by the owner
        if (
                (isRewTkn && rewardManagers[tokenAddress] == msg.sender)
                || (!isRewTkn && (msg.sender == owner))
            ) {
            TransferHelper.safeTransfer(tokenAddress, msg.sender, tokenAmount);
            return;
        }
        // If none of the above conditions are true
        else {
            revert("No valid tokens to recover");
        }
    }

    /// @notice Sets multiple variables at once
    /// @param _misc_vars The variables to set:
    /// [0]: uint256 _lock_max_multiplier,
    /// [1] uint256 _vefxs_max_multiplier,
    /// [2] uint256 _vefxs_per_frax_for_max_boost,
    /// [3] uint256 _vefxs_boost_scale_factor,
    /// [4] uint256 _lock_time_for_max_multiplier,
    /// [5] uint256 _lock_time_min
    /// [6] uint256 _max_stake_limit (must be at greater or equal to old value)
    function setMiscVariables(
        uint256[6] memory _misc_vars
        // [0]: uint256 _lock_max_multiplier, 
        // [1] uint256 _vefxs_max_multiplier, 
        // [2] uint256 _vefxs_per_frax_for_max_boost,
        // [3] uint256 _vefxs_boost_scale_factor,
        // [4] uint256 _lock_time_for_max_multiplier,
        // [5] uint256 _lock_time_min
    ) external onlyByOwnGov {
        require(_misc_vars[0] >= MULTIPLIER_PRECISION, "Must be >= MUL PREC");
        require((_misc_vars[1] >= 0) && (_misc_vars[2] >= 0) && (_misc_vars[3] >= 0), "Must be >= 0");
        require((_misc_vars[4] >= 1) && (_misc_vars[5] >= 1), "Must be >= 1");

        lock_max_multiplier = _misc_vars[0];
        vefxs_max_multiplier = _misc_vars[1];
        vefxs_per_frax_for_max_boost = _misc_vars[2];
        vefxs_boost_scale_factor = _misc_vars[3];
        lock_time_for_max_multiplier = _misc_vars[4];
        lock_time_min = _misc_vars[5];
    }

    // The owner or the reward token managers can set reward rates 
        /// @notice Allows owner or reward token managers to set the reward rate for a given reward token
    /// @param reward_token_address The address of the reward token
    /// @param _new_rate The new reward rate (token amount divided by reward period duration)
    /// @param _gauge_controller_address The address of the gauge controller for this reward token
    /// @param _rewards_distributor_address The address of the rewards distributor for this reward token
    function setRewardVars(address reward_token_address, uint256 _new_rate, address _gauge_controller_address, address _rewards_distributor_address) external onlyTknMgrs(reward_token_address) {
        rewardRatesManual[rewardTokenAddrToIdx[reward_token_address]] = _new_rate;
        gaugeControllers[rewardTokenAddrToIdx[reward_token_address]] = _gauge_controller_address;
        rewardDistributors[rewardTokenAddrToIdx[reward_token_address]] = _rewards_distributor_address;
    }

    // The owner or the reward token managers can change managers
    /// @notice Allows owner or reward token managers to change the reward manager for a given reward token
    /// @param reward_token_address The address of the reward token
    /// @param new_manager_address The new reward manager address
    function changeTokenManager(address reward_token_address, address new_manager_address) external onlyTknMgrs(reward_token_address) {
        rewardManagers[reward_token_address] = new_manager_address;
    }

    /* ========== EVENTS ========== */
    event RewardPaid(address indexed user, uint256 amount, address token_address, address destination_address);

    /* ========== A CHICKEN ========== */
    //
    //         ,~.
    //      ,-'__ `-,
    //     {,-'  `. }              ,')
    //    ,( a )   `-.__         ,',')~,
    //   <=.) (         `-.__,==' ' ' '}
    //     (   )                      /)
    //      `-'\   ,                    )
    //          |  \        `~.        /
    //          \   `._        \      /
    //           \     `._____,'    ,'
    //            `-.             ,'
    //               `-._     _,-'
    //                   77jj'
    //                  //_||
    //               __//--'/`
    //             ,--'/`  '
    //
    // [hjw] https://textart.io/art/vw6Sa3iwqIRGkZsN1BC2vweF/chicken
}


// File contracts/Staking/FraxUnifiedFarm_ERC20.sol


// ====================================================================
// |     ______                   _______                             |
// |    / _____________ __  __   / ____(_____  ____ _____  ________   |
// |   / /_  / ___/ __ `| |/_/  / /_  / / __ \/ __ `/ __ \/ ___/ _ \  |
// |  / __/ / /  / /_/ _>  <   / __/ / / / / / /_/ / / / / /__/  __/  |
// | /_/   /_/   \__,_/_/|_|  /_/   /_/_/ /_/\__,_/_/ /_/\___/\___/   |
// |                                                                  |
// ====================================================================
// ======================= FraxUnifiedFarm_ERC20 ======================
// ====================================================================
// For ERC20 Tokens
// Uses FraxUnifiedFarmTemplate.sol

// -------------------- VARIES --------------------

// Bunni

// Convex wrappers





// Fraxlend
// import '../Fraxlend/IFraxlendPair.sol';

// Fraxswap
// import '../Fraxswap/core/interfaces/IFraxswapPair.sol';

// G-UNI
// import "../Misc_AMOs/gelato/IGUniPool.sol";

// mStable
// import '../Misc_AMOs/mstable/IFeederPool.sol';

// StakeDAO sdETH-FraxPut
// import '../Misc_AMOs/stakedao/IOpynPerpVault.sol';

// StakeDAO Vault
// import '../Misc_AMOs/stakedao/IStakeDaoVault.sol';

// Uniswap V2
// import '../Uniswap/Interfaces/IUniswapV2Pair.sol';

// Vesper
// import '../Misc_AMOs/vesper/IVPool.sol';

// ------------------------------------------------

contract FraxUnifiedFarm_ERC20 is FraxUnifiedFarmTemplate {

    /* ========== STATE VARIABLES ========== */

    // -------------------- COMMON -------------------- 
    bool internal frax_is_token0;

    // -------------------- VARIES --------------------

    // Convex stkcvxFPIFRAX, stkcvxFRAXBP, etc
    IConvexStakingWrapperFrax public stakingToken;
    I2poolToken public curveToken;
    I2pool public curvePool;
    // ICurvefrxETHETHPool public curvePool;

    // Fraxswap
    // IFraxswapPair public stakingToken;

    // Fraxlend
    // IFraxlendPair public stakingToken;

    // G-UNI
    // IGUniPool public stakingToken;
    
    // mStable
    // IFeederPool public stakingToken;

    // sdETH-FraxPut Vault
    // IOpynPerpVault public stakingToken;

    // StakeDAO Vault
    // IStakeDaoVault public stakingToken;

    // Uniswap V2
    // IUniswapV2Pair public stakingToken;

    // Vesper
    // IVPool public stakingToken;

    // ------------------------------------------------

    // Stake tracking
    mapping(address => LockedStake[]) public lockedStakes;

    /* ========== STRUCTS ========== */

    // Struct for the stake
    struct LockedStake {
        bytes32 kek_id;
        uint256 start_timestamp;
        uint256 liquidity;
        uint256 ending_timestamp;
        uint256 lock_multiplier; // 6 decimals of precision. 1x = 1000000
    }
    
    /* ========== CONSTRUCTOR ========== */

    constructor (
        address _owner,
        address[] memory _rewardTokens,
        address[] memory _rewardManagers,
        uint256[] memory _rewardRatesManual,
        address[] memory _gaugeControllers,
        address[] memory _rewardDistributors,
        address _stakingToken
    ) 
    FraxUnifiedFarmTemplate(_owner, _rewardTokens, _rewardManagers, _rewardRatesManual, _gaugeControllers, _rewardDistributors)
    {

        // -------------------- VARIES (USE CHILD FOR LOGIC) --------------------

        // Bunni
        // USE CHILD

        // Convex stkcvxFPIFRAX, stkcvxFRAXBP, etc
        // USE CHILD

        // Fraxlend
        // USE CHILD

        // Fraxswap
        // USE CHILD

        // G-UNI
        // stakingToken = IGUniPool(_stakingToken);
        // address token0 = address(stakingToken.token0());
        // frax_is_token0 = (token0 == frax_address);

        // mStable
        // stakingToken = IFeederPool(_stakingToken);

        // StakeDAO sdETH-FraxPut Vault
        // stakingToken = IOpynPerpVault(_stakingToken);

        // StakeDAO Vault
        // stakingToken = IStakeDaoVault(_stakingToken);

        // Uniswap V2
        // stakingToken = IUniswapV2Pair(_stakingToken);
        // address token0 = stakingToken.token0();
        // if (token0 == frax_address) frax_is_token0 = true;
        // else frax_is_token0 = false;

        // Vesper
        // stakingToken = IVPool(_stakingToken);
    }

    /* ============= VIEWS ============= */

    // ------ FRAX RELATED ------

    function fraxPerLPToken() public virtual view override returns (uint256) {
        // Get the amount of FRAX 'inside' of the lp tokens
        uint256 frax_per_lp_token;

        // Bunni
        // ============================================
        // USE CHILD

        // Convex stkcvxFPIFRAX and stkcvxFRAXBP only
        // ============================================
        // USE CHILD

        // Convex Stable/FRAXBP
        // ============================================
        // USE CHILD

        // Convex Volatile/FRAXBP
        // ============================================
        // USE CHILD

        // Fraxlend
        // ============================================
        // USE CHILD

        // Fraxswap
        // ============================================
        // USE CHILD

        // G-UNI
        // ============================================
        // {
        //     (uint256 reserve0, uint256 reserve1) = stakingToken.getUnderlyingBalances();
        //     uint256 total_frax_reserves = frax_is_token0 ? reserve0 : reserve1;

        //     frax_per_lp_token = (total_frax_reserves * 1e18) / stakingToken.totalSupply();
        // }

        // mStable
        // ============================================
        // {
        //     uint256 total_frax_reserves;
        //     (, IFeederPool.BassetData memory vaultData) = (stakingToken.getBasset(frax_address));
        //     total_frax_reserves = uint256(vaultData.vaultBalance);
        //     frax_per_lp_token = (total_frax_reserves * 1e18) / stakingToken.totalSupply();
        // }

        // StakeDAO sdETH-FraxPut Vault
        // ============================================
        // {
        //    uint256 frax3crv_held = stakingToken.totalUnderlyingControlled();
        
        //    // Optimistically assume 50/50 FRAX/3CRV ratio in the metapool to save gas
        //    frax_per_lp_token = ((frax3crv_held * 1e18) / stakingToken.totalSupply()) / 2;
        // }

        // StakeDAO Vault
        // ============================================
        // {
        //    uint256 frax3crv_held = stakingToken.balance();
        
        //    // Optimistically assume 50/50 FRAX/3CRV ratio in the metapool to save gas
        //    frax_per_lp_token = ((frax3crv_held * 1e18) / stakingToken.totalSupply()) / 2;
        // }

        // Uniswap V2
        // ============================================
        // {
        //     uint256 total_frax_reserves;
        //     (uint256 reserve0, uint256 reserve1, ) = (stakingToken.getReserves());
        //     if (frax_is_token0) total_frax_reserves = reserve0;
        //     else total_frax_reserves = reserve1;

        //     frax_per_lp_token = (total_frax_reserves * 1e18) / stakingToken.totalSupply();
        // }

        // Vesper
        // ============================================
        // frax_per_lp_token = stakingToken.pricePerShare();

        return frax_per_lp_token;
    }

    // ------ LIQUIDITY AND WEIGHTS ------
    function calcCurrLockMultiplier(address account, uint256 stake_idx) public view returns (uint256 midpoint_lock_multiplier) {
        // Get the stake
        LockedStake memory thisStake = lockedStakes[account][stake_idx];

        // Handles corner case where user never claims for a new stake
        // Don't want the multiplier going above the max
        uint256 accrue_start_time;
        if (lastRewardClaimTime[account] < thisStake.start_timestamp) {
            accrue_start_time = thisStake.start_timestamp;
        }
        else {
            accrue_start_time = lastRewardClaimTime[account];
        }
        
        // If the lock is expired
        if (thisStake.ending_timestamp <= block.timestamp) {
            // If the lock expired in the time since the last claim, the weight needs to be proportionately averaged this time
            if (lastRewardClaimTime[account] < thisStake.ending_timestamp){
                uint256 time_before_expiry = thisStake.ending_timestamp - accrue_start_time;
                uint256 time_after_expiry = block.timestamp - thisStake.ending_timestamp;

                // Average the pre-expiry lock multiplier
                uint256 pre_expiry_avg_multiplier = lockMultiplier(time_before_expiry / 2);

                // Get the weighted-average lock_multiplier
                // uint256 numerator = (pre_expiry_avg_multiplier * time_before_expiry) + (MULTIPLIER_PRECISION * time_after_expiry);
                uint256 numerator = (pre_expiry_avg_multiplier * time_before_expiry) + (0 * time_after_expiry);
                midpoint_lock_multiplier = numerator / (time_before_expiry + time_after_expiry);
            }
            else {
                // Otherwise, it needs to just be 1x
                // midpoint_lock_multiplier = MULTIPLIER_PRECISION;

                // Otherwise, it needs to just be 0x
                midpoint_lock_multiplier = 0;
            }
        }
        // If the lock is not expired
        else {
            // Decay the lock multiplier based on the time left
            uint256 avg_time_left;
            {
                uint256 time_left_p1 = thisStake.ending_timestamp - accrue_start_time;
                uint256 time_left_p2 = thisStake.ending_timestamp - block.timestamp;
                avg_time_left = (time_left_p1 + time_left_p2) / 2;
            }
            midpoint_lock_multiplier = lockMultiplier(avg_time_left);
        }

        // Sanity check: make sure it never goes above the initial multiplier
        if (midpoint_lock_multiplier > thisStake.lock_multiplier) midpoint_lock_multiplier = thisStake.lock_multiplier;
    }

    // Calculate the combined weight for an account
    function calcCurCombinedWeight(address account) public override view
        returns (
            uint256 old_combined_weight,
            uint256 new_vefxs_multiplier,
            uint256 new_combined_weight
        )
    {
        // Get the old combined weight
        old_combined_weight = _combined_weights[account];

        // Get the veFXS multipliers
        // For the calculations, use the midpoint (analogous to midpoint Riemann sum)
        new_vefxs_multiplier = veFXSMultiplier(account);

        uint256 midpoint_vefxs_multiplier;
        if (
            (_locked_liquidity[account] == 0 && _combined_weights[account] == 0) || 
            (new_vefxs_multiplier >= _vefxsMultiplierStored[account])
        ) {
            // This is only called for the first stake to make sure the veFXS multiplier is not cut in half
            // Also used if the user increased or maintained their position
            midpoint_vefxs_multiplier = new_vefxs_multiplier;
        }
        else {
            // Handles natural decay with a non-increased veFXS position
            midpoint_vefxs_multiplier = (new_vefxs_multiplier + _vefxsMultiplierStored[account]) / 2;
        }

        // Loop through the locked stakes, first by getting the liquidity * lock_multiplier portion
        new_combined_weight = 0;
        for (uint256 i = 0; i < lockedStakes[account].length; i++) {
            LockedStake memory thisStake = lockedStakes[account][i];

            // Calculate the midpoint lock multiplier
            uint256 midpoint_lock_multiplier = calcCurrLockMultiplier(account, i);

            // Calculate the combined boost
            uint256 liquidity = thisStake.liquidity;
            uint256 combined_boosted_amount = liquidity + ((liquidity * (midpoint_lock_multiplier + midpoint_vefxs_multiplier)) / MULTIPLIER_PRECISION);
            new_combined_weight += combined_boosted_amount;
        }
    }

    // ------ LOCK RELATED ------

    // All the locked stakes for a given account
    function lockedStakesOf(address account) external view returns (LockedStake[] memory) {
        return lockedStakes[account];
    }

    // Returns the length of the locked stakes for a given account
    function lockedStakesOfLength(address account) external view returns (uint256) {
        return lockedStakes[account].length;
    }

    // // All the locked stakes for a given account [old-school method]
    // function lockedStakesOfMultiArr(address account) external view returns (
    //     bytes32[] memory kek_ids,
    //     uint256[] memory start_timestamps,
    //     uint256[] memory liquidities,
    //     uint256[] memory ending_timestamps,
    //     uint256[] memory lock_multipliers
    // ) {
    //     for (uint256 i = 0; i < lockedStakes[account].length; i++){ 
    //         LockedStake memory thisStake = lockedStakes[account][i];
    //         kek_ids[i] = thisStake.kek_id;
    //         start_timestamps[i] = thisStake.start_timestamp;
    //         liquidities[i] = thisStake.liquidity;
    //         ending_timestamps[i] = thisStake.ending_timestamp;
    //         lock_multipliers[i] = thisStake.lock_multiplier;
    //     }
    // }

    /* =============== MUTATIVE FUNCTIONS =============== */

    // ------ STAKING ------

    function _updateLiqAmts(address staker_address, uint256 amt, bool is_add) internal {
        // Get the proxy address
        address the_proxy = getProxyFor(staker_address);

        if (is_add) {
            // Update total liquidities
            _total_liquidity_locked += amt;
            _locked_liquidity[staker_address] += amt;

            // Update the proxy
            if (the_proxy != address(0)) proxy_lp_balances[the_proxy] += amt;
        }
        else {
            // Update total liquidities
            _total_liquidity_locked -= amt;
            _locked_liquidity[staker_address] -= amt;

            // Update the proxy
            if (the_proxy != address(0)) proxy_lp_balances[the_proxy] -= amt;
        }

        // Need to call to update the combined weights
        _updateRewardAndBalance(staker_address, false, true);
    }

    function _getStake(address staker_address, bytes32 kek_id) internal view returns (LockedStake memory locked_stake, uint256 arr_idx) {
        for (uint256 i = 0; i < lockedStakes[staker_address].length; i++){ 
            if (kek_id == lockedStakes[staker_address][i].kek_id){
                locked_stake = lockedStakes[staker_address][i];
                arr_idx = i;
                break;
            }
        }
        require(locked_stake.kek_id == kek_id, "Stake not found");
        
    }

    // Add additional LPs to an existing locked stake
    function lockAdditional(bytes32 kek_id, uint256 addl_liq) nonReentrant updateRewardAndBalanceMdf(msg.sender, true) public {
        // Get the stake and its index
        (LockedStake memory thisStake, uint256 theArrayIndex) = _getStake(msg.sender, kek_id);

        // Calculate the new amount
        uint256 new_amt = thisStake.liquidity + addl_liq;

        // Checks
        require(addl_liq >= 0, "Must be positive");

        // Pull the tokens from the sender
        TransferHelper.safeTransferFrom(address(stakingToken), msg.sender, address(this), addl_liq);

        // Update the stake
        lockedStakes[msg.sender][theArrayIndex] = LockedStake(
            kek_id,
            thisStake.start_timestamp,
            new_amt,
            thisStake.ending_timestamp,
            thisStake.lock_multiplier
        );

        // Update liquidities
        _updateLiqAmts(msg.sender, addl_liq, true);

        emit LockedAdditional(msg.sender, kek_id, addl_liq);
    }

    // Extends the lock of an existing stake
    function lockLonger(bytes32 kek_id, uint256 new_ending_ts) nonReentrant updateRewardAndBalanceMdf(msg.sender, true) public {
        // Get the stake and its index
        (LockedStake memory thisStake, uint256 theArrayIndex) = _getStake(msg.sender, kek_id);

        // Check
        require(new_ending_ts > block.timestamp, "Must be in the future");

        // Calculate some times
        uint256 time_left = (thisStake.ending_timestamp > block.timestamp) ? thisStake.ending_timestamp - block.timestamp : 0;
        uint256 new_secs = new_ending_ts - block.timestamp;

        // Checks
        // require(time_left > 0, "Already expired");
        require(new_secs > time_left, "Cannot shorten lock time");
        require(new_secs >= lock_time_min, "Minimum stake time not met");
        require(new_secs <= lock_time_for_max_multiplier, "Trying to lock for too long");

        // Update the stake
        lockedStakes[msg.sender][theArrayIndex] = LockedStake(
            kek_id,
            block.timestamp,
            thisStake.liquidity,
            new_ending_ts,
            lockMultiplier(new_secs)
        );

        // Need to call to update the combined weights
        _updateRewardAndBalance(msg.sender, false, true);

        emit LockedLonger(msg.sender, kek_id, new_secs, block.timestamp, new_ending_ts);
    }

    

    // Two different stake functions are needed because of delegateCall and msg.sender issues (important for proxies)
    function stakeLocked(uint256 liquidity, uint256 secs) nonReentrant external returns (bytes32) {
        return _stakeLocked(msg.sender, msg.sender, liquidity, secs, block.timestamp);
    }

    // If this were not internal, and source_address had an infinite approve, this could be exploitable
    // (pull funds from source_address and stake for an arbitrary staker_address)
    function _stakeLocked(
        address staker_address,
        address source_address,
        uint256 liquidity,
        uint256 secs,
        uint256 start_timestamp
    ) internal updateRewardAndBalanceMdf(staker_address, true) returns (bytes32) {
        require(stakingPaused == false, "Staking paused");
        require(secs >= lock_time_min, "Minimum stake time not met");
        require(secs <= lock_time_for_max_multiplier,"Trying to lock for too long");

        // Pull in the required token(s)
        // Varies per farm
        TransferHelper.safeTransferFrom(address(stakingToken), source_address, address(this), liquidity);

        // Get the lock multiplier and kek_id
        uint256 lock_multiplier = lockMultiplier(secs);
        bytes32 kek_id = keccak256(abi.encodePacked(staker_address, start_timestamp, liquidity, _locked_liquidity[staker_address]));
        
        // Create the locked stake
        lockedStakes[staker_address].push(LockedStake(
            kek_id,
            start_timestamp,
            liquidity,
            start_timestamp + secs,
            lock_multiplier
        ));

        // Update liquidities
        _updateLiqAmts(staker_address, liquidity, true);

        emit StakeLocked(staker_address, liquidity, secs, kek_id, source_address);

        return kek_id;
    }

    // ------ WITHDRAWING ------

    // Two different withdrawLocked functions are needed because of delegateCall and msg.sender issues (important for proxies)
    function withdrawLocked(bytes32 kek_id, address destination_address, bool claim_rewards) nonReentrant external returns (uint256) {
        require(withdrawalsPaused == false, "Withdrawals paused");
        return _withdrawLocked(msg.sender, destination_address, kek_id, claim_rewards);
    }

    // No withdrawer == msg.sender check needed since this is only internally callable and the checks are done in the wrapper
    function _withdrawLocked(
        address staker_address,
        address destination_address,
        bytes32 kek_id,
        bool claim_rewards
    ) internal returns (uint256) {
        // Collect rewards first and then update the balances
        // collectRewardsOnWithdrawalPaused to be used in an emergency situation if reward is overemitted or not available
        // and the user can forfeit rewards to get their principal back. User can also specify it in withdrawLocked
        if (claim_rewards || !collectRewardsOnWithdrawalPaused) _getReward(staker_address, destination_address, true);
        else {
            // Sync the rewards at least
            _updateRewardAndBalance(staker_address, true, false);
        }

        // Get the stake and its index
        (LockedStake memory thisStake, uint256 theArrayIndex) = _getStake(staker_address, kek_id);
        require(block.timestamp >= thisStake.ending_timestamp || stakesUnlocked == true, "Stake is still locked!");
        uint256 liquidity = thisStake.liquidity;

        if (liquidity > 0) {

            // Give the tokens to the destination_address
            // Should throw if insufficient balance
            TransferHelper.safeTransfer(address(stakingToken), destination_address, liquidity);

            // Remove the stake from the array
            delete lockedStakes[staker_address][theArrayIndex];

            // Update liquidities
            _updateLiqAmts(staker_address, liquidity, false);

            emit WithdrawLocked(staker_address, liquidity, kek_id, destination_address);
        }

        return liquidity;
    }


    function _getRewardExtraLogic(address rewardee, address destination_address) internal override {
        // Do nothing
    }

    /* ========== RESTRICTED FUNCTIONS - Owner or timelock only ========== */

    // Inherited...

    /* ========== EVENTS ========== */
    event LockedAdditional(address indexed user, bytes32 kek_id, uint256 amount);
    event LockedLonger(address indexed user, bytes32 kek_id, uint256 new_secs, uint256 new_start_ts, uint256 new_end_ts);
    event StakeLocked(address indexed user, uint256 amount, uint256 secs, bytes32 kek_id, address source_address);
    event WithdrawLocked(address indexed user, uint256 liquidity, bytes32 kek_id, address destination_address);
}


// File contracts/Staking/Variants/FraxUnifiedFarm_ERC20_Convex_FRAXBP_Volatile.sol






contract FraxUnifiedFarm_ERC20_Convex_FRAXBP_Volatile is FraxUnifiedFarm_ERC20 {

    constructor (
        address _owner,
        address[] memory _rewardTokens,
        address[] memory _rewardManagers,
        uint256[] memory _rewardRates,
        address[] memory _gaugeControllers,
        address[] memory _rewardDistributors,
        address _stakingToken 
    ) 
    FraxUnifiedFarm_ERC20(_owner , _rewardTokens, _rewardManagers, _rewardRates, _gaugeControllers, _rewardDistributors, _stakingToken)
    {
        // COMMENTED OUT SO COMPILER DOESNT COMPLAIN. UNCOMMENT WHEN DEPLOYING

        // Convex stkcvxFPIFRAX and stkcvxFRAXBP. Also Volatile/FRAXBP
        stakingToken = IConvexStakingWrapperFrax(_stakingToken);
        curveToken = I2poolToken(stakingToken.curveToken());
        curvePool = I2pool(curveToken.minter());
        address token0 = curvePool.coins(0);
        frax_is_token0 = (token0 == frax_address);
    }

    function fraxPerLPToken() public view override returns (uint256 frax_per_lp_token) {
        // COMMENTED OUT SO COMPILER DOESNT COMPLAIN. UNCOMMENT WHEN DEPLOYING

        // Convex Volatile/FRAXBP
        // ============================================
        {
            // Half of the LP is FRAXBP. Half of that should be FRAX.
            // Using 0.25 * lp price for gas savings
            frax_per_lp_token = (curvePool.lp_price() * (1e18)) / (4 * curvePool.price_oracle()); 
        }

    }
}

Contract Security Audit

Contract ABI

[{"inputs":[{"internalType":"address","name":"_owner","type":"address"},{"internalType":"address[]","name":"_rewardTokens","type":"address[]"},{"internalType":"address[]","name":"_rewardManagers","type":"address[]"},{"internalType":"uint256[]","name":"_rewardRates","type":"uint256[]"},{"internalType":"address[]","name":"_gaugeControllers","type":"address[]"},{"internalType":"address[]","name":"_rewardDistributors","type":"address[]"},{"internalType":"address","name":"_stakingToken","type":"address"}],"stateMutability":"nonpayable","type":"constructor"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"user","type":"address"},{"indexed":false,"internalType":"bytes32","name":"kek_id","type":"bytes32"},{"indexed":false,"internalType":"uint256","name":"amount","type":"uint256"}],"name":"LockedAdditional","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"user","type":"address"},{"indexed":false,"internalType":"bytes32","name":"kek_id","type":"bytes32"},{"indexed":false,"internalType":"uint256","name":"new_secs","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"new_start_ts","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"new_end_ts","type":"uint256"}],"name":"LockedLonger","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"oldOwner","type":"address"},{"indexed":false,"internalType":"address","name":"newOwner","type":"address"}],"name":"OwnerChanged","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"newOwner","type":"address"}],"name":"OwnerNominated","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"user","type":"address"},{"indexed":false,"internalType":"uint256","name":"amount","type":"uint256"},{"indexed":false,"internalType":"address","name":"token_address","type":"address"},{"indexed":false,"internalType":"address","name":"destination_address","type":"address"}],"name":"RewardPaid","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"user","type":"address"},{"indexed":false,"internalType":"uint256","name":"amount","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"secs","type":"uint256"},{"indexed":false,"internalType":"bytes32","name":"kek_id","type":"bytes32"},{"indexed":false,"internalType":"address","name":"source_address","type":"address"}],"name":"StakeLocked","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"user","type":"address"},{"indexed":false,"internalType":"uint256","name":"liquidity","type":"uint256"},{"indexed":false,"internalType":"bytes32","name":"kek_id","type":"bytes32"},{"indexed":false,"internalType":"address","name":"destination_address","type":"address"}],"name":"WithdrawLocked","type":"event"},{"inputs":[],"name":"acceptOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"account","type":"address"}],"name":"calcCurCombinedWeight","outputs":[{"internalType":"uint256","name":"old_combined_weight","type":"uint256"},{"internalType":"uint256","name":"new_vefxs_multiplier","type":"uint256"},{"internalType":"uint256","name":"new_combined_weight","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"account","type":"address"},{"internalType":"uint256","name":"stake_idx","type":"uint256"}],"name":"calcCurrLockMultiplier","outputs":[{"internalType":"uint256","name":"midpoint_lock_multiplier","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"reward_token_address","type":"address"},{"internalType":"address","name":"new_manager_address","type":"address"}],"name":"changeTokenManager","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"account","type":"address"}],"name":"combinedWeightOf","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"curvePool","outputs":[{"internalType":"contract I2pool","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"curveToken","outputs":[{"internalType":"contract I2poolToken","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"account","type":"address"}],"name":"earned","outputs":[{"internalType":"uint256[]","name":"new_earned","type":"uint256[]"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"fraxPerLPStored","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"fraxPerLPToken","outputs":[{"internalType":"uint256","name":"frax_per_lp_token","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"getAllRewardTokens","outputs":[{"internalType":"address[]","name":"","type":"address[]"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"addr","type":"address"}],"name":"getProxyFor","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"destination_address","type":"address"}],"name":"getReward","outputs":[{"internalType":"uint256[]","name":"","type":"uint256[]"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"destination_address","type":"address"},{"internalType":"bool","name":"claim_extra_too","type":"bool"}],"name":"getReward2","outputs":[{"internalType":"uint256[]","name":"","type":"uint256[]"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"destination_address","type":"address"}],"name":"getRewardExtraLogic","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"getRewardForDuration","outputs":[{"internalType":"uint256[]","name":"rewards_per_duration_arr","type":"uint256[]"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"caller_addr","type":"address"},{"internalType":"address","name":"reward_token_addr","type":"address"}],"name":"isTokenManagerFor","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"lastRewardClaimTime","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"lastUpdateTime","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"bytes32","name":"kek_id","type":"bytes32"},{"internalType":"uint256","name":"addl_liq","type":"uint256"}],"name":"lockAdditional","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"bytes32","name":"kek_id","type":"bytes32"},{"internalType":"uint256","name":"new_ending_ts","type":"uint256"}],"name":"lockLonger","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"secs","type":"uint256"}],"name":"lockMultiplier","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"lock_max_multiplier","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"lock_time_for_max_multiplier","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"lock_time_min","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"account","type":"address"}],"name":"lockedLiquidityOf","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"},{"internalType":"uint256","name":"","type":"uint256"}],"name":"lockedStakes","outputs":[{"internalType":"bytes32","name":"kek_id","type":"bytes32"},{"internalType":"uint256","name":"start_timestamp","type":"uint256"},{"internalType":"uint256","name":"liquidity","type":"uint256"},{"internalType":"uint256","name":"ending_timestamp","type":"uint256"},{"internalType":"uint256","name":"lock_multiplier","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"account","type":"address"}],"name":"lockedStakesOf","outputs":[{"components":[{"internalType":"bytes32","name":"kek_id","type":"bytes32"},{"internalType":"uint256","name":"start_timestamp","type":"uint256"},{"internalType":"uint256","name":"liquidity","type":"uint256"},{"internalType":"uint256","name":"ending_timestamp","type":"uint256"},{"internalType":"uint256","name":"lock_multiplier","type":"uint256"}],"internalType":"struct FraxUnifiedFarm_ERC20.LockedStake[]","name":"","type":"tuple[]"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"account","type":"address"}],"name":"lockedStakesOfLength","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"account","type":"address"}],"name":"maxLPForMaxBoost","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"account","type":"address"}],"name":"minVeFXSForMaxBoost","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"proxy_address","type":"address"}],"name":"minVeFXSForMaxBoostProxy","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_owner","type":"address"}],"name":"nominateNewOwner","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"nominatedOwner","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"owner","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"periodFinish","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"proxy_address","type":"address"}],"name":"proxyStakedFrax","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"staker_address","type":"address"}],"name":"proxyToggleStaker","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"proxy_lp_balances","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"tokenAddress","type":"address"},{"internalType":"uint256","name":"tokenAmount","type":"uint256"}],"name":"recoverERC20","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"rewardManagers","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"token_idx","type":"uint256"}],"name":"rewardRates","outputs":[{"internalType":"uint256","name":"rwd_rate","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"rewardTokenAddrToIdx","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"rewardsDuration","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"rewardsPerToken","outputs":[{"internalType":"uint256[]","name":"newRewardsPerTokenStored","type":"uint256[]"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256[6]","name":"_misc_vars","type":"uint256[6]"}],"name":"setMiscVariables","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"bool","name":"_stakingPaused","type":"bool"},{"internalType":"bool","name":"_withdrawalsPaused","type":"bool"},{"internalType":"bool","name":"_rewardsCollectionPaused","type":"bool"},{"internalType":"bool","name":"_collectRewardsOnWithdrawalPaused","type":"bool"}],"name":"setPauses","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"reward_token_address","type":"address"},{"internalType":"uint256","name":"_new_rate","type":"uint256"},{"internalType":"address","name":"_gauge_controller_address","type":"address"},{"internalType":"address","name":"_rewards_distributor_address","type":"address"}],"name":"setRewardVars","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"liquidity","type":"uint256"},{"internalType":"uint256","name":"secs","type":"uint256"}],"name":"stakeLocked","outputs":[{"internalType":"bytes32","name":"","type":"bytes32"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"proxy_address","type":"address"}],"name":"stakerSetVeFXSProxy","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"staker_designated_proxies","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"stakesUnlocked","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"stakingToken","outputs":[{"internalType":"contract IConvexStakingWrapperFrax","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"sync","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"bool","name":"force_update","type":"bool"}],"name":"sync_gauge_weights","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_proxy_addr","type":"address"}],"name":"toggleValidVeFXSProxy","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"totalCombinedWeight","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"totalLiquidityLocked","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"unlockStakes","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"account","type":"address"}],"name":"userStakedFrax","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"account","type":"address"}],"name":"veFXSMultiplier","outputs":[{"internalType":"uint256","name":"vefxs_multiplier","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"vefxs_boost_scale_factor","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"vefxs_max_multiplier","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"vefxs_per_frax_for_max_boost","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"bytes32","name":"kek_id","type":"bytes32"},{"internalType":"address","name":"destination_address","type":"address"},{"internalType":"bool","name":"claim_rewards","type":"bool"}],"name":"withdrawLocked","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"nonpayable","type":"function"}]

Deployed Bytecode

0x608060405234801561001057600080fd5b50600436106103ea5760003560e01c80637d6ef08e1161021a578063d2010fb411610135578063e1ba95d2116100c8578063f2a8d34911610097578063f77e34d11161007c578063f77e34d1146109a0578063facefb64146109b3578063fff6cae9146109c657600080fd5b8063f2a8d34914610984578063f2caeb1e1461098d57600080fd5b8063e1ba95d214610957578063e7f305821461095f578063ebe2b12b14610972578063f288baf61461097b57600080fd5b8063d7400d5611610104578063d7400d56146108f3578063d9f96e8d14610906578063de1a65511461093c578063e01f62bf1461094f57600080fd5b8063d2010fb4146108b1578063d2fbdc0d146108ba578063d42fc9b4146108cd578063d5e1a9c6146108e057600080fd5b8063aa1d4fce116101ad578063c35438261161017c578063c354382614610856578063c8f33c9114610869578063ca6df29d14610872578063cdc82e80146108a857600080fd5b8063aa1d4fce1461081e578063b85efd0614610827578063b94c4dcb1461083a578063c00007b01461084357600080fd5b80638da5cb5b116101e95780638da5cb5b146107cb57806391cf600a146107eb5780639637927f146107fe578063a0f234761461080b57600080fd5b80637d6ef08e146107585780637f472e54146107775780638980f11f1461078a5780638bad86a71461079d57600080fd5b8063387edc861161030a5780636c430dbb1161029d578063774d4ae71161026c578063774d4ae7146106ef5780637910d17b146107025780637970833e1461071557806379ba50971461075057600080fd5b80636c430dbb146106945780636e27cef9146106b457806370641a36146106bd57806372f702f3146106c557600080fd5b806353a47bb7116102d957806353a47bb7146106445780635bfd92581461066457806364f2c0601461066c578063693392451461067457600080fd5b8063387edc86146105c857806341a16f3f146105db5780634f39059c146106115780634fd2b5361461063157600080fd5b80631face856116103825780632c0c2a0a116103515780632c0c2a0a146105625780632df079f11461057557806336f89af214610588578063386a9525146105be57600080fd5b80631face856146104b1578063218751b2146104c4578063231b68dc1461050957806328408bab1461052c57600080fd5b80631627540c116103be5780631627540c1461046157806317b18c89146104765780631c1f78eb146104895780631e090f011461049157600080fd5b80628cc262146103ef5780630238b936146104185780630d7bac4f1461043957806312edb24c1461044c575b600080fd5b6104026103fd3660046149a8565b6109ce565b60405161040f91906149c3565b60405180910390f35b61042b610426366004614a15565b610b4d565b60405190815260200161040f565b61042b610447366004614a55565b610c4e565b610454610c7a565b60405161040f9190614a6e565b61047461046f3660046149a8565b610ce9565b005b61042b610484366004614abc565b610e09565b610402610e93565b6104a461049f3660046149a8565b610f34565b60405161040f9190614ade565b61042b6104bf3660046149a8565b610fe8565b6024546104e49073ffffffffffffffffffffffffffffffffffffffff1681565b60405173ffffffffffffffffffffffffffffffffffffffff909116815260200161040f565b61051c610517366004614b4c565b61102d565b604051901515815260200161040f565b6104e461053a3660046149a8565b60216020526000908152604090205473ffffffffffffffffffffffffffffffffffffffff1681565b61042b6105703660046149a8565b611113565b610474610583366004614b7f565b6113ff565b61042b6105963660046149a8565b73ffffffffffffffffffffffffffffffffffffffff166000908152601f602052604090205490565b61042b62093a8081565b6104746105d63660046149a8565b611566565b6104e46105e93660046149a8565b600f6020526000908152604090205473ffffffffffffffffffffffffffffffffffffffff1681565b6023546104e49073ffffffffffffffffffffffffffffffffffffffff1681565b61042b61063f3660046149a8565b611650565b6001546104e49073ffffffffffffffffffffffffffffffffffffffff1681565b61042b611671565b601d5461042b565b61042b6106823660046149a8565b60156020526000908152604090205481565b61042b6106a23660046149a8565b60196020526000908152604090205481565b61042b60085481565b6104026117c2565b6022546104e4906601000000000000900473ffffffffffffffffffffffffffffffffffffffff1681565b61042b6106fd366004614bdb565b611921565b610402610710366004614c05565b611b2d565b610728610723366004614bdb565b611ba9565b604080519586526020860194909452928401919091526060830152608082015260a00161040f565b610474611bf7565b61042b6107663660046149a8565b602080526000908152604090205481565b61042b6107853660046149a8565b611d42565b610474610798366004614bdb565b611d59565b6107b06107ab3660046149a8565b611ef0565b6040805193845260208401929092529082015260600161040f565b6000546104e49073ffffffffffffffffffffffffffffffffffffffff1681565b6104746107f93660046149a8565b612124565b60225461051c9060ff1681565b61042b6108193660046149a8565b612217565b61042b60095481565b610474610835366004614abc565b6122e1565b61042b60075481565b6104026108513660046149a8565b61249d565b6104e46108643660046149a8565b612525565b61042b60055481565b61042b6108803660046149a8565b73ffffffffffffffffffffffffffffffffffffffff1660009081526025602052604090205490565b61042b60065481565b61042b60035481565b6104746108c8366004614abc565b612588565b61042b6108db3660046149a8565b6128f0565b6104746108ee366004614c3c565b61292c565b6104746109013660046149a8565b612b1b565b61042b6109143660046149a8565b73ffffffffffffffffffffffffffffffffffffffff166000908152601e602052604090205490565b61047461094a366004614b4c565b612d35565b601c5461042b565b610474612e1c565b61047461096d3660046149a8565b612eed565b61042b60045481565b61042b600a5481565b61042b600b5481565b61042b61099b366004614a55565b61304d565b6104746109ae366004614c89565b613164565b6104746109c1366004614cd5565b61333a565b6104746134fa565b606060006109da6117c2565b60105490915067ffffffffffffffff8111156109f8576109f8614ca6565b604051908082528060200260200182016040528015610a21578160200160208202803683370190505b5073ffffffffffffffffffffffffffffffffffffffff84166000908152601f602052604090205490925015610b475760005b601054811015610b455773ffffffffffffffffffffffffffffffffffffffff84166000818152601860209081526040808320858452825280832054938352601782528083208584529091529020548351670de0b6b3a76400009190859085908110610ac057610ac0614d7a565b6020026020010151610ad29190614dd8565b73ffffffffffffffffffffffffffffffffffffffff87166000908152601f6020526040902054610b029190614deb565b610b0c9190614e02565b610b169190614e3d565b838281518110610b2857610b28614d7a565b602090810291909101015280610b3d81614e50565b915050610a53565b505b50919050565b60006002805403610bbf576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601f60248201527f5265656e7472616e637947756172643a207265656e7472616e742063616c6c0060448201526064015b60405180910390fd5b60028055602254610100900460ff1615610c35576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601260248201527f5769746864726177616c732070617573656400000000000000000000000000006044820152606401610bb6565b610c413384868561352a565b6001600255949350505050565b600654600754600091610c7491610c658286614deb565b610c6f9190614e02565b613708565b92915050565b60606010805480602002602001604051908101604052809291908181526020018280548015610cdf57602002820191906000526020600020905b815473ffffffffffffffffffffffffffffffffffffffff168152600190910190602001808311610cb4575b5050505050905090565b60005473ffffffffffffffffffffffffffffffffffffffff163314610d90576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152602f60248201527f4f6e6c792074686520636f6e7472616374206f776e6572206d6179207065726660448201527f6f726d207468697320616374696f6e00000000000000000000000000000000006064820152608401610bb6565b600180547fffffffffffffffffffffffff00000000000000000000000000000000000000001673ffffffffffffffffffffffffffffffffffffffff83169081179091556040519081527f906a1c6bd7e3091ea86693dd029a831c19049ce77f1dce2ce0bab1cacbabce229060200160405180910390a150565b60006002805403610e76576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601f60248201527f5265656e7472616e637947756172643a207265656e7472616e742063616c6c006044820152606401610bb6565b60028055610e873380858542613720565b60016002559392505050565b60135460609067ffffffffffffffff811115610eb157610eb1614ca6565b604051908082528060200260200182016040528015610eda578160200160208202803683370190505b50905060005b601354811015610f305762093a80610ef78261304d565b610f019190614deb565b828281518110610f1357610f13614d7a565b602090810291909101015280610f2881614e50565b915050610ee0565b5090565b73ffffffffffffffffffffffffffffffffffffffff81166000908152602560209081526040808320805482518185028101850190935280835260609492939192909184015b82821015610fdd57838290600052602060002090600502016040518060a00160405290816000820154815260200160018201548152602001600282015481526020016003820154815260200160048201548152505081526020019060010190610f79565b505050509050919050565b73ffffffffffffffffffffffffffffffffffffffff81166000908152602080526040812054600354670de0b6b3a76400009161102391614deb565b610c749190614e02565b73ffffffffffffffffffffffffffffffffffffffff811660009081526014602052604081205460ff1661106257506000610c74565b73ffffffffffffffffffffffffffffffffffffffff83161580611099575073ffffffffffffffffffffffffffffffffffffffff8216155b156110a657506000610c74565b60005473ffffffffffffffffffffffffffffffffffffffff908116908416036110d157506001610c74565b73ffffffffffffffffffffffffffffffffffffffff8281166000908152600f602052604090205481851691160361110a57506001610c74565b50600092915050565b6000808061112084612525565b905073ffffffffffffffffffffffffffffffffffffffff8116156111e5576040517f70a0823100000000000000000000000000000000000000000000000000000000815273ffffffffffffffffffffffffffffffffffffffff8216600482015273c8418af6358ffdda74e09ca9cc3fe03ca6adc5b0906370a0823190602401602060405180830381865afa1580156111bc573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906111e09190614e88565b611287565b6040517f70a0823100000000000000000000000000000000000000000000000000000000815273ffffffffffffffffffffffffffffffffffffffff8516600482015273c8418af6358ffdda74e09ca9cc3fe03ca6adc5b0906370a0823190602401602060405180830381865afa158015611263573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906112879190614e88565b91506000670de0b6b3a764000073c8418af6358ffdda74e09ca9cc3fe03ca6adc5b073ffffffffffffffffffffffffffffffffffffffff166318160ddd6040518163ffffffff1660e01b8152600401602060405180830381865afa1580156112f3573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906113179190614e88565b6113219190614deb565b600954600a546113319086614deb565b61133b9190614deb565b6113459190614e02565b905060008073ffffffffffffffffffffffffffffffffffffffff8416156113745761136f84611d42565b61137d565b61137d87611650565b905080156113cf5760008161139a670de0b6b3a764000088614deb565b6113a49190614e02565b9050670de0b6b3a7640000600a54826113bd9190614deb565b6113c79190614e02565b9250506113d4565b600091505b508082116113e257806113e4565b815b9450600a548511156113f657600a5494505b50505050919050565b60005473ffffffffffffffffffffffffffffffffffffffff163314806114385750738412ebf45bac1b340bbe8f318b928c466c4e39ca33145b61149e576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601560248201527f4e6f74206f776e6572206f722074696d656c6f636b00000000000000000000006044820152606401610bb6565b602280547fffffffffffffffffffffffffffffffffffffffffffffffffffffffff00ff00ff166301000000951515959095027fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff00ff169490941761010093151593909302929092177fffffffffffffffffffffffffffffffffffffffffffffffffffffff00ff00ffff1662010000911515919091027fffffffffffffffffffffffffffffffffffffffffffffffffffffff00ffffffff161764010000000091151591909102179055565b60028054036115d1576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601f60248201527f5265656e7472616e637947756172643a207265656e7472616e742063616c6c006044820152606401610bb6565b6002805560225462010000900460ff1615611648576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601960248201527f5265776172647320636f6c6c656374696f6e20706175736564000000000000006044820152606401610bb6565b506001600255565b6000670de0b6b3a7640000600b54611667846128f0565b6110239190614deb565b602454604080517f86fc88d3000000000000000000000000000000000000000000000000000000008152905160009273ffffffffffffffffffffffffffffffffffffffff16916386fc88d39160048083019260209291908290030181865afa1580156116e1573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906117059190614e88565b611710906004614deb565b602460009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff166354f0f7d56040518163ffffffff1660e01b8152600401602060405180830381865afa15801561177d573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906117a19190614e88565b6117b390670de0b6b3a7640000614deb565b6117bd9190614e02565b905090565b6060601c54600014806117d55750601d54155b1561182f576016805480602002602001604051908101604052809291908181526020018280548015610cdf57602002820191906000526020600020905b815481526020019060010190808311611812575050505050905090565b60105467ffffffffffffffff81111561184a5761184a614ca6565b604051908082528060200260200182016040528015611873578160200160208202803683370190505b50905060005b601654811015610f3057601d5461188f8261304d565b60055461189a613a74565b6118a49190614dd8565b6118ae9190614deb565b6118c090670de0b6b3a7640000614deb565b6118ca9190614e02565b601682815481106118dd576118dd614d7a565b90600052602060002001546118f29190614e3d565b82828151811061190457611904614d7a565b60209081029190910101528061191981614e50565b915050611879565b73ffffffffffffffffffffffffffffffffffffffff8216600090815260256020526040812080548291908490811061195b5761195b614d7a565b600091825260208083206040805160a081018252600590940290910180548452600181015484840181905260028201548584015260038201546060860152600490910154608085015273ffffffffffffffffffffffffffffffffffffffff89168552601990925283205491935011156119d957506020810151611a01565b5073ffffffffffffffffffffffffffffffffffffffff84166000908152601960205260409020545b42826060015111611ac157606082015173ffffffffffffffffffffffffffffffffffffffff86166000908152601960205260409020541015611ab8576000818360600151611a4f9190614dd8565b90506000836060015142611a639190614dd8565b90506000611a75610447600285614e02565b90506000611a838382614deb565b611a8d8584614deb565b611a979190614e3d565b9050611aa38385614e3d565b611aad9082614e02565b965050505050611b11565b60009250611b11565b600080828460600151611ad49190614dd8565b90506000428560600151611ae89190614dd8565b90506002611af68284614e3d565b611b009190614e02565b92505050611b0d81610c4e565b9350505b8160800151831115611b2557816080015192505b505092915050565b60606002805403611b9a576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601f60248201527f5265656e7472616e637947756172643a207265656e7472616e742063616c6c006044820152606401610bb6565b60028055610e87338484613a82565b60256020528160005260406000208181548110611bc557600080fd5b600091825260209091206005909102018054600182015460028301546003840154600490940154929550909350919085565b60015473ffffffffffffffffffffffffffffffffffffffff163314611c9e576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152603560248201527f596f75206d757374206265206e6f6d696e61746564206265666f726520796f7560448201527f2063616e20616363657074206f776e65727368697000000000000000000000006064820152608401610bb6565b6000546001546040805173ffffffffffffffffffffffffffffffffffffffff93841681529290911660208301527fb532073b38c83145e3e5135377a08bf9aab55bc0fd7c1179cd4fb995d2a5159c910160405180910390a160018054600080547fffffffffffffffffffffffff000000000000000000000000000000000000000090811673ffffffffffffffffffffffffffffffffffffffff841617909155169055565b6000670de0b6b3a7640000600b5461166784610fe8565b600054829073ffffffffffffffffffffffffffffffffffffffff16331480611d865750611d86338261102d565b611dec576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601460248201527f4e6f74206f776e6572206f7220746b6e206d67720000000000000000000000006044820152606401610bb6565b73ffffffffffffffffffffffffffffffffffffffff831660009081526014602052604090205460ff16808015611e48575073ffffffffffffffffffffffffffffffffffffffff8481166000908152600f60205260409020541633145b80611e73575080158015611e73575060005473ffffffffffffffffffffffffffffffffffffffff1633145b15611e8957611e83843385613d50565b50505050565b6040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601a60248201527f4e6f2076616c696420746f6b656e7320746f207265636f7665720000000000006044820152606401610bb6565b505050565b73ffffffffffffffffffffffffffffffffffffffff81166000908152601f60205260408120549080611f2184611113565b73ffffffffffffffffffffffffffffffffffffffff85166000908152601e602052604081205491935090158015611f7b575073ffffffffffffffffffffffffffffffffffffffff85166000908152601f6020526040902054155b80611fab575073ffffffffffffffffffffffffffffffffffffffff85166000908152600c60205260409020548310155b15611fb7575081611ff7565b73ffffffffffffffffffffffffffffffffffffffff85166000908152600c6020526040902054600290611fea9085614e3d565b611ff49190614e02565b90505b6000915060005b73ffffffffffffffffffffffffffffffffffffffff861660009081526025602052604090205481101561211b5773ffffffffffffffffffffffffffffffffffffffff8616600090815260256020526040812080548390811061206257612062614d7a565b90600052602060002090600502016040518060a001604052908160008201548152602001600182015481526020016002820154815260200160038201548152602001600482015481525050905060006120bb8884611921565b60408301519091506000670de0b6b3a76400006120d88785614e3d565b6120e29084614deb565b6120ec9190614e02565b6120f69083614e3d565b90506121028188614e3d565b965050505050808061211390614e50565b915050611ffe565b50509193909250565b60005473ffffffffffffffffffffffffffffffffffffffff1633148061215d5750738412ebf45bac1b340bbe8f318b928c466c4e39ca33145b6121c3576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601560248201527f4e6f74206f776e6572206f722074696d656c6f636b00000000000000000000006044820152606401610bb6565b73ffffffffffffffffffffffffffffffffffffffff166000908152600d6020526040902080547fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff00811660ff90911615179055565b6000600354600b546122299190614deb565b6040517f70a0823100000000000000000000000000000000000000000000000000000000815273ffffffffffffffffffffffffffffffffffffffff84166004820152670de0b6b3a764000090819073c8418af6358ffdda74e09ca9cc3fe03ca6adc5b0906370a0823190602401602060405180830381865afa1580156122b3573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906122d79190614e88565b6116679190614deb565b600280540361234c576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601f60248201527f5265656e7472616e637947756172643a207265656e7472616e742063616c6c006044820152606401610bb6565b6002805533600161235f82826000613ec0565b60008061236c338761401a565b9150915060008583604001516123829190614e3d565b90506022546123b3906601000000000000900473ffffffffffffffffffffffffffffffffffffffff163330896141df565b6040805160a08101825288815260208581015181830152818301849052606080870151908301526080808701519083015233600090815260259091529190912080548490811061240557612405614d7a565b9060005260206000209060050201600082015181600001556020820151816001015560408201518160020155606082015181600301556080820151816004015590505061245433876001614375565b604080518881526020810188905233917f2640b32e7e5d0fa2a21ea06b22fbd75fda0fda384a895a5fdeef43646de47a0c910160405180910390a2505060016002555050505050565b6060600280540361250a576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601f60248201527f5265656e7472616e637947756172643a207265656e7472616e742063616c6c006044820152606401610bb6565b6002805561251a33836001613a82565b600160025592915050565b73ffffffffffffffffffffffffffffffffffffffff81166000908152600d602052604081205460ff1615612557575090565b5073ffffffffffffffffffffffffffffffffffffffff9081166000908152602160205260409020541690565b919050565b60028054036125f3576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601f60248201527f5265656e7472616e637947756172643a207265656e7472616e742063616c6c006044820152606401610bb6565b6002805533600161260682826000613ec0565b600080612613338761401a565b91509150428511612680576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601560248201527f4d75737420626520696e207468652066757475726500000000000000000000006044820152606401610bb6565b6000428360600151116126945760006126a4565b4283606001516126a49190614dd8565b905060006126b24288614dd8565b905081811161271d576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601860248201527f43616e6e6f742073686f7274656e206c6f636b2074696d6500000000000000006044820152606401610bb6565b600854811015612789576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601a60248201527f4d696e696d756d207374616b652074696d65206e6f74206d65740000000000006044820152606401610bb6565b6007548111156127f5576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601b60248201527f547279696e6720746f206c6f636b20666f7220746f6f206c6f6e6700000000006044820152606401610bb6565b6040518060a001604052808981526020014281526020018560400151815260200188815260200161282583610c4e565b905233600090815260256020526040902080548590811061284857612848614d7a565b906000526020600020906005020160008201518160000155602082015181600101556040820151816002015560608201518160030155608082015181600401559050506128983360006001613ec0565b6040805189815260208101839052428183015260608101899052905133917fc2cf1aae6decacbc52f96b4e4fec96d4ebab5236e4ed987165537bc463014a43919081900360800190a250506001600255505050505050565b73ffffffffffffffffffffffffffffffffffffffff81166000908152601e6020526040812054600354670de0b6b3a76400009161102391614deb565b600054849073ffffffffffffffffffffffffffffffffffffffff163314806129595750612959338261102d565b6129bf576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601460248201527f4e6f74206f776e6572206f7220746b6e206d67720000000000000000000000006044820152606401610bb6565b73ffffffffffffffffffffffffffffffffffffffff85166000908152601560205260409020546013805486929081106129fa576129fa614d7a565b9060005260206000200181905550826011601560008873ffffffffffffffffffffffffffffffffffffffff1673ffffffffffffffffffffffffffffffffffffffff1681526020019081526020016000205481548110612a5b57612a5b614d7a565b600091825260208083209190910180547fffffffffffffffffffffffff00000000000000000000000000000000000000001673ffffffffffffffffffffffffffffffffffffffff948516179055918716815260159091526040902054601280548492908110612acc57612acc614d7a565b9060005260206000200160006101000a81548173ffffffffffffffffffffffffffffffffffffffff021916908373ffffffffffffffffffffffffffffffffffffffff1602179055505050505050565b73ffffffffffffffffffffffffffffffffffffffff81166000908152600d602052604090205460ff16612baa576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152600d60248201527f496e76616c69642070726f7879000000000000000000000000000000000000006044820152606401610bb6565b73ffffffffffffffffffffffffffffffffffffffff81166000908152600e6020908152604080832033845290915290205460ff16612c44576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601d60248201527f50726f787920686173206e6f7420616c6c6f77656420796f75207965740000006044820152606401610bb6565b3360009081526021602052604090205473ffffffffffffffffffffffffffffffffffffffff168015612cb857336000908152601e602090815260408083205473ffffffffffffffffffffffffffffffffffffffff851684529180528220805491929091612cb2908490614dd8565b90915550505b33600090815260216020908152604080832080547fffffffffffffffffffffffff00000000000000000000000000000000000000001673ffffffffffffffffffffffffffffffffffffffff8716908117909155601e8352818420549084529180528220805491929091612d2c908490614e3d565b90915550505050565b600054829073ffffffffffffffffffffffffffffffffffffffff16331480612d625750612d62338261102d565b612dc8576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601460248201527f4e6f74206f776e6572206f7220746b6e206d67720000000000000000000000006044820152606401610bb6565b5073ffffffffffffffffffffffffffffffffffffffff9182166000908152600f6020526040902080547fffffffffffffffffffffffff00000000000000000000000000000000000000001691909216179055565b60005473ffffffffffffffffffffffffffffffffffffffff16331480612e555750738412ebf45bac1b340bbe8f318b928c466c4e39ca33145b612ebb576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601560248201527f4e6f74206f776e6572206f722074696d656c6f636b00000000000000000000006044820152606401610bb6565b602280547fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff00811660ff90911615179055565b336000908152600d602052604090205460ff16612f66576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152600d60248201527f496e76616c69642070726f7879000000000000000000000000000000000000006044820152606401610bb6565b336000818152600e6020908152604080832073ffffffffffffffffffffffffffffffffffffffff8681168552908352818420805460ff8116157fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff00909116179055602190925290912054160361304a5773ffffffffffffffffffffffffffffffffffffffff8116600090815260216020908152604080832080547fffffffffffffffffffffffff0000000000000000000000000000000000000000169055601e8252808320543384529180528220805491929091613044908490614dd8565b90915550505b50565b6000806011838154811061306357613063614d7a565b60009182526020909120015473ffffffffffffffffffffffffffffffffffffffff169050801561313e57670de0b6b3a7640000601a84815481106130a9576130a9614d7a565b90600052602060002001548273ffffffffffffffffffffffffffffffffffffffff16630a3be7576040518163ffffffff1660e01b8152600401602060405180830381865afa1580156130ff573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906131239190614e88565b61312d9190614deb565b6131379190614e02565b9150610b47565b6013838154811061315157613151614d7a565b9060005260206000200154915050919050565b60005b6011548110156133365760006011828154811061318657613186614d7a565b60009182526020909120015473ffffffffffffffffffffffffffffffffffffffff16905080156133235782806131d85750601b82815481106131ca576131ca614d7a565b906000526020600020015442115b15613323576040517f6472eee100000000000000000000000000000000000000000000000000000000815230600482015242602482015273ffffffffffffffffffffffffffffffffffffffff821690636472eee1906044016020604051808303816000875af115801561324f573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906132739190614e88565b601a838154811061328657613286614d7a565b90600052602060002001819055508073ffffffffffffffffffffffffffffffffffffffff1663513872bd6040518163ffffffff1660e01b8152600401602060405180830381865afa1580156132df573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906133039190614e88565b601b838154811061331657613316614d7a565b6000918252602090912001555b508061332e81614e50565b915050613167565b5050565b60005473ffffffffffffffffffffffffffffffffffffffff163314806133735750738412ebf45bac1b340bbe8f318b928c466c4e39ca33145b6133d9576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601560248201527f4e6f74206f776e6572206f722074696d656c6f636b00000000000000000000006044820152606401610bb6565b8051670de0b6b3a7640000111561344c576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601360248201527f4d757374206265203e3d204d554c2050524543000000000000000000000000006044820152606401610bb6565b6080810151600111801590613466575060a0810151600111155b6134cc576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152600c60248201527f4d757374206265203e3d203100000000000000000000000000000000000000006044820152606401610bb6565b80516006556020810151600a556040810151600b556060810151600955608081015160075560a00151600855565b6135046000613164565b61350c611671565b6003556004544210613522576135206144e8565b565b613520614811565b600081806135435750602254640100000000900460ff16155b1561355a5761355485856001613a82565b50613567565b6135678560016000613ec0565b600080613574878661401a565b91509150816060015142101580613592575060225460ff1615156001145b6135f8576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601660248201527f5374616b65206973207374696c6c206c6f636b656421000000000000000000006044820152606401610bb6565b604082015180156136fd57602254613631906601000000000000900473ffffffffffffffffffffffffffffffffffffffff168883613d50565b73ffffffffffffffffffffffffffffffffffffffff8816600090815260256020526040902080548390811061366857613668614d7a565b6000918252602082206005909102018181556001810182905560028101829055600381018290556004018190556136a29089908390614375565b604080518281526020810188905273ffffffffffffffffffffffffffffffffffffffff898116828401529151918a16917f1d9308f6b22a2754a1c622bb30889e8f8f956c83e524d039e9d65d5f052eb9089181900360600190a25b979650505050505050565b60008183106137175781613719565b825b9392505050565b600085600161373182826000613ec0565b6022546301000000900460ff16156137a5576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152600e60248201527f5374616b696e67207061757365640000000000000000000000000000000000006044820152606401610bb6565b600854851015613811576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601a60248201527f4d696e696d756d207374616b652074696d65206e6f74206d65740000000000006044820152606401610bb6565b60075485111561387d576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601b60248201527f547279696e6720746f206c6f636b20666f7220746f6f206c6f6e6700000000006044820152606401610bb6565b6022546138ac906601000000000000900473ffffffffffffffffffffffffffffffffffffffff168830896141df565b60006138b786610c4e565b73ffffffffffffffffffffffffffffffffffffffff8a166000908152601e602090815260408083205490517fffffffffffffffffffffffffffffffffffffffff00000000000000000000000060608f901b169281019290925260348201899052605482018b9052607482015291925090609401604080517fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe0818403018152828252805160209182012073ffffffffffffffffffffffffffffffffffffffff8e1660009081526025835283902060a0850184528185529184018a90529183018b905290925090606081016139aa8a8a614e3d565b81526020908101859052825460018181018555600094855293829020835160059092020190815590820151818401556040820151600282015560608201516003820155608090910151600490910155613a06908b908a90614375565b604080518981526020810189905290810182905273ffffffffffffffffffffffffffffffffffffffff8a811660608301528b16907ff400e72e69ef4402819dfc57eeddc66f5eb69bf405e0e8098b1946ec1ac14a229060800160405180910390a29998505050505050505050565b60006117bd42600454613708565b6060836001613a9382826000613ec0565b73ffffffffffffffffffffffffffffffffffffffff8616600090815260196020526040902042905560225462010000900460ff1615613b2e576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601960248201527f5265776172647320636f6c6c656374696f6e20706175736564000000000000006044820152606401610bb6565b60105467ffffffffffffffff811115613b4957613b49614ca6565b604051908082528060200260200182016040528015613b72578160200160208202803683370190505b50925060005b601054811015613d465773ffffffffffffffffffffffffffffffffffffffff871660009081526018602090815260408083208484529091529020548451859083908110613bc757613bc7614d7a565b60209081029190910181019190915273ffffffffffffffffffffffffffffffffffffffff881660009081526018825260408082208483529092529081208190558451859083908110613c1b57613c1b614d7a565b60200260200101511115613d3457613c8a60108281548110613c3f57613c3f614d7a565b9060005260206000200160009054906101000a900473ffffffffffffffffffffffffffffffffffffffff1687868481518110613c7d57613c7d614d7a565b6020026020010151613d50565b8673ffffffffffffffffffffffffffffffffffffffff167f1d2f2ca53af5d2f333bd32fdd45f9c52ad8ebe31414f7792912077fcb3876dff858381518110613cd457613cd4614d7a565b602002602001015160108481548110613cef57613cef614d7a565b600091825260209182902001546040805193845273ffffffffffffffffffffffffffffffffffffffff918216928401929092528a169082015260600160405180910390a25b80613d3e81614e50565b915050613b78565b5050509392505050565b6040805173ffffffffffffffffffffffffffffffffffffffff8481166024830152604480830185905283518084039091018152606490920183526020820180517bffffffffffffffffffffffffffffffffffffffffffffffffffffffff167fa9059cbb000000000000000000000000000000000000000000000000000000001790529151600092839290871691613de79190614ec5565b6000604051808303816000865af19150503d8060008114613e24576040519150601f19603f3d011682016040523d82523d6000602084013e613e29565b606091505b5091509150818015613e53575080511580613e53575080806020019051810190613e539190614ee1565b613eb9576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152601f60248201527f5472616e7366657248656c7065723a205452414e534645525f4641494c4544006044820152606401610bb6565b5050505050565b8115613ece57613ece6134fa565b8015613f0457613edd83611113565b73ffffffffffffffffffffffffffffffffffffffff84166000908152600c60205260409020555b73ffffffffffffffffffffffffffffffffffffffff831615611eeb576000806000613f2e86611ef0565b925092509250613f3d86614884565b73ffffffffffffffffffffffffffffffffffffffff86166000908152600c60205260409020829055828110613fc1576000613f788483614dd8565b905080601d54613f889190614e3d565b601d55613f958185614e3d565b73ffffffffffffffffffffffffffffffffffffffff88166000908152601f602052604090205550614012565b6000613fcd8285614dd8565b905080601d54613fdd9190614dd8565b601d55613fea8185614dd8565b73ffffffffffffffffffffffffffffffffffffffff88166000908152601f6020526040902055505b505050505050565b61404f6040518060a0016040528060008019168152602001600081526020016000815260200160008152602001600081525090565b6000805b73ffffffffffffffffffffffffffffffffffffffff851660009081526025602052604090205481101561416d5773ffffffffffffffffffffffffffffffffffffffff851660009081526025602052604090208054829081106140b7576140b7614d7a565b906000526020600020906005020160000154840361415b5773ffffffffffffffffffffffffffffffffffffffff8516600090815260256020526040902080548290811061410657614106614d7a565b90600052602060002090600502016040518060a001604052908160008201548152602001600182015481526020016002820154815260200160038201548152602001600482015481525050925080915061416d565b8061416581614e50565b915050614053565b50815183146141d8576040517f08c379a000000000000000000000000000000000000000000000000000000000815260206004820152600f60248201527f5374616b65206e6f7420666f756e6400000000000000000000000000000000006044820152606401610bb6565b9250929050565b6040805173ffffffffffffffffffffffffffffffffffffffff85811660248301528481166044830152606480830185905283518084039091018152608490920183526020820180517bffffffffffffffffffffffffffffffffffffffffffffffffffffffff167f23b872dd00000000000000000000000000000000000000000000000000000000179052915160009283929088169161427e9190614ec5565b6000604051808303816000865af19150503d80600081146142bb576040519150601f19603f3d011682016040523d82523d6000602084013e6142c0565b606091505b50915091508180156142ea5750805115806142ea5750808060200190518101906142ea9190614ee1565b614012576040517f08c379a0000000000000000000000000000000000000000000000000000000008152602060048201526024808201527f5472616e7366657248656c7065723a205452414e534645525f46524f4d5f464160448201527f494c4544000000000000000000000000000000000000000000000000000000006064820152608401610bb6565b600061438084612525565b905081156144345782601c600082825461439a9190614e3d565b909155505073ffffffffffffffffffffffffffffffffffffffff84166000908152601e6020526040812080548592906143d4908490614e3d565b909155505073ffffffffffffffffffffffffffffffffffffffff81161561442f5773ffffffffffffffffffffffffffffffffffffffff8116600090815260208052604081208054859290614429908490614e3d565b90915550505b6144db565b82601c60008282546144469190614dd8565b909155505073ffffffffffffffffffffffffffffffffffffffff84166000908152601e602052604081208054859290614480908490614dd8565b909155505073ffffffffffffffffffffffffffffffffffffffff8116156144db5773ffffffffffffffffffffffffffffffffffffffff81166000908152602080526040812080548592906144d5908490614dd8565b90915550505b611e838460006001613ec0565b6144f0614811565b60005b6012548110156145e15760006012828154811061451257614512614d7a565b60009182526020909120015473ffffffffffffffffffffffffffffffffffffffff16905080156145ce576040517f092193ab00000000000000000000000000000000000000000000000000000000815230600482015273ffffffffffffffffffffffffffffffffffffffff82169063092193ab9060240160408051808303816000875af11580156145a7573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906145cb9190614efe565b50505b50806145d981614e50565b9150506144f3565b50600062093a80600454426145f69190614dd8565b6146009190614e02565b905060005b6010548110156147e3576010818154811061462257614622614d7a565b6000918252602090912001546040517f70a0823100000000000000000000000000000000000000000000000000000000815230600482015273ffffffffffffffffffffffffffffffffffffffff909116906370a0823190602401602060405180830381865afa158015614699573d6000803e3d6000fd5b505050506040513d601f19601f820116820180604052508101906146bd9190614e88565b6146c8836001614e3d565b62093a806146d58461304d565b6146df9190614deb565b6146e99190614deb565b1115601082815481106146fe576146fe614d7a565b60009182526020918290200154604080517f4e6f7420656e6f7567682072657761726420746f6b656e7320617661696c6162938101939093527f6c653a20000000000000000000000000000000000000000000000000000000009083015260601b7fffffffffffffffffffffffffffffffffffffffff000000000000000000000000166044820152605801604051602081830303815290604052906147d0576040517f08c379a0000000000000000000000000000000000000000000000000000000008152600401610bb69190614f22565b50806147db81614e50565b915050614605565b5062093a806147f3826001614e3d565b6147fd9190614deb565b60045461480a9190614e3d565b60045561304a5b600061481b6117c2565b905060005b6016548110156148755781818151811061483c5761483c614d7a565b60200260200101516016828154811061485757614857614d7a565b6000918252602090912001558061486d81614e50565b915050614820565b5061487e613a74565b60055550565b73ffffffffffffffffffffffffffffffffffffffff81161561304a5760006148ab826109ce565b905060005b8151811015614919578181815181106148cb576148cb614d7a565b60209081029190910181015173ffffffffffffffffffffffffffffffffffffffff8516600090815260188352604080822085835290935291909120558061491181614e50565b9150506148b0565b5060005b8151811015611eeb576016818154811061493957614939614d7a565b600091825260208083209091015473ffffffffffffffffffffffffffffffffffffffff861683526017825260408084208585529092529120558061497c81614e50565b91505061491d565b803573ffffffffffffffffffffffffffffffffffffffff8116811461258357600080fd5b6000602082840312156149ba57600080fd5b61371982614984565b6020808252825182820181905260009190848201906040850190845b818110156149fb578351835292840192918401916001016149df565b50909695505050505050565b801515811461304a57600080fd5b600080600060608486031215614a2a57600080fd5b83359250614a3a60208501614984565b91506040840135614a4a81614a07565b809150509250925092565b600060208284031215614a6757600080fd5b5035919050565b6020808252825182820181905260009190848201906040850190845b818110156149fb57835173ffffffffffffffffffffffffffffffffffffffff1683529284019291840191600101614a8a565b60008060408385031215614acf57600080fd5b50508035926020909101359150565b602080825282518282018190526000919060409081850190868401855b82811015614b3f5781518051855286810151878601528581015186860152606080820151908601526080908101519085015260a09093019290850190600101614afb565b5091979650505050505050565b60008060408385031215614b5f57600080fd5b614b6883614984565b9150614b7660208401614984565b90509250929050565b60008060008060808587031215614b9557600080fd5b8435614ba081614a07565b93506020850135614bb081614a07565b92506040850135614bc081614a07565b91506060850135614bd081614a07565b939692955090935050565b60008060408385031215614bee57600080fd5b614bf783614984565b946020939093013593505050565b60008060408385031215614c1857600080fd5b614c2183614984565b91506020830135614c3181614a07565b809150509250929050565b60008060008060808587031215614c5257600080fd5b614c5b85614984565b935060208501359250614c7060408601614984565b9150614c7e60608601614984565b905092959194509250565b600060208284031215614c9b57600080fd5b813561371981614a07565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052604160045260246000fd5b600060c08284031215614ce757600080fd5b82601f830112614cf657600080fd5b60405160c0810181811067ffffffffffffffff82111715614d40577f4e487b7100000000000000000000000000000000000000000000000000000000600052604160045260246000fd5b6040528060c0840185811115614d5557600080fd5b845b81811015614d6f578035835260209283019201614d57565b509195945050505050565b7f4e487b7100000000000000000000000000000000000000000000000000000000600052603260045260246000fd5b7f4e487b7100000000000000000000000000000000000000000000000000000000600052601160045260246000fd5b81810381811115610c7457610c74614da9565b8082028115828204841417610c7457610c74614da9565b600082614e38577f4e487b7100000000000000000000000000000000000000000000000000000000600052601260045260246000fd5b500490565b80820180821115610c7457610c74614da9565b60007fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff8203614e8157614e81614da9565b5060010190565b600060208284031215614e9a57600080fd5b5051919050565b60005b83811015614ebc578181015183820152602001614ea4565b50506000910152565b60008251614ed7818460208701614ea1565b9190910192915050565b600060208284031215614ef357600080fd5b815161371981614a07565b60008060408385031215614f1157600080fd5b505080516020909101519092909150565b6020815260008251806020840152614f41816040850160208701614ea1565b601f017fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe016919091016040019291505056fea264697066735822122079a11f13c04fb3948cc8f5871bc318534a596accdd8782fcfaa7c2474ca9ae3364736f6c63430008110033

Deployed Bytecode Sourcemap

128837:1488:0:-:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;78559:529;;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;:::-;;;;;;;;125934:294;;;;;;:::i;:::-;;:::i;:::-;;;1700:25:1;;;1688:2;1673:18;125934:294:0;1554:177:1;81457:222:0;;;;;;:::i;:::-;;:::i;76561:109::-;;;:::i;:::-;;;;;;;:::i;62514:141::-;;;;;;:::i;:::-;;:::i;:::-;;124018:190;;;;;;:::i;:::-;;:::i;79267:339::-;;;:::i;118700:133::-;;;;;;:::i;:::-;;:::i;:::-;;;;;;;:::i;82289:179::-;;;;;;:::i;:::-;;:::i;108111:23::-;;;;;;;;;;;;4249:42:1;4237:55;;;4219:74;;4207:2;4192:18;108111:23:0;4058:241:1;75973:459:0;;;;;;:::i;:::-;;:::i;:::-;;;4734:14:1;;4727:22;4709:41;;4697:2;4682:18;75973:459:0;4569:187:1;72818:60:0;;;;;;:::i;:::-;;;;;;;;;;;;;;;;84816:1752;;;;;;:::i;:::-;;:::i;100248:439::-;;;;;;:::i;:::-;;:::i;80668:127::-;;;;;;:::i;:::-;80761:26;;80734:7;80761:26;;;:17;:26;;;;;;;80668:127;71654:48;;71696:6;71654:48;;91303:238;;;;;;:::i;:::-;;:::i;71243:49::-;;;;;;:::i;:::-;;;;;;;;;;;;;;;;108075:29;;;;;;;;;83451:181;;;;;;:::i;:::-;;:::i;62282:29::-;;;;;;;;;129808:514;;;:::i;80276:111::-;80357:22;;80276:111;;71564:55;;;;;;:::i;:::-;;;;;;;;;;;;;;72046:54;;;;;;:::i;:::-;;;;;;;;;;;;;;70473:37;;;;;;77566:681;;;:::i;108023:45::-;;;;;;;;;;;;113891:2692;;;;;;:::i;:::-;;:::i;92489:206::-;;;;;;:::i;:::-;;:::i;108774:53::-;;;;;;:::i;:::-;;:::i;:::-;;;;6998:25:1;;;7054:2;7039:18;;7032:34;;;;7082:18;;;7075:34;;;;7140:2;7125:18;;7118:34;7183:3;7168:19;;7161:35;6985:3;6970:19;108774:53:0;6739:463:1;62663:271:0;;;:::i;72613:52::-;;;;;;:::i;:::-;;;;;;;;;;;;;;83828:199;;;;;;:::i;:::-;;:::i;101483:818::-;;;;;;:::i;:::-;;:::i;116644:1961::-;;;;;;:::i;:::-;;:::i;:::-;;;;7409:25:1;;;7465:2;7450:18;;7443:34;;;;7493:18;;;7486:34;7397:2;7382:18;116644:1961:0;7207:319:1;62255:20:0;;;;;;;;;101087:161;;;;;;:::i;:::-;;:::i;72940:26::-;;;;;;;;;82745:224;;;;;;:::i;:::-;;:::i;70567:55::-;;;;;;121459:1006;;;;;;:::i;:::-;;:::i;70318:62::-;;;;;;92027:172;;;;;;:::i;:::-;;:::i;84180:444::-;;;;;;:::i;:::-;;:::i;70164:29::-;;;;;;118909:133;;;;;;:::i;:::-;119006:21;;118979:7;119006:21;;;:12;:21;;;;;:28;;118909:133;70244:50;;;;;;69837:30;;;;;;122519:1364;;;;;;:::i;:::-;;:::i;81914:166::-;;;;;;:::i;:::-;;:::i;104276:483::-;;;;;;:::i;:::-;;:::i;87588:800::-;;;;;;:::i;:::-;;:::i;80034:128::-;;;;;;:::i;:::-;80128:26;;80101:7;80128:26;;;:17;:26;;;;;;;80034:128;105079:207;;;;;;:::i;:::-;;:::i;79769:113::-;79851:23;;79769:113;;100851:97;;;:::i;86822:610::-;;;;;;:::i;:::-;;:::i;70057:27::-;;;;;;70705:51;;;;;;70781:59;;;;;;77044:452;;;;;;:::i;:::-;;:::i;98423:761::-;;;;;;:::i;:::-;;:::i;102764:958::-;;;;;;:::i;:::-;;:::i;99283:363::-;;;:::i;78559:529::-;78613:27;78653;78683:17;:15;:17::i;:::-;78738:12;:19;78653:47;;-1:-1:-1;78724:34:0;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;78724:34:0;-1:-1:-1;78775:26:0;;;78804:1;78775:26;;;:17;:26;;;;;;78711:47;;-1:-1:-1;78775:30:0;78771:310;;78826:9;78821:249;78845:12;:19;78841:23;;78821:249;;;79035:16;;;;;;;:7;:16;;;;;;;;:19;;;;;;;;;78954:32;;;:23;:32;;;;;:35;;;;;;;;;78938:13;;78994:4;;78954:35;78938:10;;79052:1;;78938:13;;;;;;:::i;:::-;;;;;;;:51;;;;:::i;:::-;78908:26;;;;;;;:17;:26;;;;;;:82;;;;:::i;:::-;78907:91;;;;:::i;:::-;78906:148;;;;:::i;:::-;78890:10;78901:1;78890:13;;;;;;;;:::i;:::-;;;;;;;;;;:164;78866:3;;;;:::i;:::-;;;;78821:249;;;;78771:310;78642:446;78559:529;;;:::o;125934:294::-;126054:7;66429:1;67035:7;;:19;67027:63;;;;;;;11042:2:1;67027:63:0;;;11024:21:1;11081:2;11061:18;;;11054:30;11120:33;11100:18;;;11093:61;11171:18;;67027:63:0;;;;;;;;;66429:1;67168:18;;126082:17:::1;::::0;::::1;::::0;::::1;;;:26;126074:57;;;::::0;::::1;::::0;;11402:2:1;126074:57:0::1;::::0;::::1;11384:21:1::0;11441:2;11421:18;;;11414:30;11480:20;11460:18;;;11453:48;11518:18;;126074:57:0::1;11200:342:1::0;126074:57:0::1;126149:71;126165:10;126177:19;126198:6;126206:13;126149:15;:71::i;:::-;66385:1:::0;67347:7;:22;126142:78;125934:294;-1:-1:-1;;;;125934:294:0:o;81457:222::-;81566:19;;81631:28;;81516:7;;81543:127;;81601:26;81566:19;81601:4;:26;:::i;:::-;81600:59;;;;:::i;:::-;81543:8;:127::i;:::-;81536:134;81457:222;-1:-1:-1;;81457:222:0:o;76561:109::-;76614:16;76650:12;76643:19;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;76561:109;:::o;62514:141::-;62994:5;;;;62980:10;:19;62972:79;;;;;;;11749:2:1;62972:79:0;;;11731:21:1;11788:2;11768:18;;;11761:30;11827:34;11807:18;;;11800:62;11898:17;11878:18;;;11871:45;11933:19;;62972:79:0;11547:411:1;62972:79:0;62586:14:::1;:23:::0;;;::::1;;::::0;::::1;::::0;;::::1;::::0;;;62625:22:::1;::::0;4219:74:1;;;62625:22:0::1;::::0;4207:2:1;4192:18;62625:22:0::1;;;;;;;62514:141:::0;:::o;124018:190::-;124103:7;66429:1;67035:7;;:19;67027:63;;;;;;;11042:2:1;67027:63:0;;;11024:21:1;11081:2;11061:18;;;11054:30;11120:33;11100:18;;;11093:61;11171:18;;67027:63:0;10840:355:1;67027:63:0;66429:1;67168:18;;124130:70:::1;124143:10;::::0;124167:9;124178:4;124184:15:::1;124130:12;:70::i;:::-;66385:1:::0;67347:7;:22;124123:77;124018:190;-1:-1:-1;;;124018:190:0:o;79267:339::-;79417:17;:24;79322:41;;79403:39;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;79403:39:0;;79376:66;;79460:9;79455:144;79479:17;:24;79475:28;;79455:144;;;71696:6;79555:14;79567:1;79555:11;:14::i;:::-;:32;;;;:::i;:::-;79525:24;79550:1;79525:27;;;;;;;;:::i;:::-;;;;;;;;;;:62;79505:3;;;;:::i;:::-;;;;79455:144;;;;79267:339;:::o;118700:133::-;118804:21;;;;;;;:12;:21;;;;;;;;118797:28;;;;;;;;;;;;;;;;;118764:20;;118797:28;;118804:21;;118797:28;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;118700:133;;;:::o;82289:179::-;82404:32;;;82358:7;82404:32;;;:17;:32;;;;;;82386:15;;69965:4;;82386:50;;;:::i;:::-;82385:75;;;;:::i;75973:459::-;76090:32;;;76069:4;76090:32;;;:13;:32;;;;;;;;76085:297;;-1:-1:-1;76131:5:0;76124:12;;76085:297;76156:25;;;;;:60;;-1:-1:-1;76185:31:0;;;;76156:60;76152:230;;;-1:-1:-1;76225:5:0;76218:12;;76152:230;76265:5;;;;;;76250:20;;;;76246:136;;-1:-1:-1;76279:4:0;76272:11;;76246:136;76321:48;:33;;;;;;;:14;:33;;;;;;:48;;;:33;;:48;76317:65;;-1:-1:-1;76378:4:0;76371:11;;76317:65;-1:-1:-1;76418:5:0;75973:459;;;;:::o;84816:1752::-;84879:24;;;85040:20;85052:7;85040:11;:20::i;:::-;85020:40;-1:-1:-1;85091:23:0;;;;85090:81;;85145:26;;;;;4249:42:1;4237:55;;85145:26:0;;;4219:74:1;69572:42:0;;85145:15;;4192:18:1;;85145:26:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;85090:81;;;85118:24;;;;;4249:42:1;4237:55;;85118:24:0;;;4219:74:1;69572:42:0;;85118:15;;4192:18:1;;85118:24:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;85071:100;;85277:19;69965:4;69572:42;85401:17;;;:19;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;:42;;;;:::i;:::-;85342:24;;85319:20;;85300:39;;:16;:39;:::i;:::-;:66;;;;:::i;:::-;85299:145;;;;:::i;:::-;85277:167;-1:-1:-1;85553:19:0;;85768:23;;;;85767:94;;85826:35;85851:9;85826:24;:35::i;:::-;85767:94;;;85795:28;85815:7;85795:19;:28::i;:::-;85738:123;-1:-1:-1;85882:30:0;;85878:323;;85933:27;86007:26;85964:39;69965:4;85964:16;:39;:::i;:::-;85963:70;;;;:::i;:::-;85933:100;;69965:4;86107:20;;86085:19;:42;;;;:::i;:::-;86084:67;;;;:::i;:::-;86070:81;;85913:254;85878:323;;;86200:1;86186:15;;85878:323;85583:699;86369:11;86355;:25;:53;;86397:11;86355:53;;;86383:11;86355:53;86335:74;;86499:20;;86480:16;:39;86476:84;;;86540:20;;86521:39;;86476:84;84905:1663;;;;84816:1752;;;:::o;100248:439::-;73462:5;;;;73448:10;:19;;:79;;-1:-1:-1;73485:42:0;73471:10;:56;73448:79;73440:113;;;;;;;12354:2:1;73440:113:0;;;12336:21:1;12393:2;12373:18;;;12366:30;12432:23;12412:18;;;12405:51;12473:18;;73440:113:0;12152:345:1;73440:113:0;100460:13:::1;:30:::0;;100501:38;;100460:30;;::::1;;::::0;;;::::1;100501:38:::0;;;;;;100460:30:::1;100501:38:::0;::::1;;::::0;;;::::1;::::0;;;::::1;100611:68:::0;;100550:50;;::::1;;::::0;;;::::1;100611:68:::0;;;;;::::1;;::::0;;;::::1;;::::0;;100248:439::o;91303:238::-;66429:1;67035:7;;:19;67027:63;;;;;;;11042:2:1;67027:63:0;;;11024:21:1;11081:2;11061:18;;;11054:30;11120:33;11100:18;;;11093:61;11171:18;;67027:63:0;10840:355:1;67027:63:0;66429:1;67168:18;;91400:23:::1;::::0;;;::::1;;;:32;91392:70;;;::::0;::::1;::::0;;12704:2:1;91392:70:0::1;::::0;::::1;12686:21:1::0;12743:2;12723:18;;;12716:30;12782:27;12762:18;;;12755:55;12827:18;;91392:70:0::1;12502:349:1::0;91392:70:0::1;-1:-1:-1::0;66385:1:0;67347:7;:22;91303:238::o;83451:181::-;83518:7;69965:4;83572:28;;83546:23;83561:7;83546:14;:23::i;:::-;:54;;;;:::i;129808:514::-;130275:9;;:24;;;;;;;;129864:25;;130275:9;;;:22;;:24;;;;;;;;;;;;;;:9;:24;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;130271:28;;:1;:28;:::i;:::-;130237:9;;;;;;;;;;;:18;;;:20;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;:29;;130261:4;130237:29;:::i;:::-;130236:64;;;;:::i;:::-;130216:84;;129808:514;:::o;77566:681::-;77614:41;77672:23;;77699:1;77672:28;:59;;;-1:-1:-1;77704:22:0;;:27;77672:59;77668:572;;;77755:21;77748:28;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;77566:681;:::o;77668:572::-;77859:12;:19;77845:34;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;77845:34:0;;77818:61;;77899:9;77894:289;77918:21;:28;77914:32;;77894:289;;;78126:22;;78101:14;78113:1;78101:11;:14::i;:::-;78083;;78054:26;:24;:26::i;:::-;:43;;;;:::i;:::-;78053:62;;;;:::i;:::-;:69;;78118:4;78053:69;:::i;:::-;78052:96;;;;:::i;:::-;78002:21;78024:1;78002:24;;;;;;;;:::i;:::-;;;;;;;;;:165;;;;:::i;:::-;77972:24;77997:1;77972:27;;;;;;;;:::i;:::-;;;;;;;;;;:195;77948:3;;;;:::i;:::-;;;;77894:289;;113891:2692;114082:21;;;113980:32;114082:21;;;:12;:21;;;;;:32;;113980;;114082:21;114104:9;;114082:32;;;;;;:::i;:::-;;;;;;;;;114051:63;;;;;;;;114082:32;;;;;;;114051:63;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;114297:28;;;;;:19;:28;;;;;;114051:63;;-1:-1:-1;;114293:224:0;;;-1:-1:-1;114390:25:0;;;;114293:224;;;-1:-1:-1;114477:28:0;;;;;;;:19;:28;;;;;;114293:224;114606:15;114576:9;:26;;;:45;114572:1802;;114801:26;;;;114770:28;;;;;;;:19;:28;;;;;;:57;114766:1087;;;114847:26;114905:17;114876:9;:26;;;:46;;;;:::i;:::-;114847:75;;114941:25;114987:9;:26;;;114969:15;:44;;;;:::i;:::-;114941:72;-1:-1:-1;115093:33:0;115129:38;115144:22;115165:1;115144:18;:22;:::i;115129:38::-;115093:74;-1:-1:-1;115384:17:0;115456:21;115460:17;115384;115456:21;:::i;:::-;115405:46;115433:18;115405:25;:46;:::i;:::-;115404:74;;;;:::i;:::-;115384:94;-1:-1:-1;115537:38:0;115558:17;115537:18;:38;:::i;:::-;115524:52;;:9;:52;:::i;:::-;115497:79;;114828:764;;;;114572:1802;;114766:1087;115836:1;115809:28;;114572:1802;;;115998:21;116053:20;116105:17;116076:9;:26;;;:46;;;;:::i;:::-;116053:69;;116141:20;116193:15;116164:9;:26;;;:44;;;;:::i;:::-;116141:67;-1:-1:-1;116275:1:0;116244:27;116141:67;116244:12;:27;:::i;:::-;116243:33;;;;:::i;:::-;116227:49;;116034:258;;116333:29;116348:13;116333:14;:29::i;:::-;116306:56;;115918:456;114572:1802;116496:9;:25;;;116469:24;:52;116465:110;;;116550:9;:25;;;116523:52;;116465:110;114014:2569;;113891:2692;;;;:::o;92489:206::-;92591:16;66429:1;67035:7;;:19;67027:63;;;;;;;11042:2:1;67027:63:0;;;11024:21:1;11081:2;11061:18;;;11054:30;11120:33;11100:18;;;11093:61;11171:18;;67027:63:0;10840:355:1;67027:63:0;66429:1;67168:18;;92627:60:::1;92638:10;92650:19:::0;92671:15;92627:10:::1;:60::i;108774:53::-:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;108774:53:0;;-1:-1:-1;108774:53:0;;;:::o;62663:271::-;62732:14;;;;62718:10;:28;62710:94;;;;;;;13058:2:1;62710:94:0;;;13040:21:1;13097:2;13077:18;;;13070:30;13136:34;13116:18;;;13109:62;13207:23;13187:18;;;13180:51;13248:19;;62710:94:0;12856:417:1;62710:94:0;62833:5;;;62840:14;62820:35;;;62833:5;;;;13513:34:1;;62840:14:0;;;;13578:2:1;13563:18;;13556:43;62820:35:0;;13425:18:1;62820:35:0;;;;;;;62874:14;;;;62866:22;;;;;;62874:14;;;62866:22;;;;62899:27;;;62663:271::o;83828:199::-;83906:7;69965:4;83967:28;;83934:30;83950:13;83934:15;:30::i;101483:818::-;73665:5;;101569:12;;73665:5;;73651:10;:19;;:74;;;73674:51;73692:10;73704:20;73674:17;:51::i;:::-;73643:107;;;;;;;13812:2:1;73643:107:0;;;13794:21:1;13851:2;13831:18;;;13824:30;13890:22;13870:18;;;13863:50;13930:18;;73643:107:0;13610:344:1;73643:107:0;101667:27:::1;::::0;::::1;101651:13;101667:27:::0;;;:13:::1;:27;::::0;;;;;::::1;;::::0;101921:54;::::1;;;-1:-1:-1::0;101933:42:0::1;:28:::0;;::::1;;::::0;;;:14:::1;:28;::::0;;;;;::::1;101965:10;101933:42;101921:54;101920:113;;;;101999:8;101998:9;:34;;;;-1:-1:-1::0;102026:5:0::1;::::0;::::1;;102012:10;:19;101998:34;101898:396;;;102064:66;102092:12;102106:10;102118:11;102064:27;:66::i;:::-;102145:7;101483:818:::0;;;:::o;101898:396::-:1;102246:36;::::0;::::1;::::0;;14161:2:1;102246:36:0::1;::::0;::::1;14143:21:1::0;14200:2;14180:18;;;14173:30;14239:28;14219:18;;;14212:56;14285:18;;102246:36:0::1;13959:350:1::0;73761:1:0::1;101483:818:::0;;;:::o;116644:1961::-;116947:26;;;116745:27;116947:26;;;:17;:26;;;;;;;116745:27;117134:24;116965:7;117134:15;:24::i;:::-;117234:26;;;117171:33;117234:26;;;:17;:26;;;;;;117111:47;;-1:-1:-1;117171:33:0;117234:31;:66;;;;-1:-1:-1;117269:26:0;;;;;;;:17;:26;;;;;;:31;117234:66;117233:143;;;-1:-1:-1;117344:31:0;;;;;;;:22;:31;;;;;;117320:55;;;117233:143;117215:638;;;-1:-1:-1;117617:20:0;117215:638;;;117805:31;;;;;;;:22;:31;;;;;;117840:1;;117782:54;;:20;:54;:::i;:::-;117781:60;;;;:::i;:::-;117753:88;;117215:638;117988:1;117966:23;;118005:9;118000:598;118024:21;;;;;;;:12;:21;;;;;:28;118020:32;;118000:598;;;118105:21;;;118074:28;118105:21;;;:12;:21;;;;;:24;;118127:1;;118105:24;;;;;;:::i;:::-;;;;;;;;;;;118074:55;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;118201:32;118236:34;118259:7;118268:1;118236:22;:34::i;:::-;118352:19;;;;118201:69;;-1:-1:-1;118332:17:0;69965:4;118447:52;118474:25;118201:69;118447:52;:::i;:::-;118434:66;;:9;:66;:::i;:::-;118433:91;;;;:::i;:::-;118420:105;;:9;:105;:::i;:::-;118386:139;-1:-1:-1;118540:46:0;118386:139;118540:46;;:::i;:::-;;;118059:539;;;;118054:3;;;;;:::i;:::-;;;;118000:598;;;;116874:1731;116644:1961;;;;;:::o;101087:161::-;73462:5;;;;73448:10;:19;;:79;;-1:-1:-1;73485:42:0;73471:10;:56;73448:79;73440:113;;;;;;;12354:2:1;73440:113:0;;;12336:21:1;12393:2;12373:18;;;12366:30;12432:23;12412:18;;;12405:51;12473:18;;73440:113:0;12152:345:1;73440:113:0;101208:32:::1;;;::::0;;;:19:::1;:32;::::0;;;;;;101172:68;;::::1;101208:32;::::0;;::::1;101207:33;101172:68;::::0;;101087:161::o;82745:224::-;82811:7;82945:15;;82914:28;;:46;;;;:::i;:::-;82839:24;;;;;4249:42:1;4237:55;;82839:24:0;;;4219:74:1;69965:4:0;;;;69572:42;;82839:15;;4192:18:1;;82839:24:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;:47;;;;:::i;121459:1006::-;66429:1;67035:7;;:19;67027:63;;;;;;;11042:2:1;67027:63:0;;;11024:21:1;11081:2;11061:18;;;11054:30;11120:33;11100:18;;;11093:61;11171:18;;67027:63:0;10840:355:1;67027:63:0;66429:1;67168:18;;121556:10:::1;121568:4;73856:49;121556:10:::0;121568:4;73899:5:::1;73856:23;:49::i;:::-;121633:28:::2;121663:21:::0;121688:29:::2;121698:10;121710:6;121688:9;:29::i;:::-;121632:85;;;;121767:15;121807:8;121785:9;:19;;;:30;;;;:::i;:::-;121767:48:::0;-1:-1:-1;121986:12:0::2;::::0;121946:91:::2;::::0;121986:12;;::::2;;;122001:10;122021:4;122028:8:::0;121946:31:::2;:91::i;:::-;122121:186;::::0;;::::2;::::0;::::2;::::0;;;;;::::2;122168:25:::0;;::::2;::::0;122121:186;;::::2;::::0;;;;;;;;122230:26;;::::2;::::0;122121:186;;;;;122271:25;;::::2;::::0;122121:186;;;;122092:10:::2;-1:-1:-1::0;122079:24:0;;;:12:::2;:24:::0;;;;;;;:39;;122104:13;;122079:39;::::2;;;;;:::i;:::-;;;;;;;;;;;:228;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;122351:42;122366:10;122378:8;122388:4;122351:14;:42::i;:::-;122411:46;::::0;;14833:25:1;;;14889:2;14874:18;;14867:34;;;122428:10:0::2;::::0;122411:46:::2;::::0;14806:18:1;122411:46:0::2;;;;;;;-1:-1:-1::0;;66385:1:0;67347:7;:22;-1:-1:-1;;;;;121459:1006:0:o;92027:172::-;92106:16;66429:1;67035:7;;:19;67027:63;;;;;;;11042:2:1;67027:63:0;;;11024:21:1;11081:2;11061:18;;;11054:30;11120:33;11100:18;;;11093:61;11171:18;;67027:63:0;10840:355:1;67027:63:0;66429:1;67168:18;;92142:49:::1;92153:10;92165:19:::0;92186:4:::1;92142:10;:49::i;:::-;66385:1:::0;67347:7;:22;92135:56;92027:172;-1:-1:-1;;92027:172:0:o;84180:444::-;84259:25;;;84236:7;84259:25;;;:19;:25;;;;;;;;84255:362;;;-1:-1:-1;84462:4:0;84180:444::o;84255:362::-;-1:-1:-1;84574:31:0;;;;;;;;:25;:31;;;;;;;;84180:444::o;84255:362::-;84180:444;;;:::o;122519:1364::-;66429:1;67035:7;;:19;67027:63;;;;;;;11042:2:1;67027:63:0;;;11024:21:1;11081:2;11061:18;;;11054:30;11120:33;11100:18;;;11093:61;11171:18;;67027:63:0;10840:355:1;67027:63:0;66429:1;67168:18;;122617:10:::1;122629:4;73856:49;122617:10:::0;122629:4;73899:5:::1;73856:23;:49::i;:::-;122694:28:::2;122724:21:::0;122749:29:::2;122759:10;122771:6;122749:9;:29::i;:::-;122693:85;;;;122833:15;122817:13;:31;122809:65;;;::::0;::::2;::::0;;15114:2:1;122809:65:0::2;::::0;::::2;15096:21:1::0;15153:2;15133:18;;;15126:30;15192:23;15172:18;;;15165:51;15233:18;;122809:65:0::2;14912:345:1::0;122809:65:0::2;122920:17;122970:15;122941:9;:26;;;:44;122940:97;;123036:1;122940:97;;;123018:15;122989:9;:26;;;:44;;;;:::i;:::-;122920:117:::0;-1:-1:-1;123048:16:0::2;123067:31;123083:15;123067:13:::0;:31:::2;:::i;:::-;123048:50;;123204:9;123193:8;:20;123185:57;;;::::0;::::2;::::0;;15464:2:1;123185:57:0::2;::::0;::::2;15446:21:1::0;15503:2;15483:18;;;15476:30;15542:26;15522:18;;;15515:54;15586:18;;123185:57:0::2;15262:348:1::0;123185:57:0::2;123273:13;;123261:8;:25;;123253:64;;;::::0;::::2;::::0;;15817:2:1;123253:64:0::2;::::0;::::2;15799:21:1::0;15856:2;15836:18;;;15829:30;15895:28;15875:18;;;15868:56;15941:18;;123253:64:0::2;15615:350:1::0;123253:64:0::2;123348:28;;123336:8;:40;;123328:80;;;::::0;::::2;::::0;;16172:2:1;123328:80:0::2;::::0;::::2;16154:21:1::0;16211:2;16191:18;;;16184:30;16250:29;16230:18;;;16223:57;16297:18;;123328:80:0::2;15970:351:1::0;123328:80:0::2;123492:174;;;;;;;;123518:6;123492:174;;;;123539:15;123492:174;;;;123569:9;:19;;;123492:174;;;;123603:13;123492:174;;;;123631:24;123646:8;123631:14;:24::i;:::-;123492:174:::0;;123463:10:::2;123450:24;::::0;;;:12:::2;:24;::::0;;;;:39;;123475:13;;123450:39;::::2;;;;;:::i;:::-;;;;;;;;;;;:216;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;123735:48;123759:10;123771:5;123778:4;123735:23;:48::i;:::-;123801:74;::::0;;16557:25:1;;;16613:2;16598:18;;16591:34;;;123844:15:0::2;16641:18:1::0;;;16634:34;16699:2;16684:18;;16677:34;;;123801:74:0;;123814:10:::2;::::0;123801:74:::2;::::0;;;;;16544:3:1;123801:74:0;;::::2;-1:-1:-1::0;;66385:1:0;67347:7;:22;-1:-1:-1;;;;;;122519:1364:0:o;81914:166::-;82022:26;;;81976:7;82022:26;;;:17;:26;;;;;;82004:15;;69965:4;;82004:44;;;:::i;104276:483::-;73665:5;;104442:20;;73665:5;;73651:10;:19;;:74;;;73674:51;73692:10;73704:20;73674:17;:51::i;:::-;73643:107;;;;;;;13812:2:1;73643:107:0;;;13794:21:1;13851:2;13831:18;;;13824:30;13890:22;13870:18;;;13863:50;13930:18;;73643:107:0;13610:344:1;73643:107:0;104493:42:::1;::::0;::::1;;::::0;;;:20:::1;:42;::::0;;;;;104475:17:::1;:61:::0;;104539:9;;104493:42;104475:61;::::1;;;;;:::i;:::-;;;;;;;;:73;;;;104622:25;104559:16;104576:20;:42;104597:20;104576:42;;;;;;;;;;;;;;;;104559:60;;;;;;;;:::i;:::-;;::::0;;;::::1;::::0;;;;;;::::1;:88:::0;;;::::1;;::::0;;::::1;;::::0;;104677:42;;::::1;::::0;;:20:::1;:42:::0;;;;;;;104658:18:::1;:62:::0;;104723:28;;104677:42;104658:62;::::1;;;;;:::i;:::-;;;;;;;;;:93;;;;;;;;;;;;;;;;;;104276:483:::0;;;;;:::o;87588:800::-;87668:34;;;;;;;:19;:34;;;;;;;;87660:60;;;;;;;16924:2:1;87660:60:0;;;16906:21:1;16963:2;16943:18;;;16936:30;17002:15;16982:18;;;16975:43;17035:18;;87660:60:0;16722:337:1;87660:60:0;87739:36;;;;;;;:21;:36;;;;;;;;87776:10;87739:48;;;;;;;;;;87731:90;;;;;;;17266:2:1;87731:90:0;;;17248:21:1;17305:2;17285:18;;;17278:30;17344:31;17324:18;;;17317:59;17393:18;;87731:90:0;17064:353:1;87731:90:0;87967:10;87916:22;87941:37;;;:25;:37;;;;;;;;87993:28;;87989:182;;88148:10;88130:29;;;;:17;:29;;;;;;;;;;88093:33;;;;;;;;;:66;;88130:29;;88093:33;;:66;;88130:29;;88093:66;:::i;:::-;;;;-1:-1:-1;;87989:182:0;88239:10;88213:37;;;;:25;:37;;;;;;;;:53;;;;:37;:53;;;;;;;;88351:17;:29;;;;;;88315:32;;;;;;;;:65;;88351:29;;88315:32;;:65;;88351:29;;88315:65;:::i;:::-;;;;-1:-1:-1;;;;87588:800:0:o;105079:207::-;73665:5;;105187:20;;73665:5;;73651:10;:19;;:74;;;73674:51;73692:10;73704:20;73674:17;:51::i;:::-;73643:107;;;;;;;13812:2:1;73643:107:0;;;13794:21:1;13851:2;13831:18;;;13824:30;13890:22;13870:18;;;13863:50;13930:18;;73643:107:0;13610:344:1;73643:107:0;-1:-1:-1;105220:36:0::1;::::0;;::::1;;::::0;;;:14:::1;:36;::::0;;;;:58;;;::::1;::::0;;;::::1;;::::0;;105079:207::o;100851:97::-;73462:5;;;;73448:10;:19;;:79;;-1:-1:-1;73485:42:0;73471:10;:56;73448:79;73440:113;;;;;;;12354:2:1;73440:113:0;;;12336:21:1;12393:2;12373:18;;;12366:30;12432:23;12412:18;;;12405:51;12473:18;;73440:113:0;12152:345:1;73440:113:0;100926:14:::1;::::0;;100908:32;;::::1;100926:14;::::0;;::::1;100925:15;100908:32;::::0;;100851:97::o;86822:610::-;86921:10;86901:31;;;;:19;:31;;;;;;;;86893:57;;;;;;;16924:2:1;86893:57:0;;;16906:21:1;16963:2;16943:18;;;16936:30;17002:15;16982:18;;;16975:43;17035:18;;86893:57:0;16722:337:1;86893:57:0;87036:10;87014:33;;;;:21;:33;;;;;;;;;:49;;;;;;;;;;;;;;;;87013:50;86961:102;;;;;;;87166:25;:41;;;;;;;;:55;87162:263;;87237:41;;;87289:1;87237:41;;;:25;:41;;;;;;;;:54;;;;;;87380:17;:33;;;;;;87365:10;87347:29;;;;;;;:66;;87380:33;;87347:29;;:66;;87380:33;;87347:66;:::i;:::-;;;;-1:-1:-1;;87162:263:0;86822:610;:::o;77044:452::-;77105:16;77134:32;77169:16;77186:9;77169:27;;;;;;;;:::i;:::-;;;;;;;;;;;;;;-1:-1:-1;77211:38:0;;77207:282;;77392:4;77350:27;77378:9;77350:38;;;;;;;;:::i;:::-;;;;;;;;;77299:24;77278:67;;;:69;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;:110;;;;:::i;:::-;77277:119;;;;:::i;:::-;77266:130;;77207:282;;;77449:17;77467:9;77449:28;;;;;;;;:::i;:::-;;;;;;;;;77438:39;;77123:373;77044:452;;;:::o;98423:761::-;98540:9;98535:642;98559:16;:23;98555:27;;98535:642;;;98604:32;98639:16;98656:1;98639:19;;;;;;;;:::i;:::-;;;;;;;;;;;;;;-1:-1:-1;98677:38:0;;98673:493;;98740:12;:61;;;;98775:22;98798:1;98775:25;;;;;;;;:::i;:::-;;;;;;;;;98757:15;:43;98740:61;98736:415;;;98915:106;;;;;98998:4;98915:106;;;17596:74:1;99005:15:0;17686:18:1;;;17679:34;98915:74:0;;;;;;17569:18:1;;98915:106:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;98882:27;98910:1;98882:30;;;;;;;;:::i;:::-;;;;;;;;:139;;;;99093:24;99072:57;;;:59;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;99044:22;99067:1;99044:25;;;;;;;;:::i;:::-;;;;;;;;;;:87;98736:415;-1:-1:-1;98584:3:0;;;;:::i;:::-;;;;98535:642;;;;98423:761;:::o;102764:958::-;73462:5;;;;73448:10;:19;;:79;;-1:-1:-1;73485:42:0;73471:10;:56;73448:79;73440:113;;;;;;;12354:2:1;73440:113:0;;;12336:21:1;12393:2;12373:18;;;12366:30;12432:23;12412:18;;;12405:51;12473:18;;73440:113:0;12152:345:1;73440:113:0;103173:13;;69965:4:::1;-1:-1:-1::0;103173:37:0::1;103165:69;;;::::0;::::1;::::0;;17926:2:1;103165:69:0::1;::::0;::::1;17908:21:1::0;17965:2;17945:18;;;17938:30;18004:21;17984:18;;;17977:49;18043:18;;103165:69:0::1;17724:343:1::0;103165:69:0::1;103358:13:::0;;::::1;::::0;103375:1:::1;-1:-1:-1::0;103358:18:0;::::1;::::0;103357:44:::1;;-1:-1:-1::0;103382:13:0;;::::1;::::0;103399:1:::1;-1:-1:-1::0;103382:18:0::1;103357:44;103349:69;;;::::0;::::1;::::0;;18615:2:1;103349:69:0::1;::::0;::::1;18597:21:1::0;18654:2;18634:18;;;18627:30;18693:14;18673:18;;;18666:42;18725:18;;103349:69:0::1;18413:336:1::0;103349:69:0::1;103453:13:::0;;103431:19:::1;:35:::0;103453:13:::1;103500::::0;::::1;::::0;103477:20:::1;:36:::0;103555:13;;::::1;::::0;103524:28:::1;:44:::0;103606:13;;::::1;::::0;103579:24:::1;:40:::0;103661:13;;::::1;::::0;103630:28:::1;:44:::0;103701:13;::::1;::::0;103685::::1;:29:::0;102764:958::o;99283:363::-;99366:25;99385:5;99366:18;:25::i;:::-;99461:16;:14;:16::i;:::-;99443:15;:34;99513:12;;99494:15;:31;99490:149;;99542:14;:12;:14::i;:::-;99283:363::o;99490:149::-;99598:29;:27;:29::i;126363:1651::-;126538:7;126866:13;:50;;;-1:-1:-1;126884:32:0;;;;;;;126883:33;126866:50;126862:246;;;126918:53;126929:14;126945:19;126966:4;126918:10;:53::i;:::-;;126862:246;;;127044:52;127068:14;127084:4;127090:5;127044:23;:52::i;:::-;127161:28;127191:21;127216:33;127226:14;127242:6;127216:9;:33::i;:::-;127160:89;;;;127287:9;:26;;;127268:15;:45;;:71;;;-1:-1:-1;127317:14:0;;;;:22;;:14;:22;127268:71;127260:106;;;;;;;18956:2:1;127260:106:0;;;18938:21:1;18995:2;18975:18;;;18968:30;19034:24;19014:18;;;19007:52;19076:18;;127260:106:0;18754:346:1;127260:106:0;127397:19;;;;127433:13;;127429:549;;127613:12;;127577:82;;127613:12;;;;;127628:19;127649:9;127577:27;:82::i;:::-;127731:28;;;;;;;:12;:28;;;;;:43;;127760:13;;127731:43;;;;;;:::i;:::-;;;;;;;;;;;;;127724:50;;;;;;;;;;;;;;;;;;;;;;;;;;127826:48;;127841:14;;127857:9;;127826:14;:48::i;:::-;127896:70;;;19307:25:1;;;19363:2;19348:18;;19341:34;;;127896:70:0;19411:55:1;;;19391:18;;;19384:83;127896:70:0;;;;;;;;;;;19295:2:1;127896:70:0;;;127429:549;127997:9;126363:1651;-1:-1:-1;;;;;;;126363:1651:0:o;60152:106::-;60210:7;60241:1;60237;:5;:13;;60249:1;60237:13;;;60245:1;60237:13;60230:20;60152:106;-1:-1:-1;;;60152:106:0:o;124404:1358::-;124650:7;124619:14;124635:4;73856:49;73880:7;73889:8;73899:5;73856:23;:49::i;:::-;124678:13:::1;::::0;;;::::1;;;:22;124670:49;;;::::0;::::1;::::0;;19680:2:1;124670:49:0::1;::::0;::::1;19662:21:1::0;19719:2;19699:18;;;19692:30;19758:16;19738:18;;;19731:44;19792:18;;124670:49:0::1;19478:338:1::0;124670:49:0::1;124746:13;;124738:4;:21;;124730:60;;;::::0;::::1;::::0;;15817:2:1;124730:60:0::1;::::0;::::1;15799:21:1::0;15856:2;15836:18;;;15829:30;15895:28;15875:18;;;15868:56;15941:18;;124730:60:0::1;15615:350:1::0;124730:60:0::1;124817:28;;124809:4;:36;;124801:75;;;::::0;::::1;::::0;;16172:2:1;124801:75:0::1;::::0;::::1;16154:21:1::0;16211:2;16191:18;;;16184:30;16250:29;16230:18;;;16223:57;16297:18;;124801:75:0::1;15970:351:1::0;124801:75:0::1;124999:12;::::0;124959:96:::1;::::0;124999:12;;::::1;;;125014:14:::0;125038:4:::1;125045:9:::0;124959:31:::1;:96::i;:::-;125115:23;125141:20;125156:4;125141:14;:20::i;:::-;125260:33;::::0;::::1;125172:14;125260:33:::0;;;:17:::1;:33;::::0;;;;;;;;125199:95;;20067:66:1;20054:2;20050:15;;;20046:88;125199:95:0;;::::1;20034:101:1::0;;;;20151:12;;;20144:28;;;20188:12;;;20181:28;;;20225:12;;;20218:28;125115:46:0;;-1:-1:-1;125172:14:0;20262:13:1;;125199:95:0::1;::::0;;;;;::::1;::::0;;;;;;125189:106;;125199:95:::1;125189:106:::0;;::::1;::::0;125352:28:::1;::::0;::::1;;::::0;;;:12:::1;:28:::0;;;;;125386:164:::1;::::0;::::1;::::0;;;;;;;::::1;::::0;;;;;;;;;125189:106;;-1:-1:-1;125352:28:0;125386:164;;;125487:22:::1;125505:4:::0;125433:15;125487:22:::1;:::i;:::-;125386:164:::0;;::::1;::::0;;::::1;::::0;;;125352:199;;::::1;::::0;;::::1;::::0;;-1:-1:-1;125352:199:0;;;;;;;;;::::1;::::0;;::::1;;::::0;;;;;::::1;::::0;;;::::1;::::0;::::1;::::0;::::1;::::0;::::1;::::0;::::1;::::0;::::1;::::0;::::1;::::0;::::1;::::0;::::1;::::0;::::1;::::0;;::::1;::::0;::::1;::::0;;::::1;::::0;125595:47:::1;::::0;125610:14;;125626:9;;125595:14:::1;:47::i;:::-;125660:68;::::0;;20517:25:1;;;20573:2;20558:18;;20551:34;;;20601:18;;;20594:34;;;125660:68:0::1;20664:55:1::0;;;20659:2;20644:18;;20637:83;125660:68:0;::::1;::::0;::::1;::::0;20504:3:1;20489:19;125660:68:0::1;;;;;;;125748:6:::0;124404:1358;-1:-1:-1;;;;;;;;;124404:1358:0:o;76722:133::-;76781:7;76808:39;76817:15;76834:12;;76808:8;:39::i;92791:1194::-;92939:31;92914:8;92924:4;73856:49;73880:7;73889:8;73899:5;73856:23;:49::i;:::-;93069:29:::1;::::0;::::1;;::::0;;;:19:::1;:29;::::0;;;;93101:15:::1;93069:47:::0;;93199:23:::1;::::0;;;::::1;;;:32;93191:70;;;::::0;::::1;::::0;;12704:2:1;93191:70:0::1;::::0;::::1;12686:21:1::0;12743:2;12723:18;;;12716:30;12782:27;12762:18;;;12755:55;12827:18;;93191:70:0::1;12502:349:1::0;93191:70:0::1;93373:12;:19:::0;93359:34:::1;::::0;::::1;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;::::0;-1:-1:-1;93359:34:0::1;;93342:51;;93411:9;93406:419;93430:12;:19:::0;93426:23;::::1;93406:419;;;93491:17;::::0;::::1;;::::0;;;:7:::1;:17;::::0;;;;;;;:20;;;;;;;;;93471:17;;:14;;93509:1;;93471:17;::::1;;;;;:::i;:::-;;::::0;;::::1;::::0;;;;;;:40;;;;93526:17:::1;::::0;::::1;93549:1;93526:17:::0;;;:7:::1;:17:::0;;;;;;:20;;;;;;;;;:24;;;93569:17;;:14;;93544:1;;93569:17;::::1;;;;;:::i;:::-;;;;;;;:21;93565:249;;;93611:84;93639:12;93652:1;93639:15;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;93656:19;93677:14;93692:1;93677:17;;;;;;;;:::i;:::-;;;;;;;93611:27;:84::i;:::-;93732:8;93721:77;;;93742:14;93757:1;93742:17;;;;;;;;:::i;:::-;;;;;;;93761:12;93774:1;93761:15;;;;;;;;:::i;:::-;;::::0;;;::::1;::::0;;;;::::1;::::0;93721:77:::1;::::0;;20933:25:1;;;93761:15:0::1;::::0;;::::1;21035:18:1::0;;;21028:43;;;;21107:15;;21087:18;;;21080:43;20921:2;20906:18;93721:77:0::1;;;;;;;93565:249;93451:3:::0;::::1;::::0;::::1;:::i;:::-;;;;93406:419;;;;92791:1194:::0;;;;;;;:::o;63746:361::-;63941:45;;;63930:10;17614:55:1;;;63941:45:0;;;17596:74:1;17686:18;;;;17679:34;;;63941:45:0;;;;;;;;;;17569:18:1;;;;63941:45:0;;;;;;;;;;;;;63930:57;;-1:-1:-1;;;;63930:10:0;;;;:57;;63941:45;63930:57;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;63894:93;;;;64006:7;:57;;;;-1:-1:-1;64018:11:0;;:16;;:44;;;64049:4;64038:24;;;;;;;;;;;;:::i;:::-;63998:101;;;;;;;22133:2:1;63998:101:0;;;22115:21:1;22172:2;22152:18;;;22145:30;22211:33;22191:18;;;22184:61;22262:18;;63998:101:0;21931:355:1;63998:101:0;63816:291;;63746:361;;;:::o;88704:1788::-;88920:8;88916:46;;;88944:6;:4;:6::i;:::-;89094:19;89090:109;;;89163:24;89179:7;89163:15;:24::i;:::-;89129:31;;;;;;;:22;:31;;;;;:58;89090:109;89223:21;;;;89219:1266;;89437:27;89483:28;89530:27;89575:30;89597:7;89575:21;:30::i;:::-;89415:190;;;;;;89667:20;89679:7;89667:11;:20::i;:::-;89763:31;;;;;;;:22;:31;;;;;:54;;;89904:42;;;89900:572;;89967:19;89989:41;90011:19;89989;:41;:::i;:::-;89967:63;;90099:11;90074:22;;:36;;;;:::i;:::-;90049:22;:61;90158:33;90180:11;90158:19;:33;:::i;:::-;90129:26;;;;;;;:17;:26;;;;;:62;-1:-1:-1;89900:572:0;;;90232:19;90254:41;90276:19;90254;:41;:::i;:::-;90232:63;;90364:11;90339:22;;:36;;;;:::i;:::-;90314:22;:61;90423:33;90445:11;90423:19;:33;:::i;:::-;90394:26;;;;;;;:17;:26;;;;;:62;-1:-1:-1;89900:572:0;89246:1239;;;88704:1788;;;:::o;120888:508::-;120970:31;-1:-1:-1;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;120970:31:0;121003:15;;121031:280;121055:28;;;;;;;:12;:28;;;;;:35;121051:39;;121031:280;;;121126:28;;;;;;;:12;:28;;;;;:31;;121155:1;;121126:31;;;;;;:::i;:::-;;;;;;;;;;;:38;;;121116:6;:48;121112:188;;121199:28;;;;;;;:12;:28;;;;;:31;;121228:1;;121199:31;;;;;;:::i;:::-;;;;;;;;;;;121184:46;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;121259:1;121249:11;;121279:5;;121112:188;121092:3;;;;:::i;:::-;;;;121031:280;;;-1:-1:-1;121329:19:0;;:29;;121321:57;;;;;;;22493:2:1;121321:57:0;;;22475:21:1;22532:2;22512:18;;;22505:30;22571:17;22551:18;;;22544:45;22606:18;;121321:57:0;22291:339:1;121321:57:0;120888:508;;;;;:::o;64115:402::-;64340:51;;;64329:10;22916:15:1;;;64340:51:0;;;22898:34:1;22968:15;;;22948:18;;;22941:43;23000:18;;;;22993:34;;;64340:51:0;;;;;;;;;;22810:18:1;;;;64340:51:0;;;;;;;;;;;;;64329:63;;-1:-1:-1;;;;64329:10:0;;;;:63;;64340:51;64329:63;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;64293:99;;;;64411:7;:57;;;;-1:-1:-1;64423:11:0;;:16;;:44;;;64454:4;64443:24;;;;;;;;;;;;:::i;:::-;64403:106;;;;;;;23240:2:1;64403:106:0;;;23222:21:1;23279:2;23259:18;;;23252:30;23318:34;23298:18;;;23291:62;23389:6;23369:18;;;23362:34;23413:19;;64403:106:0;23038:400:1;120003:877:0;120131:17;120151:27;120163:14;120151:11;:27::i;:::-;120131:47;;120195:6;120191:561;;;120286:3;120259:23;;:30;;;;;;;:::i;:::-;;;;-1:-1:-1;;120304:33:0;;;;;;;:17;:33;;;;;:40;;120341:3;;120304:33;:40;;120341:3;;120304:40;:::i;:::-;;;;-1:-1:-1;;120398:23:0;;;;120394:64;;120423:28;;;;;;;:17;:28;;;;;:35;;120455:3;;120423:28;:35;;120455:3;;120423:35;:::i;:::-;;;;-1:-1:-1;;120394:64:0;120191:561;;;120568:3;120541:23;;:30;;;;;;;:::i;:::-;;;;-1:-1:-1;;120586:33:0;;;;;;;:17;:33;;;;;:40;;120623:3;;120586:33;:40;;120623:3;;120586:40;:::i;:::-;;;;-1:-1:-1;;120680:23:0;;;;120676:64;;120705:28;;;;;;;:17;:28;;;;;:35;;120737:3;;120705:28;:35;;120737:3;;120705:35;:::i;:::-;;;;-1:-1:-1;;120676:64:0;120820:52;120844:14;120860:5;120867:4;120820:23;:52::i;94072:3748::-;94159:29;:27;:29::i;:::-;94278:9;94273:327;94297:18;:25;94293:29;;94273:327;;;94344:34;94381:18;94400:1;94381:21;;;;;;;;:::i;:::-;;;;;;;;;;;;;;-1:-1:-1;94421:40:0;;94417:172;;94482:91;;;;;94567:4;94482:91;;;4219:74:1;94482:76:0;;;;;;4192:18:1;;94482:91:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;;;94417:172;-1:-1:-1;94324:3:0;;;;:::i;:::-;;;;94273:327;;;;94960:27;71696:6;95016:12;;94998:15;:30;;;;:::i;:::-;94990:57;;;;:::i;:::-;94960:87;;95186:9;95181:284;95205:12;:19;95201:23;;95181:284;;;95327:12;95340:1;95327:15;;;;;;;;:::i;:::-;;;;;;;;;;;95320:48;;;;;95362:4;95320:48;;;4219:74:1;95327:15:0;;;;;95320:33;;4192:18:1;;95320:48:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;:::i;:::-;95291:23;:19;95313:1;95291:23;:::i;:::-;71696:6;95255:14;95267:1;95255:11;:14::i;:::-;:32;;;;:::i;:::-;:60;;;;:::i;:::-;95254:114;;95434:12;95447:1;95434:15;;;;;;;;:::i;:::-;;;;;;;;;;;;95377:73;;;23935:34:1;95377:73:0;;;23923:47:1;;;;24000:6;23986:12;;;23979:28;24045:2;24041:15;;;24023:12;;;24016:110;24142:12;;95377:73:0;;;;;;;;;;;;95246:207;;;;;;;;;;;;;;:::i;:::-;-1:-1:-1;95226:3:0;;;;:::i;:::-;;;;95181:284;;;-1:-1:-1;71696:6:0;95677:23;:19;95699:1;95677:23;:::i;:::-;95676:43;;;;:::i;:::-;95660:12;;:60;;;;:::i;:::-;95645:12;:75;97783:29;97828:435;97915:34;97952:17;:15;:17::i;:::-;97915:54;;98032:9;98027:133;98051:21;:28;98047:32;;98027:133;;;98128:17;98146:1;98128:20;;;;;;;;:::i;:::-;;;;;;;98101:21;98123:1;98101:24;;;;;;;;:::i;:::-;;;;;;;;;;:47;98081:3;;;;:::i;:::-;;;;98027:133;;;;98229:26;:24;:26::i;:::-;98212:14;:43;-1:-1:-1;97828:435:0:o;90500:585::-;90562:21;;;;90558:520;;90639:27;90669:15;90676:7;90669:6;:15::i;:::-;90639:45;;90747:9;90742:118;90766:10;:17;90762:1;:21;90742:118;;;90831:10;90842:1;90831:13;;;;;;;;:::i;:::-;;;;;;;;;;;;90809:16;;;;;;;:7;:16;;;;;;:19;;;;;;;;;;:35;90826:1;90785:3;90826:1;90785:3;:::i;:::-;;;;90742:118;;;;90927:9;90922:145;90946:10;:17;90942:1;:21;90922:145;;;91027:21;91049:1;91027:24;;;;;;;;:::i;:::-;;;;;;;;;;;;;90989:32;;;;;:23;:32;;;;;;:35;;;;;;;;:62;91022:1;90965:3;91022:1;90965:3;:::i;:::-;;;;90922:145;;14:196:1;82:20;;142:42;131:54;;121:65;;111:93;;200:1;197;190:12;215:186;274:6;327:2;315:9;306:7;302:23;298:32;295:52;;;343:1;340;333:12;295:52;366:29;385:9;366:29;:::i;406:632::-;577:2;629:21;;;699:13;;602:18;;;721:22;;;548:4;;577:2;800:15;;;;774:2;759:18;;;548:4;843:169;857:6;854:1;851:13;843:169;;;918:13;;906:26;;987:15;;;;952:12;;;;879:1;872:9;843:169;;;-1:-1:-1;1029:3:1;;406:632;-1:-1:-1;;;;;;406:632:1:o;1043:118::-;1129:5;1122:13;1115:21;1108:5;1105:32;1095:60;;1151:1;1148;1141:12;1166:383;1240:6;1248;1256;1309:2;1297:9;1288:7;1284:23;1280:32;1277:52;;;1325:1;1322;1315:12;1277:52;1361:9;1348:23;1338:33;;1390:38;1424:2;1413:9;1409:18;1390:38;:::i;:::-;1380:48;;1478:2;1467:9;1463:18;1450:32;1491:28;1513:5;1491:28;:::i;:::-;1538:5;1528:15;;;1166:383;;;;;:::o;1736:180::-;1795:6;1848:2;1836:9;1827:7;1823:23;1819:32;1816:52;;;1864:1;1861;1854:12;1816:52;-1:-1:-1;1887:23:1;;1736:180;-1:-1:-1;1736:180:1:o;1921:681::-;2092:2;2144:21;;;2214:13;;2117:18;;;2236:22;;;2063:4;;2092:2;2315:15;;;;2289:2;2274:18;;;2063:4;2358:218;2372:6;2369:1;2366:13;2358:218;;;2437:13;;2452:42;2433:62;2421:75;;2551:15;;;;2516:12;;;;2394:1;2387:9;2358:218;;2607:248;2675:6;2683;2736:2;2724:9;2715:7;2711:23;2707:32;2704:52;;;2752:1;2749;2742:12;2704:52;-1:-1:-1;;2775:23:1;;;2845:2;2830:18;;;2817:32;;-1:-1:-1;2607:248:1:o;3042:1011::-;3271:2;3323:21;;;3393:13;;3296:18;;;3415:22;;;3242:4;;3271:2;3456;;3474:18;;;;3515:15;;;3242:4;3558:469;3572:6;3569:1;3566:13;3558:469;;;3631:13;;3669:9;;3657:22;;3719:11;;;3713:18;3699:12;;;3692:40;3772:11;;;3766:18;3752:12;;;3745:40;3808:4;3852:11;;;3846:18;3832:12;;;3825:40;3888:4;3932:11;;;3926:18;3912:12;;;3905:40;3974:4;3965:14;;;;4002:15;;;;3594:1;3587:9;3558:469;;;-1:-1:-1;4044:3:1;;3042:1011;-1:-1:-1;;;;;;;3042:1011:1:o;4304:260::-;4372:6;4380;4433:2;4421:9;4412:7;4408:23;4404:32;4401:52;;;4449:1;4446;4439:12;4401:52;4472:29;4491:9;4472:29;:::i;:::-;4462:39;;4520:38;4554:2;4543:9;4539:18;4520:38;:::i;:::-;4510:48;;4304:260;;;;;:::o;4992:647::-;5066:6;5074;5082;5090;5143:3;5131:9;5122:7;5118:23;5114:33;5111:53;;;5160:1;5157;5150:12;5111:53;5199:9;5186:23;5218:28;5240:5;5218:28;:::i;:::-;5265:5;-1:-1:-1;5322:2:1;5307:18;;5294:32;5335:30;5294:32;5335:30;:::i;:::-;5384:7;-1:-1:-1;5443:2:1;5428:18;;5415:32;5456:30;5415:32;5456:30;:::i;:::-;5505:7;-1:-1:-1;5564:2:1;5549:18;;5536:32;5577:30;5536:32;5577:30;:::i;:::-;4992:647;;;;-1:-1:-1;4992:647:1;;-1:-1:-1;;4992:647:1:o;6160:254::-;6228:6;6236;6289:2;6277:9;6268:7;6264:23;6260:32;6257:52;;;6305:1;6302;6295:12;6257:52;6328:29;6347:9;6328:29;:::i;:::-;6318:39;6404:2;6389:18;;;;6376:32;;-1:-1:-1;;;6160:254:1:o;6419:315::-;6484:6;6492;6545:2;6533:9;6524:7;6520:23;6516:32;6513:52;;;6561:1;6558;6551:12;6513:52;6584:29;6603:9;6584:29;:::i;:::-;6574:39;;6663:2;6652:9;6648:18;6635:32;6676:28;6698:5;6676:28;:::i;:::-;6723:5;6713:15;;;6419:315;;;;;:::o;7784:403::-;7870:6;7878;7886;7894;7947:3;7935:9;7926:7;7922:23;7918:33;7915:53;;;7964:1;7961;7954:12;7915:53;7987:29;8006:9;7987:29;:::i;:::-;7977:39;;8063:2;8052:9;8048:18;8035:32;8025:42;;8086:38;8120:2;8109:9;8105:18;8086:38;:::i;:::-;8076:48;;8143:38;8177:2;8166:9;8162:18;8143:38;:::i;:::-;8133:48;;7784:403;;;;;;;:::o;8192:241::-;8248:6;8301:2;8289:9;8280:7;8276:23;8272:32;8269:52;;;8317:1;8314;8307:12;8269:52;8356:9;8343:23;8375:28;8397:5;8375:28;:::i;8438:184::-;8490:77;8487:1;8480:88;8587:4;8584:1;8577:15;8611:4;8608:1;8601:15;8627:915;8709:6;8762:3;8750:9;8741:7;8737:23;8733:33;8730:53;;;8779:1;8776;8769:12;8730:53;8828:7;8821:4;8810:9;8806:20;8802:34;8792:62;;8850:1;8847;8840:12;8792:62;8883:2;8877:9;8925:3;8917:6;8913:16;8995:6;8983:10;8980:22;8959:18;8947:10;8944:34;8941:62;8938:242;;;9036:77;9033:1;9026:88;9137:4;9134:1;9127:15;9165:4;9162:1;9155:15;8938:242;9196:2;9189:22;9231:6;9275:3;9260:19;;9291;;;9288:39;;;9323:1;9320;9313:12;9288:39;9347:9;9365:146;9381:6;9376:3;9373:15;9365:146;;;9449:17;;9437:30;;9496:4;9487:14;;;;9398;9365:146;;;-1:-1:-1;9530:6:1;;8627:915;-1:-1:-1;;;;;8627:915:1:o;9547:184::-;9599:77;9596:1;9589:88;9696:4;9693:1;9686:15;9720:4;9717:1;9710:15;9736:184;9788:77;9785:1;9778:88;9885:4;9882:1;9875:15;9909:4;9906:1;9899:15;9925:128;9992:9;;;10013:11;;;10010:37;;;10027:18;;:::i;10058:168::-;10131:9;;;10162;;10179:15;;;10173:22;;10159:37;10149:71;;10200:18;;:::i;10231:274::-;10271:1;10297;10287:189;;10332:77;10329:1;10322:88;10433:4;10430:1;10423:15;10461:4;10458:1;10451:15;10287:189;-1:-1:-1;10490:9:1;;10231:274::o;10510:125::-;10575:9;;;10596:10;;;10593:36;;;10609:18;;:::i;10640:195::-;10679:3;10710:66;10703:5;10700:77;10697:103;;10780:18;;:::i;:::-;-1:-1:-1;10827:1:1;10816:13;;10640:195::o;11963:184::-;12033:6;12086:2;12074:9;12065:7;12061:23;12057:32;12054:52;;;12102:1;12099;12092:12;12054:52;-1:-1:-1;12125:16:1;;11963:184;-1:-1:-1;11963:184:1:o;21134:250::-;21219:1;21229:113;21243:6;21240:1;21237:13;21229:113;;;21319:11;;;21313:18;21300:11;;;21293:39;21265:2;21258:10;21229:113;;;-1:-1:-1;;21376:1:1;21358:16;;21351:27;21134:250::o;21389:287::-;21518:3;21556:6;21550:13;21572:66;21631:6;21626:3;21619:4;21611:6;21607:17;21572:66;:::i;:::-;21654:16;;;;;21389:287;-1:-1:-1;;21389:287:1:o;21681:245::-;21748:6;21801:2;21789:9;21780:7;21776:23;21772:32;21769:52;;;21817:1;21814;21807:12;21769:52;21849:9;21843:16;21868:28;21890:5;21868:28;:::i;23443:245::-;23522:6;23530;23583:2;23571:9;23562:7;23558:23;23554:32;23551:52;;;23599:1;23596;23589:12;23551:52;-1:-1:-1;;23622:16:1;;23678:2;23663:18;;;23657:25;23622:16;;23657:25;;-1:-1:-1;23443:245:1:o;24165:455::-;24314:2;24303:9;24296:21;24277:4;24346:6;24340:13;24389:6;24384:2;24373:9;24369:18;24362:34;24405:79;24477:6;24472:2;24461:9;24457:18;24452:2;24444:6;24440:15;24405:79;:::i;:::-;24536:2;24524:15;24541:66;24520:88;24505:104;;;;24611:2;24501:113;;24165:455;-1:-1:-1;;24165:455:1:o

Swarm Source

ipfs://79a11f13c04fb3948cc8f5871bc318534a596accdd8782fcfaa7c2474ca9ae33

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.