ETH Price: $3,147.44 (+1.80%)

Token

Babylon.Finance (BABL)
 

Overview

Max Total Supply

1,000,000 BABL

Holders

1,122 (0.00%)

Market

Onchain Market Cap

$0.00

Circulating Supply Market Cap

-

Other Info

Token Contract (WITH 18 Decimals)

Balance
5 BABL

Value
$0.00
0xab94aD5b4e71A160664f3Aa86dF61B364FBa3f00
Loading...
Loading
Loading...
Loading
Loading...
Loading

OVERVIEW

Babylon is a community-led asset management protocol that enables users to create investment clubs (we call them gardens) and invest in DeFi together. It’s built on the Ethereum network and it’s non-custodial, transparent, permission-less, and governed by the community.

# Exchange Pair Price  24H Volume % Volume

Contract Source Code Verified (Exact Match)

Contract Name:
BABLToken

Compiler Version
v0.7.6+commit.7338295f

Optimization Enabled:
Yes with 999 runs

Other Settings:
default evmVersion
File 1 of 18 : BABLToken.sol
/*
    Copyright 2021 Babylon Finance.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at
    http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.

    SPDX-License-Identifier: Apache License, Version 2.0
*/

pragma solidity 0.7.6;
import {LowGasSafeMath} from '../lib/LowGasSafeMath.sol';
import {TimeLockedToken} from './TimeLockedToken.sol';
import {Address} from '@openzeppelin/contracts/utils/Address.sol';
import {ReentrancyGuard} from '@openzeppelin/contracts/utils/ReentrancyGuard.sol';
import {IBabController} from '../interfaces/IBabController.sol';

/**
 * @title BABL Token
 * @dev The BABLToken contract is ERC20 using 18 decimals as a standard
 * Is Ownable to transfer ownership to Governor Alpha for Decentralized Governance
 * It overrides the mint and maximum supply to control the timing and maximum cap allowed along the time.
 */

contract BABLToken is TimeLockedToken {
    using LowGasSafeMath for uint256;
    using Address for address;

    /* ============ Events ============ */

    /// @notice An event that emitted when a new mint ocurr
    event MintedNewTokens(address account, uint256 tokensminted);

    /// @notice An event thats emitted when maxSupplyAllowed changes
    event MaxSupplyChanged(uint256 previousMaxValue, uint256 newMaxValue);

    /// @notice An event that emitted when maxSupplyAllowedAfter changes
    event MaxSupplyAllowedAfterChanged(uint256 previousAllowedAfterValue, uint256 newAllowedAfterValue);

    /* ============ Modifiers ============ */

    /* ============ State Variables ============ */

    /// @dev EIP-20 token name for this token
    string private constant NAME = 'Babylon.Finance';

    /// @dev EIP-20 token symbol for this token
    string private constant SYMBOL = 'BABL';

    /// @dev Maximum number of tokens in circulation of 1 million for the first 8 years (using 18 decimals as ERC20 standard)
    uint256 public maxSupplyAllowed = 1_000_000e18; //

    /// @notice The timestamp after which a change on maxSupplyAllowed may occur
    uint256 public maxSupplyAllowedAfter;

    /// @notice Cap on the percentage of maxSupplyAllowed that can be increased per year after maxSupplyAllowedAfter
    uint8 public constant MAX_SUPPLY_CAP = 5;

    /// @notice Cap on the percentage of totalSupply that can be minted at each mint after the initial 1 Million BABL
    uint8 public constant MINT_CAP = 2;

    /// @notice The timestamp after which minting may occur after FIRST_EPOCH_MINT (8 years)
    uint256 public mintingAllowedAfter;

    /// @notice The timestamp of BABL Token deployment
    uint256 public BABLTokenDeploymentTimestamp;

    /// @dev First Epoch Mint where no more than 1 Million BABL can be minted (>= 8 Years)
    uint32 private constant FIRST_EPOCH_MINT = 365 days * 8;

    /// @dev Minimum time between mints after
    uint32 private constant MIN_TIME_BETWEEN_MINTS = 365 days;

    /* ============ Functions ============ */

    /* ============ Constructor ============ */

    /**
     * @notice Construct a new BABL token and gives ownership to sender
     */
    constructor(IBabController newController) TimeLockedToken(NAME, SYMBOL) {
        // Timestamp of contract deployment
        BABLTokenDeploymentTimestamp = block.timestamp;

        // Set-up the minimum time of 8 years to wait until the maxSupplyAllowed can be changed (it will also include a max cap)
        maxSupplyAllowedAfter = block.timestamp.add(FIRST_EPOCH_MINT);

        //Starting with a maxSupplyAllowed of 1 million for the first 8 years
        _mint(msg.sender, 1_000_000e18);

        //Set-up the minimum time of 8 years for additional mints
        mintingAllowedAfter = block.timestamp.add(FIRST_EPOCH_MINT);

        // Set the Babylon Controller
        controller = newController;
    }

    /* ============ External Functions ============ */

    /* ===========  Token related Gov Functions ====== */

    /**
     * PRIVILEGED GOVERNANCE FUNCTION. Allows to mint new tokens
     *
     * @notice Mint new BABL tokens. Initial 1 Million BABL. After 8 years new BABL could be minted by governance decision
     * @dev MINT_CAP The new maximum limit, limited by a 2% cap of totalSupply for each new mint and always limited by maxSupplyAllowed.
     * mintingAllowedAfter Defines the next time allowed for a new mint
     * @param _to The address of the destination account that will receive the new BABL tokens
     * @param _amount The number of tokens to be minted
     * @return Whether or not the mint succeeded
     */
    function mint(address _to, uint256 _amount) external onlyOwner returns (bool) {
        require(totalSupply().add(_amount) <= maxSupplyAllowed, 'BABLToken::mint: max supply exceeded');
        require(
            block.timestamp >= BABLTokenDeploymentTimestamp.add(FIRST_EPOCH_MINT),
            'BABLToken::mint: minting not allowed after the FIRST_EPOCH_MINT has passed >= 8 years'
        );
        require(_amount > 0, 'BABLToken::mint: mint should be higher than zero');
        require(
            block.timestamp >= mintingAllowedAfter,
            'BABLToken::mint: minting not allowed yet because mintingAllowedAfter'
        );
        require(_to != address(0), 'BABLToken::mint: cannot transfer to the zero address');
        require(_to != address(this), 'BABLToken::mint: cannot mint to the address of this contract');

        // set-up the new time where a new (the next) mint can be allowed
        mintingAllowedAfter = block.timestamp.add(MIN_TIME_BETWEEN_MINTS);

        // mint the amount
        uint96 amount = safe96(_amount, 'BABLToken::mint: amount exceeds 96 bits');

        // After FIRST_EPOCH_MINT (8 years) a MINT_CAP applies
        require(
            amount <= totalSupply().mul(MINT_CAP).div(100),
            'BABLToken::mint: exceeded mint cap of 2% of total supply'
        );
        _mint(_to, amount);

        emit MintedNewTokens(_to, amount);

        // move delegates to add voting power to the destination
        _moveDelegates(address(0), delegates[_to], amount);

        return true;
    }

    /**
     * PRIVILEGED GOVERNANCE FUNCTION. Allows governance to change maxSupplyAllowed
     *
     * @notice Set-up a greater maxSupplyAllowed value to allow more tokens to be minted
     * @param newMaxSupply The new maximum limit, limited by a maximum of 5% cap per year
     * @param newMaxSupplyAllowedAfter The new waiting period to change the maxSupplyAllowed limited for a minimum of 1 year
     * @return Whether or not the changeMaxSupply succeeded
     */
    function changeMaxSupply(uint256 newMaxSupply, uint256 newMaxSupplyAllowedAfter) external onlyOwner returns (bool) {
        require(
            block.timestamp >= BABLTokenDeploymentTimestamp.add(FIRST_EPOCH_MINT),
            'BABLToken::changeMaxSupply: a change on maxSupplyAllowed not allowed until 8 years after deployment'
        );
        require(
            block.timestamp >= maxSupplyAllowedAfter,
            'BABLToken::changeMaxSupply: a change on maxSupplyAllowed not allowed yet'
        );

        // update the amount
        require(
            newMaxSupply > maxSupplyAllowed,
            'BABLToken::changeMaxSupply: changeMaxSupply should be higher than previous value'
        );
        uint256 limitedNewSupply = maxSupplyAllowed.add(maxSupplyAllowed.mul(MAX_SUPPLY_CAP).div(100));
        require(newMaxSupply <= limitedNewSupply, 'BABLToken::changeMaxSupply: exceeded of allowed 5% cap');
        emit MaxSupplyChanged(maxSupplyAllowed, newMaxSupply);
        maxSupplyAllowed = safe96(newMaxSupply, 'BABLToken::changeMaxSupply: potential max amount exceeds 96 bits');

        // update the new waiting time until a new change could be done >= 1 year since this change
        uint256 futureTime = block.timestamp.add(365 days);
        require(
            newMaxSupplyAllowedAfter >= futureTime,
            'BABLToken::changeMaxSupply: the newMaxSupplyAllowedAfter should be at least 1 year in the future'
        );
        emit MaxSupplyAllowedAfterChanged(maxSupplyAllowedAfter, newMaxSupplyAllowedAfter);
        maxSupplyAllowedAfter = safe96(
            newMaxSupplyAllowedAfter,
            'BABLToken::changeMaxSupply: new newMaxSupplyAllowedAfter exceeds 96 bits'
        );

        return true;
    }

    /**
     * PUBLIC FUNCTION. Get the value of maxSupplyAllowed
     *
     * @return Returns the value of maxSupplyAllowed at the time
     */
    function maxSupply() external view returns (uint96, uint256) {
        uint96 safeMaxSupply =
            safe96(maxSupplyAllowed, 'BABLToken::maxSupplyAllowed: maxSupplyAllowed exceeds 96 bits'); // Overflow check
        return (safeMaxSupply, maxSupplyAllowedAfter);
    }
}

File 2 of 18 : LowGasSafeMath.sol
// SPDX-License-Identifier: GPL-2.0-or-later

pragma solidity 0.7.6;

/// @title Optimized overflow and underflow safe math operations
/// @notice Contains methods for doing math operations that revert on overflow or underflow for minimal gas cost
library LowGasSafeMath {
    /// @notice Returns x + y, reverts if sum overflows uint256
    /// @param x The augend
    /// @param y The addend
    /// @return z The sum of x and y
    function add(uint256 x, uint256 y) internal pure returns (uint256 z) {
        require((z = x + y) >= x);
    }

    /// @notice Returns x - y, reverts if underflows
    /// @param x The minuend
    /// @param y The subtrahend
    /// @return z The difference of x and y
    function sub(uint256 x, uint256 y) internal pure returns (uint256 z) {
        require((z = x - y) <= x);
    }

    /// @notice Returns x * y, reverts if overflows
    /// @param x The multiplicand
    /// @param y The multiplier
    /// @return z The product of x and y
    function mul(uint256 x, uint256 y) internal pure returns (uint256 z) {
        require(x == 0 || (z = x * y) / x == y);
    }

    /// @notice Returns x + y, reverts if overflows or underflows
    /// @param x The augend
    /// @param y The addend
    /// @return z The sum of x and y
    function add(int256 x, int256 y) internal pure returns (int256 z) {
        require((z = x + y) >= x == (y >= 0));
    }

    /// @notice Returns x - y, reverts if overflows or underflows
    /// @param x The minuend
    /// @param y The subtrahend
    /// @return z The difference of x and y
    function sub(int256 x, int256 y) internal pure returns (int256 z) {
        require((z = x - y) <= x == (y >= 0));
    }

    /**
     * @dev Returns the integer division of two unsigned integers, reverting 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) {
        require(b > 0, 'SafeMath: division by zero');
        return a / b;
    }
}

File 3 of 18 : TimeLockedToken.sol
/*
    Copyright 2021 Babylon Finance.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at
    http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.

    SPDX-License-Identifier: Apache License, Version 2.0
*/

pragma solidity 0.7.6;
import {IBabController} from '../interfaces/IBabController.sol';
import {TimeLockRegistry} from './TimeLockRegistry.sol';
import {IRewardsDistributor} from '../interfaces/IRewardsDistributor.sol';
import {VoteToken} from './VoteToken.sol';
import {Ownable} from '@openzeppelin/contracts/access/Ownable.sol';
import {Errors, _require} from '../lib/BabylonErrors.sol';
import {LowGasSafeMath} from '../lib/LowGasSafeMath.sol';
import {IBabController} from '../interfaces/IBabController.sol';

/**
 * @title TimeLockedToken
 * @notice Time Locked ERC20 Token
 * @author Babylon Finance
 * @dev Contract which gives the ability to time-lock tokens specially for vesting purposes usage
 *
 * By overriding the balanceOf() and transfer() functions in ERC20,
 * an account can show its full, post-distribution balance and use it for voting power
 * but only transfer or spend up to an allowed amount
 *
 * A portion of previously non-spendable tokens are allowed to be transferred
 * along the time depending on each vesting conditions, and after all epochs have passed, the full
 * account balance is unlocked. In case on non-completion vesting period, only the Time Lock Registry can cancel
 * the delivery of the pending tokens and only can cancel the remaining locked ones.
 */

abstract contract TimeLockedToken is VoteToken {
    using LowGasSafeMath for uint256;

    /* ============ Events ============ */

    /// @notice An event that emitted when a new lockout ocurr
    event NewLockout(
        address account,
        uint256 tokenslocked,
        bool isTeamOrAdvisor,
        uint256 startingVesting,
        uint256 endingVesting
    );

    /// @notice An event that emitted when a new Time Lock is registered
    event NewTimeLockRegistration(address previousAddress, address newAddress);

    /// @notice An event that emitted when a new Rewards Distributor is registered
    event NewRewardsDistributorRegistration(address previousAddress, address newAddress);

    /// @notice An event that emitted when a cancellation of Lock tokens is registered
    event Cancel(address account, uint256 amount);

    /// @notice An event that emitted when a claim of tokens are registered
    event Claim(address _receiver, uint256 amount);

    /// @notice An event that emitted when a lockedBalance query is done
    event LockedBalance(address _account, uint256 amount);

    /* ============ Modifiers ============ */

    modifier onlyTimeLockRegistry() {
        require(
            msg.sender == address(timeLockRegistry),
            'TimeLockedToken:: onlyTimeLockRegistry: can only be executed by TimeLockRegistry'
        );
        _;
    }

    modifier onlyTimeLockOwner() {
        if (address(timeLockRegistry) != address(0)) {
            require(
                msg.sender == Ownable(timeLockRegistry).owner(),
                'TimeLockedToken:: onlyTimeLockOwner: can only be executed by the owner of TimeLockRegistry'
            );
        }
        _;
    }
    modifier onlyUnpaused() {
        // Do not execute if Globally or individually paused
        _require(!IBabController(controller).isPaused(address(this)), Errors.ONLY_UNPAUSED);
        _;
    }

    /* ============ State Variables ============ */

    // represents total distribution for locked balances
    mapping(address => uint256) distribution;

    /// @notice The profile of each token owner under its particular vesting conditions
    /**
     * @param team Indicates whether or not is a Team member or Advisor (true = team member/advisor, false = private investor)
     * @param vestingBegin When the vesting begins for such token owner
     * @param vestingEnd When the vesting ends for such token owner
     * @param lastClaim When the last claim was done
     */
    struct VestedToken {
        bool teamOrAdvisor;
        uint256 vestingBegin;
        uint256 vestingEnd;
        uint256 lastClaim;
    }

    /// @notice A record of token owners under vesting conditions for each account, by index
    mapping(address => VestedToken) public vestedToken;

    // address of Time Lock Registry contract
    IBabController public controller;

    // address of Time Lock Registry contract
    TimeLockRegistry public timeLockRegistry;

    // address of Rewards Distriburor contract
    IRewardsDistributor public rewardsDistributor;

    // Enable Transfer of ERC20 BABL Tokens
    // Only Minting or transfers from/to TimeLockRegistry and Rewards Distributor can transfer tokens until the protocol is fully decentralized
    bool private tokenTransfersEnabled;
    bool private tokenTransfersWereDisabled;

    /* ============ Functions ============ */

    /* ============ Constructor ============ */

    constructor(string memory _name, string memory _symbol) VoteToken(_name, _symbol) {
        tokenTransfersEnabled = true;
    }

    /* ============ External Functions ============ */

    /* ===========  Token related Gov Functions ====== */

    /**
     * PRIVILEGED GOVERNANCE FUNCTION. Disables transfers of ERC20 BABL Tokens
     */
    function disableTokensTransfers() external onlyOwner {
        require(!tokenTransfersWereDisabled, 'BABL must flow');
        tokenTransfersEnabled = false;
        tokenTransfersWereDisabled = true;
    }

    /**
     * PRIVILEGED GOVERNANCE FUNCTION. Allows transfers of ERC20 BABL Tokens
     * Can only happen after the protocol is fully decentralized.
     */
    function enableTokensTransfers() external onlyOwner {
        tokenTransfersEnabled = true;
    }

    /**
     * PRIVILEGED GOVERNANCE FUNCTION. Set the Time Lock Registry contract to control token vesting conditions
     *
     * @notice Set the Time Lock Registry contract to control token vesting conditions
     * @param newTimeLockRegistry Address of TimeLockRegistry contract
     */
    function setTimeLockRegistry(TimeLockRegistry newTimeLockRegistry) external onlyTimeLockOwner returns (bool) {
        require(address(newTimeLockRegistry) != address(0), 'cannot be zero address');
        require(address(newTimeLockRegistry) != address(this), 'cannot be this contract');
        require(address(newTimeLockRegistry) != address(timeLockRegistry), 'must be new TimeLockRegistry');
        emit NewTimeLockRegistration(address(timeLockRegistry), address(newTimeLockRegistry));

        timeLockRegistry = newTimeLockRegistry;

        return true;
    }

    /**
     * PRIVILEGED GOVERNANCE FUNCTION. Set the Rewards Distributor contract to control either BABL Mining or profit rewards
     *
     * @notice Set the Rewards Distriburor contract to control both types of rewards (profit and BABL Mining program)
     * @param newRewardsDistributor Address of Rewards Distributor contract
     */
    function setRewardsDistributor(IRewardsDistributor newRewardsDistributor) external onlyOwner returns (bool) {
        require(address(newRewardsDistributor) != address(0), 'cannot be zero address');
        require(address(newRewardsDistributor) != address(this), 'cannot be this contract');
        require(address(newRewardsDistributor) != address(rewardsDistributor), 'must be new Rewards Distributor');
        emit NewRewardsDistributorRegistration(address(rewardsDistributor), address(newRewardsDistributor));

        rewardsDistributor = newRewardsDistributor;

        return true;
    }

    /**
     * PRIVILEGED GOVERNANCE FUNCTION. Register new token lockup conditions for vested tokens defined only by Time Lock Registry
     *
     * @notice Tokens are completely delivered during the registration however lockup conditions apply for vested tokens
     * locking them according to the distribution epoch periods and the type of recipient (Team, Advisor, Investor)
     * Emits a transfer event showing a transfer to the recipient
     * Only the registry can call this function
     * @param _receiver Address to receive the tokens
     * @param _amount Tokens to be transferred
     * @param _profile True if is a Team Member or Advisor
     * @param _vestingBegin Unix Time when the vesting for that particular address
     * @param _vestingEnd Unix Time when the vesting for that particular address
     * @param _lastClaim Unix Time when the claim was done from that particular address
     *
     */
    function registerLockup(
        address _receiver,
        uint256 _amount,
        bool _profile,
        uint256 _vestingBegin,
        uint256 _vestingEnd,
        uint256 _lastClaim
    ) external onlyTimeLockRegistry returns (bool) {
        require(balanceOf(msg.sender) >= _amount, 'insufficient balance');
        require(_receiver != address(0), 'cannot be zero address');
        require(_receiver != address(this), 'cannot be this contract');
        require(_receiver != address(timeLockRegistry), 'cannot be the TimeLockRegistry contract itself');
        require(_receiver != msg.sender, 'the owner cannot lockup itself');

        // update amount of locked distribution
        distribution[_receiver] = distribution[_receiver].add(_amount);

        VestedToken storage newVestedToken = vestedToken[_receiver];

        newVestedToken.teamOrAdvisor = _profile;
        newVestedToken.vestingBegin = _vestingBegin;
        newVestedToken.vestingEnd = _vestingEnd;
        newVestedToken.lastClaim = _lastClaim;

        // transfer tokens to the recipient
        _transfer(msg.sender, _receiver, _amount);
        emit NewLockout(_receiver, _amount, _profile, _vestingBegin, _vestingEnd);

        return true;
    }

    /**
     * PRIVILEGED GOVERNANCE FUNCTION. Cancel and remove locked tokens due to non-completion of vesting period
     * applied only by Time Lock Registry and specifically to Team or Advisors as it does not apply to investors.
     *
     * @dev Cancel distribution registration
     * @param lockedAccount that should have its still locked distribution removed due to non-completion of its vesting period
     */
    function cancelVestedTokens(address lockedAccount) external onlyTimeLockRegistry returns (uint256) {
        return _cancelVestedTokensFromTimeLock(lockedAccount);
    }

    /**
     * GOVERNANCE FUNCTION. Each token owner can claim its own specific tokens with its own specific vesting conditions from the Time Lock Registry
     *
     * @dev Claim msg.sender tokens (if any available in the registry)
     */
    function claimMyTokens() external {
        // claim msg.sender tokens from timeLockRegistry
        uint256 amount = timeLockRegistry.claim(msg.sender);
        // After a proper claim, locked tokens of Team and Advisors profiles are under restricted special vesting conditions so they automatic grant
        // rights to the Time Lock Registry to only retire locked tokens if non-compliance vesting conditions take places along the vesting periods.
        // It does not apply to Investors under vesting (their locked tokens cannot be removed).
        if (vestedToken[msg.sender].teamOrAdvisor == true) {
            approve(address(timeLockRegistry), amount);
        }
        // emit claim event
        emit Claim(msg.sender, amount);
    }

    /**
     * GOVERNANCE FUNCTION. Get unlocked balance for an account
     *
     * @notice Get unlocked balance for an account
     * @param account Account to check
     * @return Amount that is unlocked and available eg. to transfer
     */
    function unlockedBalance(address account) public returns (uint256) {
        // totalBalance - lockedBalance
        return balanceOf(account).sub(lockedBalance(account));
    }

    /**
     * GOVERNANCE FUNCTION. View the locked balance for an account
     *
     * @notice View locked balance for an account
     * @param account Account to check
     * @return Amount locked in the time of checking
     */

    function viewLockedBalance(address account) public view returns (uint256) {
        // distribution of locked tokens
        // get amount from distributions

        uint256 amount = distribution[account];
        uint256 lockedAmount = amount;

        // Team and investors cannot transfer tokens in the first year
        if (vestedToken[account].vestingBegin.add(365 days) > block.timestamp && amount != 0) {
            return lockedAmount;
        }

        // in case of vesting has passed, all tokens are now available, if no vesting lock is 0 as well
        if (block.timestamp >= vestedToken[account].vestingEnd || amount == 0) {
            lockedAmount = 0;
        } else if (amount != 0) {
            // in case of still under vesting period, locked tokens are recalculated
            lockedAmount = amount.mul(vestedToken[account].vestingEnd.sub(block.timestamp)).div(
                vestedToken[account].vestingEnd.sub(vestedToken[account].vestingBegin)
            );
        }
        return lockedAmount;
    }

    /**
     * GOVERNANCE FUNCTION. Get locked balance for an account
     *
     * @notice Get locked balance for an account
     * @param account Account to check
     * @return Amount locked in the time of checking
     */
    function lockedBalance(address account) public returns (uint256) {
        // get amount from distributions locked tokens (if any)
        uint256 lockedAmount = viewLockedBalance(account);
        // in case of vesting has passed, all tokens are now available so we set mapping to 0 only for accounts under vesting
        if (
            block.timestamp >= vestedToken[account].vestingEnd &&
            msg.sender == account &&
            lockedAmount == 0 &&
            vestedToken[account].vestingEnd != 0
        ) {
            delete distribution[account];
        }
        emit LockedBalance(account, lockedAmount);
        return lockedAmount;
    }

    /**
     * PUBLIC FUNCTION. Get the address of Time Lock Registry
     *
     * @notice Get the address of Time Lock Registry
     * @return Address of the Time Lock Registry
     */
    function getTimeLockRegistry() external view returns (address) {
        return address(timeLockRegistry);
    }

    /**
     * PRIVILEGED GOVERNANCE FUNCTION. Override the Approval of allowances of ERC20 with special conditions for vesting
     *
     * @notice Override of "Approve" function to allow the `spender` to transfer up to `amount` from `src`
     * @dev This will overwrite the approval amount for `spender` except in the case of spender is Time Lock Registry
     * and is subject to issues noted [here](https://eips.ethereum.org/EIPS/eip-20#approve)
     * @param spender The address of the account which may transfer tokens
     * @param rawAmount The number of tokens that are approved (2^256-1 means infinite)
     * @return Whether or not the approval succeeded
     */
    function approve(address spender, uint256 rawAmount) public override nonReentrant returns (bool) {
        require(spender != address(0), 'TimeLockedToken::approve: spender cannot be zero address');
        require(spender != msg.sender, 'TimeLockedToken::approve: spender cannot be the msg.sender');

        uint96 amount;
        if (rawAmount == uint256(-1)) {
            amount = uint96(-1);
        } else {
            amount = safe96(rawAmount, 'TimeLockedToken::approve: amount exceeds 96 bits');
        }

        // There is no option to decreaseAllowance to timeLockRegistry in case of vested tokens
        if ((spender == address(timeLockRegistry)) && (amount < allowance(msg.sender, address(timeLockRegistry)))) {
            amount = safe96(
                allowance(msg.sender, address(timeLockRegistry)),
                'TimeLockedToken::approve: cannot decrease allowance to timelockregistry'
            );
        }
        _approve(msg.sender, spender, amount);
        emit Approval(msg.sender, spender, amount);
        return true;
    }

    /**
     * PRIVILEGED GOVERNANCE FUNCTION. Override the Increase of allowances of ERC20 with special conditions for vesting
     *
     * @notice Atomically increases the allowance granted to `spender` by the caller.
     *
     * @dev This is an override with respect to the fulfillment of vesting conditions along the way
     * However an user can increase allowance many times, it will never be able to transfer locked tokens during vesting period
     * @return Whether or not the increaseAllowance succeeded
     */
    function increaseAllowance(address spender, uint256 addedValue) public override nonReentrant returns (bool) {
        require(
            unlockedBalance(msg.sender) >= allowance(msg.sender, spender).add(addedValue) ||
                spender == address(timeLockRegistry),
            'TimeLockedToken::increaseAllowance:Not enough unlocked tokens'
        );
        require(spender != address(0), 'TimeLockedToken::increaseAllowance:Spender cannot be zero address');
        require(spender != msg.sender, 'TimeLockedToken::increaseAllowance:Spender cannot be the msg.sender');
        _approve(msg.sender, spender, allowance(msg.sender, spender).add(addedValue));
        return true;
    }

    /**
     * PRIVILEGED GOVERNANCE FUNCTION. Override the decrease of allowances of ERC20 with special conditions for vesting
     *
     * @notice Atomically decrease the allowance granted to `spender` by the caller.
     *
     * @dev Atomically decreases the allowance granted to `spender` by the caller.
     * This is an override with respect to the fulfillment of vesting conditions along the way
     * An user cannot decrease the allowance to the Time Lock Registry who is in charge of vesting conditions
     * @return Whether or not the decreaseAllowance succeeded
     */
    function decreaseAllowance(address spender, uint256 subtractedValue) public override nonReentrant returns (bool) {
        require(spender != address(0), 'TimeLockedToken::decreaseAllowance:Spender cannot be zero address');
        require(spender != msg.sender, 'TimeLockedToken::decreaseAllowance:Spender cannot be the msg.sender');
        require(
            allowance(msg.sender, spender) >= subtractedValue,
            'TimeLockedToken::decreaseAllowance:Underflow condition'
        );

        // There is no option to decreaseAllowance to timeLockRegistry in case of vested tokens
        require(
            address(spender) != address(timeLockRegistry),
            'TimeLockedToken::decreaseAllowance:cannot decrease allowance to timeLockRegistry'
        );

        _approve(msg.sender, spender, allowance(msg.sender, spender).sub(subtractedValue));
        return true;
    }

    /* ============ Internal Only Function ============ */

    /**
     * PRIVILEGED GOVERNANCE FUNCTION. Override the _transfer of ERC20 BABL tokens only allowing the transfer of unlocked tokens
     *
     * @dev Transfer function which includes only unlocked tokens
     * Locked tokens can always be transfered back to the returns address
     * Transferring to owner allows re-issuance of funds through registry
     *
     * @param _from The address to send tokens from
     * @param _to The address that will receive the tokens
     * @param _value The amount of tokens to be transferred
     */
    function _transfer(
        address _from,
        address _to,
        uint256 _value
    ) internal override onlyUnpaused {
        require(_from != address(0), 'TimeLockedToken:: _transfer: cannot transfer from the zero address');
        require(_to != address(0), 'TimeLockedToken:: _transfer: cannot transfer to the zero address');
        require(
            _to != address(this),
            'TimeLockedToken:: _transfer: do not transfer tokens to the token contract itself'
        );

        require(balanceOf(_from) >= _value, 'TimeLockedToken:: _transfer: insufficient balance');

        // check if enough unlocked balance to transfer
        require(unlockedBalance(_from) >= _value, 'TimeLockedToken:: _transfer: attempting to transfer locked funds');
        super._transfer(_from, _to, _value);
        // voting power
        _moveDelegates(
            delegates[_from],
            delegates[_to],
            safe96(_value, 'TimeLockedToken:: _transfer: uint96 overflow')
        );
    }

    /**
     * PRIVILEGED GOVERNANCE FUNCTION. Disable BABL token transfer until certain conditions are met
     *
     * @dev Override the _beforeTokenTransfer of ERC20 BABL tokens until certain conditions are met:
     * Only allowing minting or transfers from Time Lock Registry and Rewards Distributor until transfers are allowed in the controller
     * Transferring to owner allows re-issuance of funds through registry
     *
     * @param _from The address to send tokens from
     * @param _to The address that will receive the tokens
     * @param _value The amount of tokens to be transferred
     */

    // Disable garden token transfers. Allow minting and burning.
    function _beforeTokenTransfer(
        address _from,
        address _to,
        uint256 _value
    ) internal virtual override {
        super._beforeTokenTransfer(_from, _to, _value);
        _require(
            _from == address(0) ||
                _from == address(timeLockRegistry) ||
                _from == address(rewardsDistributor) ||
                _to == address(timeLockRegistry) ||
                tokenTransfersEnabled,
            Errors.BABL_TRANSFERS_DISABLED
        );
    }

    /**
     * PRIVILEGED GOVERNANCE FUNCTION. Cancel and remove locked tokens due to non-completion of  vesting period
     * applied only by Time Lock Registry and specifically to Team or Advisors
     *
     * @dev Cancel distribution registration
     * @param lockedAccount that should have its still locked distribution removed due to non-completion of its vesting period
     */
    function _cancelVestedTokensFromTimeLock(address lockedAccount) internal onlyTimeLockRegistry returns (uint256) {
        require(distribution[lockedAccount] != 0, 'TimeLockedToken::cancelTokens:Not registered');

        // get an update on locked amount from distributions at this precise moment
        uint256 loosingAmount = lockedBalance(lockedAccount);

        require(loosingAmount > 0, 'TimeLockedToken::cancelTokens:There are no more locked tokens');
        require(
            vestedToken[lockedAccount].teamOrAdvisor == true,
            'TimeLockedToken::cancelTokens:cannot cancel locked tokens to Investors'
        );

        // set distribution mapping to 0
        delete distribution[lockedAccount];

        // set tokenVested mapping to 0
        delete vestedToken[lockedAccount];

        // transfer only locked tokens back to TimeLockRegistry Owner (msg.sender)
        require(
            transferFrom(lockedAccount, address(timeLockRegistry), loosingAmount),
            'TimeLockedToken::cancelTokens:Transfer failed'
        );

        // emit cancel event
        emit Cancel(lockedAccount, loosingAmount);

        return loosingAmount;
    }
}

File 4 of 18 : Address.sol
// SPDX-License-Identifier: MIT

pragma solidity >=0.6.2 <0.8.0;

/**
 * @dev Collection of functions related to the address type
 */
library Address {
    /**
     * @dev Returns true if `account` is a contract.
     *
     * [IMPORTANT]
     * ====
     * It is unsafe to assume that an address for which this function returns
     * false is an externally-owned account (EOA) and not a contract.
     *
     * Among others, `isContract` will return false for the following
     * types of addresses:
     *
     *  - an externally-owned account
     *  - a contract in construction
     *  - an address where a contract will be created
     *  - an address where a contract lived, but was destroyed
     * ====
     */
    function isContract(address account) internal view returns (bool) {
        // This method relies on extcodesize, which returns 0 for contracts in
        // construction, since the code is only stored at the end of the
        // constructor execution.

        uint256 size;
        // solhint-disable-next-line no-inline-assembly
        assembly { size := extcodesize(account) }
        return size > 0;
    }

    /**
     * @dev Replacement for Solidity's `transfer`: sends `amount` wei to
     * `recipient`, forwarding all available gas and reverting on errors.
     *
     * https://eips.ethereum.org/EIPS/eip-1884[EIP1884] increases the gas cost
     * of certain opcodes, possibly making contracts go over the 2300 gas limit
     * imposed by `transfer`, making them unable to receive funds via
     * `transfer`. {sendValue} removes this limitation.
     *
     * https://diligence.consensys.net/posts/2019/09/stop-using-soliditys-transfer-now/[Learn more].
     *
     * IMPORTANT: because control is transferred to `recipient`, care must be
     * taken to not create reentrancy vulnerabilities. Consider using
     * {ReentrancyGuard} or the
     * https://solidity.readthedocs.io/en/v0.5.11/security-considerations.html#use-the-checks-effects-interactions-pattern[checks-effects-interactions pattern].
     */
    function sendValue(address payable recipient, uint256 amount) internal {
        require(address(this).balance >= amount, "Address: insufficient balance");

        // solhint-disable-next-line avoid-low-level-calls, avoid-call-value
        (bool success, ) = recipient.call{ value: amount }("");
        require(success, "Address: unable to send value, recipient may have reverted");
    }

    /**
     * @dev Performs a Solidity function call using a low level `call`. A
     * plain`call` is an unsafe replacement for a function call: use this
     * function instead.
     *
     * If `target` reverts with a revert reason, it is bubbled up by this
     * function (like regular Solidity function calls).
     *
     * Returns the raw returned data. To convert to the expected return value,
     * use https://solidity.readthedocs.io/en/latest/units-and-global-variables.html?highlight=abi.decode#abi-encoding-and-decoding-functions[`abi.decode`].
     *
     * Requirements:
     *
     * - `target` must be a contract.
     * - calling `target` with `data` must not revert.
     *
     * _Available since v3.1._
     */
    function functionCall(address target, bytes memory data) internal returns (bytes memory) {
      return functionCall(target, data, "Address: low-level call failed");
    }

    /**
     * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], but with
     * `errorMessage` as a fallback revert reason when `target` reverts.
     *
     * _Available since v3.1._
     */
    function functionCall(address target, bytes memory data, string memory errorMessage) internal returns (bytes memory) {
        return functionCallWithValue(target, data, 0, errorMessage);
    }

    /**
     * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
     * but also transferring `value` wei to `target`.
     *
     * Requirements:
     *
     * - the calling contract must have an ETH balance of at least `value`.
     * - the called Solidity function must be `payable`.
     *
     * _Available since v3.1._
     */
    function functionCallWithValue(address target, bytes memory data, uint256 value) internal returns (bytes memory) {
        return functionCallWithValue(target, data, value, "Address: low-level call with value failed");
    }

    /**
     * @dev Same as {xref-Address-functionCallWithValue-address-bytes-uint256-}[`functionCallWithValue`], but
     * with `errorMessage` as a fallback revert reason when `target` reverts.
     *
     * _Available since v3.1._
     */
    function functionCallWithValue(address target, bytes memory data, uint256 value, string memory errorMessage) internal returns (bytes memory) {
        require(address(this).balance >= value, "Address: insufficient balance for call");
        require(isContract(target), "Address: call to non-contract");

        // solhint-disable-next-line avoid-low-level-calls
        (bool success, bytes memory returndata) = target.call{ value: value }(data);
        return _verifyCallResult(success, returndata, errorMessage);
    }

    /**
     * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
     * but performing a static call.
     *
     * _Available since v3.3._
     */
    function functionStaticCall(address target, bytes memory data) internal view returns (bytes memory) {
        return functionStaticCall(target, data, "Address: low-level static call failed");
    }

    /**
     * @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`],
     * but performing a static call.
     *
     * _Available since v3.3._
     */
    function functionStaticCall(address target, bytes memory data, string memory errorMessage) internal view returns (bytes memory) {
        require(isContract(target), "Address: static call to non-contract");

        // solhint-disable-next-line avoid-low-level-calls
        (bool success, bytes memory returndata) = target.staticcall(data);
        return _verifyCallResult(success, returndata, errorMessage);
    }

    /**
     * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
     * but performing a delegate call.
     *
     * _Available since v3.4._
     */
    function functionDelegateCall(address target, bytes memory data) internal returns (bytes memory) {
        return functionDelegateCall(target, data, "Address: low-level delegate call failed");
    }

    /**
     * @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`],
     * but performing a delegate call.
     *
     * _Available since v3.4._
     */
    function functionDelegateCall(address target, bytes memory data, string memory errorMessage) internal returns (bytes memory) {
        require(isContract(target), "Address: delegate call to non-contract");

        // solhint-disable-next-line avoid-low-level-calls
        (bool success, bytes memory returndata) = target.delegatecall(data);
        return _verifyCallResult(success, returndata, errorMessage);
    }

    function _verifyCallResult(bool success, bytes memory returndata, string memory errorMessage) private pure returns(bytes memory) {
        if (success) {
            return returndata;
        } else {
            // Look for revert reason and bubble it up if present
            if (returndata.length > 0) {
                // The easiest way to bubble the revert reason is using memory via assembly

                // solhint-disable-next-line no-inline-assembly
                assembly {
                    let returndata_size := mload(returndata)
                    revert(add(32, returndata), returndata_size)
                }
            } else {
                revert(errorMessage);
            }
        }
    }
}

File 5 of 18 : ReentrancyGuard.sol
// SPDX-License-Identifier: MIT

pragma solidity >=0.6.0 <0.8.0;

/**
 * @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 6 of 18 : IBabController.sol
/*
    Copyright 2021 Babylon Finance.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

    http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.

    SPDX-License-Identifier: Apache License, Version 2.0
*/
pragma solidity 0.7.6;

/**
 * @title IBabController
 * @author Babylon Finance
 *
 * Interface for interacting with BabController
 */
interface IBabController {
    /* ============ Functions ============ */

    function createGarden(
        address _reserveAsset,
        string memory _name,
        string memory _symbol,
        string memory _tokenURI,
        uint256 _seed,
        uint256[] calldata _gardenParams,
        uint256 _initialContribution,
        bool[] memory _publicGardenStrategistsStewards,
        uint256[] memory _profitSharing
    ) external payable returns (address);

    function removeGarden(address _garden) external;

    function addReserveAsset(address _reserveAsset) external;

    function removeReserveAsset(address _reserveAsset) external;

    function disableGarden(address _garden) external;

    function editPriceOracle(address _priceOracle) external;

    function editIshtarGate(address _ishtarGate) external;

    function editGardenValuer(address _gardenValuer) external;

    function editRewardsDistributor(address _rewardsDistributor) external;

    function editTreasury(address _newTreasury) external;

    function editGardenFactory(address _newGardenFactory) external;

    function editGardenNFT(address _newGardenNFT) external;

    function editStrategyNFT(address _newStrategyNFT) external;

    function editStrategyFactory(address _newStrategyFactory) external;

    function editBabylonViewer(address _newBabylonViewer) external;

    function addIntegration(string memory _name, address _integration) external;

    function editIntegration(string memory _name, address _integration) external;

    function removeIntegration(string memory _name) external;

    function setOperation(uint8 _kind, address _operation) external;

    function setDefaultTradeIntegration(address _newDefaultTradeIntegation) external;

    function addKeeper(address _keeper) external;

    function addKeepers(address[] memory _keepers) external;

    function removeKeeper(address _keeper) external;

    function enableGardenTokensTransfers() external;

    function enableBABLMiningProgram() external;

    function setAllowPublicGardens() external;

    function editLiquidityReserve(address _reserve, uint256 _minRiskyPairLiquidityEth) external;

    function maxContributorsPerGarden() external view returns (uint256);

    function gardenCreationIsOpen() external view returns (bool);

    function openPublicGardenCreation() external;

    function setMaxContributorsPerGarden(uint256 _newMax) external;

    function owner() external view returns (address);

    function guardianGlobalPaused() external view returns (bool);

    function guardianPaused(address _address) external view returns (bool);

    function setPauseGuardian(address _guardian) external;

    function setGlobalPause(bool _state) external returns (bool);

    function setSomePause(address[] memory _address, bool _state) external returns (bool);

    function isPaused(address _contract) external view returns (bool);

    function priceOracle() external view returns (address);

    function gardenValuer() external view returns (address);

    function gardenNFT() external view returns (address);

    function babViewer() external view returns (address);

    function strategyNFT() external view returns (address);

    function rewardsDistributor() external view returns (address);

    function gardenFactory() external view returns (address);

    function treasury() external view returns (address);

    function ishtarGate() external view returns (address);

    function strategyFactory() external view returns (address);

    function defaultTradeIntegration() external view returns (address);

    function gardenTokensTransfersEnabled() external view returns (bool);

    function bablMiningProgramEnabled() external view returns (bool);

    function allowPublicGardens() external view returns (bool);

    function enabledOperations(uint256 _kind) external view returns (address);

    function getProfitSharing()
        external
        view
        returns (
            uint256,
            uint256,
            uint256
        );

    function getBABLSharing()
        external
        view
        returns (
            uint256,
            uint256,
            uint256,
            uint256
        );

    function getGardens() external view returns (address[] memory);

    function getOperations() external view returns (address[20] memory);

    function isGarden(address _garden) external view returns (bool);

    function getIntegrationByName(string memory _name) external view returns (address);

    function getIntegrationWithHash(bytes32 _nameHashP) external view returns (address);

    function isValidReserveAsset(address _reserveAsset) external view returns (bool);

    function isValidKeeper(address _keeper) external view returns (bool);

    function isSystemContract(address _contractAddress) external view returns (bool);

    function isValidIntegration(string memory _name, address _integration) external view returns (bool);

    function getMinCooldownPeriod() external view returns (uint256);

    function getMaxCooldownPeriod() external view returns (uint256);

    function protocolPerformanceFee() external view returns (uint256);

    function protocolManagementFee() external view returns (uint256);

    function minLiquidityPerReserve(address _reserve) external view returns (uint256);
}

File 7 of 18 : TimeLockRegistry.sol
/*
    Copyright 2021 Babylon Finance.
    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at
    http://www.apache.org/licenses/LICENSE-2.0
    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
    SPDX-License-Identifier: Apache License, Version 2.0
*/

pragma solidity 0.7.6;
pragma experimental ABIEncoderV2;
import {Ownable} from '@openzeppelin/contracts/access/Ownable.sol';
import {Address} from '@openzeppelin/contracts/utils/Address.sol';
import {SafeERC20} from '@openzeppelin/contracts/token/ERC20/SafeERC20.sol';
import {IERC20} from '@openzeppelin/contracts/token/ERC20/IERC20.sol';

import {TimeLockedToken} from './TimeLockedToken.sol';
import {AddressArrayUtils} from '../lib/AddressArrayUtils.sol';

import {LowGasSafeMath} from '../lib/LowGasSafeMath.sol';

/**
 * @title TimeLockRegistry
 * @notice Register Lockups for TimeLocked ERC20 Token BABL (e.g. vesting)
 * @author Babylon Finance
 * @dev This contract allows owner to register distributions for a TimeLockedToken
 *
 * To register a distribution, register method should be called by the owner.
 * claim() should be called only by the BABL Token smartcontract (modifier onlyBABLToken)
 *  when any account registered to receive tokens make its own claim
 * If case of a mistake, owner can cancel registration before the claim is done by the account
 *
 * Note this contract address must be setup in the TimeLockedToken's contract pointing
 * to interact with (e.g. setTimeLockRegistry() function)
 */

contract TimeLockRegistry is Ownable {
    using LowGasSafeMath for uint256;
    using Address for address;
    using AddressArrayUtils for address[];

    /* ============ Events ============ */

    event Register(address receiver, uint256 distribution);
    event Cancel(address receiver, uint256 distribution);
    event Claim(address account, uint256 distribution);

    /* ============ Modifiers ============ */

    modifier onlyBABLToken() {
        require(msg.sender == address(token), 'only BABL Token');
        _;
    }

    /* ============ State Variables ============ */

    // time locked token
    TimeLockedToken public token;

    /**
     * @notice The profile of each token owner under vesting conditions and its special conditions
     * @param receiver Account being registered
     * @param investorType Indicates whether or not is a Team member (true = team member / advisor, false = private investor)
     * @param vestingStarting Date When the vesting begins for such token owner
     * @param distribution Tokens amount that receiver is due to get
     */
    struct Registration {
        address receiver;
        uint256 distribution;
        bool investorType;
        uint256 vestingStartingDate;
    }

    /**
     * @notice The profile of each token owner under vesting conditions and its special conditions
     * @param team Indicates whether or not is a Team member (true = team member / advisor, false = private investor)
     * @param vestingBegin When the vesting begins for such token owner
     * @param vestingEnd When the vesting ends for such token owner
     * @param lastClaim When the last claim was done
     */
    struct TokenVested {
        bool team;
        bool cliff;
        uint256 vestingBegin;
        uint256 vestingEnd;
        uint256 lastClaim;
    }

    /// @notice A record of token owners under vesting conditions for each account, by index
    mapping(address => TokenVested) public tokenVested;

    // mapping from token owners under vesting conditions to BABL due amount (e.g. SAFT addresses, team members, advisors)
    mapping(address => uint256) public registeredDistributions;

    // array of all registrations
    address[] public registrations;

    // total amount of tokens registered
    uint256 public totalTokens;

    // vesting for Team Members
    uint256 private constant teamVesting = 365 days * 4;

    // vesting for Investors and Advisors
    uint256 private constant investorVesting = 365 days * 3;

    /* ============ Functions ============ */

    /* ============ Constructor ============ */

    /**
     * @notice Construct a new Time Lock Registry and gives ownership to sender
     * @param _token TimeLockedToken contract to use in this registry
     */
    constructor(TimeLockedToken _token) {
        token = _token;
    }

    /* ============ External Functions ============ */

    /* ============ External Getter Functions ============ */

    /**
     * Gets registrations
     *
     * @return  address[]        Returns list of registrations
     */

    function getRegistrations() external view returns (address[] memory) {
        return registrations;
    }

    /* ===========  Token related Gov Functions ====== */

    /**
     * PRIVILEGED GOVERNANCE FUNCTION
     *
     * @notice Register multiple investors/team in a batch
     * @param _registrations Registrations to process
     */
    function registerBatch(Registration[] memory _registrations) external onlyOwner {
        for (uint256 i = 0; i < _registrations.length; i++) {
            register(
                _registrations[i].receiver,
                _registrations[i].distribution,
                _registrations[i].investorType,
                _registrations[i].vestingStartingDate
            );
        }
    }

    /**
     * PRIVILEGED GOVERNANCE FUNCTION
     *
     * @notice Register new account under vesting conditions (Team, Advisors, Investors e.g. SAFT purchaser)
     * @param receiver Address belonging vesting conditions
     * @param distribution Tokens amount that receiver is due to get
     */
    function register(
        address receiver,
        uint256 distribution,
        bool investorType,
        uint256 vestingStartingDate
    ) public onlyOwner {
        require(receiver != address(0), 'TimeLockRegistry::register: cannot register the zero address');
        require(
            receiver != address(this),
            'TimeLockRegistry::register: Time Lock Registry contract cannot be an investor'
        );
        require(distribution != 0, 'TimeLockRegistry::register: Distribution = 0');
        require(
            registeredDistributions[receiver] == 0,
            'TimeLockRegistry::register:Distribution for this address is already registered'
        );
        require(block.timestamp >= 1614553200, 'Cannot register earlier than March 2021'); // 1614553200 is UNIX TIME of 2021 March the 1st
        require(totalTokens.add(distribution) <= IERC20(token).balanceOf(address(this)), 'Not enough tokens');

        totalTokens = totalTokens.add(distribution);
        // register distribution
        registeredDistributions[receiver] = distribution;
        registrations.push(receiver);

        // register token vested conditions
        TokenVested storage newTokenVested = tokenVested[receiver];
        newTokenVested.team = investorType;
        newTokenVested.vestingBegin = vestingStartingDate;

        if (newTokenVested.team == true) {
            newTokenVested.vestingEnd = vestingStartingDate.add(teamVesting);
        } else {
            newTokenVested.vestingEnd = vestingStartingDate.add(investorVesting);
        }
        newTokenVested.lastClaim = vestingStartingDate;

        tokenVested[receiver] = newTokenVested;

        // emit register event
        emit Register(receiver, distribution);
    }

    /**
     * PRIVILEGED GOVERNANCE FUNCTION. Cancel distribution registration in case of mistake and before a claim is done
     *
     * @notice Cancel distribution registration
     * @dev A claim has not to be done earlier
     * @param receiver Address that should have it's distribution removed
     * @return Whether or not it succeeded
     */
    function cancelRegistration(address receiver) external onlyOwner returns (bool) {
        require(registeredDistributions[receiver] != 0, 'Not registered');

        // get amount from distributions
        uint256 amount = registeredDistributions[receiver];

        // set distribution mapping to 0
        delete registeredDistributions[receiver];

        // set tokenVested mapping to 0
        delete tokenVested[receiver];

        // remove from the list of all registrations
        registrations.remove(receiver);

        // decrease total tokens
        totalTokens = totalTokens.sub(amount);

        // emit cancel event
        emit Cancel(receiver, amount);

        return true;
    }

    /**
     * PRIVILEGED GOVERNANCE FUNCTION. Cancel distribution registration in case of mistake and before a claim is done
     *
     * @notice Cancel already delivered tokens. It might only apply when non-completion of vesting period of Team members or Advisors
     * @dev An automatic override allowance is granted during the claim process
     * @param account Address that should have it's distribution removed
     * @return Whether or not it succeeded
     */
    function cancelDeliveredTokens(address account) external onlyOwner returns (bool) {
        uint256 loosingAmount = token.cancelVestedTokens(account);

        // emit cancel event
        emit Cancel(account, loosingAmount);
        return true;
    }

    /**
     * PRIVILEGED GOVERNANCE FUNCTION. Recover tokens in Time Lock Registry smartcontract address by the owner
     *
     * @notice Send tokens from smartcontract address to the owner.
     * It might only apply after a cancellation of vested tokens
     * @param amount Amount to be recovered by the owner of the Time Lock Registry smartcontract from its balance
     * @return Whether or not it succeeded
     */
    function transferToOwner(uint256 amount) external onlyOwner returns (bool) {
        SafeERC20.safeTransfer(token, msg.sender, amount);
        return true;
    }

    /**
     * PRIVILEGED GOVERNANCE FUNCTION. Claim locked tokens by the registered account
     *
     * @notice Claim tokens due amount.
     * @dev Claim is done by the user in the TimeLocked contract and the contract is the only allowed to call
     * this function on behalf of the user to make the claim
     * @return The amount of tokens registered and delivered after the claim
     */
    function claim(address _receiver) external onlyBABLToken returns (uint256) {
        require(registeredDistributions[_receiver] != 0, 'Not registered');

        // get amount from distributions
        uint256 amount = registeredDistributions[_receiver];
        TokenVested storage claimTokenVested = tokenVested[_receiver];

        claimTokenVested.lastClaim = block.timestamp;

        // set distribution mapping to 0
        delete registeredDistributions[_receiver];

        // decrease total tokens
        totalTokens = totalTokens.sub(amount);

        // register lockup in TimeLockedToken
        // this will transfer funds from this contract and lock them for sender
        token.registerLockup(
            _receiver,
            amount,
            claimTokenVested.team,
            claimTokenVested.vestingBegin,
            claimTokenVested.vestingEnd,
            claimTokenVested.lastClaim
        );

        // set tokenVested mapping to 0
        delete tokenVested[_receiver];

        // emit claim event
        emit Claim(_receiver, amount);

        return amount;
    }

    /* ============ Getter Functions ============ */

    function checkVesting(address address_)
        external
        view
        returns (
            bool team,
            uint256 start,
            uint256 end,
            uint256 last
        )
    {
        TokenVested storage checkTokenVested = tokenVested[address_];

        return (
            checkTokenVested.team,
            checkTokenVested.vestingBegin,
            checkTokenVested.vestingEnd,
            checkTokenVested.lastClaim
        );
    }

    function checkRegisteredDistribution(address address_) external view returns (uint256 amount) {
        return registeredDistributions[address_];
    }
}

File 8 of 18 : IRewardsDistributor.sol
/*
    Copyright 2021 Babylon Finance.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at
    http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.

    SPDX-License-Identifier: Apache License, Version 2.0
*/

pragma solidity 0.7.6;

/**
 * @title IRewardsDistributor
 * @author Babylon Finance
 *
 * Interface for the distribute rewards of the BABL Mining Program.
 */

interface IRewardsDistributor {
    // Structs
    struct PrincipalPerTimestamp {
        uint256 principal;
        uint256 time;
        uint256 timeListPointer;
    }

    // solhint-disable-next-line
    function START_TIME() external view returns (uint256);

    function updateProtocolPrincipal(uint256 _capital, bool _addOrSubstract) external;

    function getStrategyRewards(address _strategy) external view returns (uint96);

    function sendTokensToContributor(address _to, uint256 _amount) external;

    function startBABLRewards() external;

    function getRewards(
        address _garden,
        address _contributor,
        address[] calldata _finalizedStrategies
    ) external view returns (uint256[] memory);

    function getContributorPower(
        address _garden,
        address _contributor,
        uint256 _from,
        uint256 _to
    ) external view returns (uint256);

    function getGardenProfitsSharing(address _garden) external view returns (uint256[3] memory);

    function setProfitRewards(
        address _garden,
        uint256 _strategistShare,
        uint256 _stewardsShare,
        uint256 _lpShare
    ) external;

    function updateGardenPowerAndContributor(
        address _garden,
        address _contributor,
        uint256 _previousBalance,
        bool _depositOrWithdraw,
        uint256 _pid
    ) external;

    function tokenSupplyPerQuarter(uint256 quarter) external view returns (uint96);

    function checkProtocol(uint256 _time)
        external
        view
        returns (
            uint256 principal,
            uint256 time,
            uint256 quarterBelonging,
            uint256 timeListPointer,
            uint256 power
        );

    function checkQuarter(uint256 _num)
        external
        view
        returns (
            uint256 quarterPrincipal,
            uint256 quarterNumber,
            uint256 quarterPower,
            uint96 supplyPerQuarter
        );
}

File 9 of 18 : VoteToken.sol
/*
    Copyright 2021 Babylon Finance.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at
    http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.

    SPDX-License-Identifier: Apache License, Version 2.0
*/

pragma solidity 0.7.6;

import {ERC20} from '@openzeppelin/contracts/token/ERC20/ERC20.sol';
import {IVoteToken} from '../interfaces/IVoteToken.sol';
import {ReentrancyGuard} from '@openzeppelin/contracts/utils/ReentrancyGuard.sol';
import {LowGasSafeMath} from '../lib/LowGasSafeMath.sol';
import {Context} from '@openzeppelin/contracts/utils/Context.sol';
import {Ownable} from '@openzeppelin/contracts/access/Ownable.sol';
import {Address} from '@openzeppelin/contracts/utils/Address.sol';

/**
 * @title VoteToken
 * @notice Custom token which tracks voting power for governance
 * @dev This is an abstraction of a fork of the Compound governance contract
 * VoteToken is used by BABL to allow tracking voting power
 * Checkpoints are created every time state is changed which record voting power
 * Inherits standard ERC20 behavior
 */

abstract contract VoteToken is Context, ERC20, Ownable, IVoteToken, ReentrancyGuard {
    using LowGasSafeMath for uint256;
    using Address for address;

    /* ============ Events ============ */

    event DelegateChanged(address indexed delegator, address indexed fromDelegate, address indexed toDelegate);
    event DelegateVotesChanged(address indexed delegate, uint256 previousBalance, uint256 newBalance);

    /* ============ Modifiers ============ */

    /* ============ State Variables ============ */

    /// @notice The EIP-712 typehash for the contract's domain
    bytes32 public constant DOMAIN_TYPEHASH =
        keccak256('EIP712Domain(string name,uint256 chainId,address verifyingContract)');

    /// @notice The EIP-712 typehash for the delegation struct used by the contract
    bytes32 public constant DELEGATION_TYPEHASH =
        keccak256('Delegation(address delegatee,uint256 nonce,uint256 expiry)');

    /// @dev A record of votes checkpoints for each account, by index
    mapping(address => address) public delegates;

    /// @notice A checkpoint for marking number of votes from a given block
    struct Checkpoint {
        uint32 fromBlock;
        uint96 votes;
    }

    /// @notice A record of votes checkpoints for each account, by index
    mapping(address => mapping(uint32 => Checkpoint)) public checkpoints;

    /// @notice The number of checkpoints for each account
    mapping(address => uint32) public numCheckpoints;

    /// @notice A record of states for signing / validating signatures
    mapping(address => uint256) public nonces;

    /* ============ Functions ============ */

    /* ============ Constructor ============ */

    constructor(string memory _name, string memory _symbol) ERC20(_name, _symbol) {}

    /* ============ External Functions ============ */

    /* ===========  Token related Gov Functions ====== */

    /**
     * PRIVILEGED GOVERNANCE FUNCTION. Delegating votes from msg.sender to delegatee
     *
     * @notice Delegate votes from `msg.sender` to `delegatee`
     * @param delegatee The address to delegate votes to
     */

    function delegate(address delegatee) external override {
        return _delegate(msg.sender, delegatee);
    }

    /**
     * PRIVILEGED GOVERNANCE FUNCTION. Delegate votes using signature to 'delegatee'
     *
     * @notice Delegates votes from signatory to `delegatee`
     * @param delegatee The address to delegate votes to
     * @param nonce The contract state required to match the signature
     * @param expiry The time at which to expire the signature
     * @param v The recovery byte of the signature
     * @param r Half of the ECDSA signature pair
     * @param s Half of the ECDSA signature pair
     */

    function delegateBySig(
        address delegatee,
        uint256 nonce,
        uint256 expiry,
        uint8 v,
        bytes32 r,
        bytes32 s,
        bool prefix
    ) external override {
        address signatory;
        bytes32 domainSeparator =
            keccak256(abi.encode(DOMAIN_TYPEHASH, keccak256(bytes(name())), getChainId(), address(this)));
        bytes32 structHash = keccak256(abi.encode(DELEGATION_TYPEHASH, delegatee, nonce, expiry));
        bytes32 digest = keccak256(abi.encodePacked('\x19\x01', domainSeparator, structHash));
        if (prefix) {
            bytes32 digestHash = keccak256(abi.encodePacked('\x19Ethereum Signed Message:\n32', digest));
            signatory = ecrecover(digestHash, v, r, s);
        } else {
            signatory = ecrecover(digest, v, r, s);
        }

        require(balanceOf(signatory) > 0, 'VoteToken::delegateBySig: invalid delegator');
        require(signatory != address(0), 'VoteToken::delegateBySig: invalid signature');
        require(nonce == nonces[signatory], 'VoteToken::delegateBySig: invalid nonce');
        nonces[signatory]++;
        require(block.timestamp <= expiry, 'VoteToken::delegateBySig: signature expired');
        return _delegate(signatory, delegatee);
    }

    /**
     * GOVERNANCE FUNCTION. Check Delegate votes using signature to 'delegatee'
     *
     * @notice Get current voting power for an account
     * @param account Account to get voting power for
     * @return Voting power for an account
     */
    function getCurrentVotes(address account) external view virtual override returns (uint96) {
        uint32 nCheckpoints = numCheckpoints[account];
        return nCheckpoints > 0 ? checkpoints[account][nCheckpoints - 1].votes : 0;
    }

    /**
     * GOVERNANCE FUNCTION. Get voting power at a specific block for an account
     *
     * @param account Account to get voting power for
     * @param blockNumber Block to get voting power at
     * @return Voting power for an account at specific block
     */
    function getPriorVotes(address account, uint256 blockNumber) external view virtual override returns (uint96) {
        require(blockNumber < block.number, 'BABLToken::getPriorVotes: not yet determined');

        uint32 nCheckpoints = numCheckpoints[account];
        if (nCheckpoints == 0) {
            return 0;
        }

        // First check most recent balance
        if (checkpoints[account][nCheckpoints - 1].fromBlock <= blockNumber) {
            return checkpoints[account][nCheckpoints - 1].votes;
        }

        // Next check implicit zero balance
        if (checkpoints[account][0].fromBlock > blockNumber) {
            return 0;
        }

        uint32 lower = 0;
        uint32 upper = nCheckpoints - 1;
        while (upper > lower) {
            uint32 center = upper - (upper - lower) / 2; // ceil, avoiding overflow
            Checkpoint memory cp = checkpoints[account][center];
            if (cp.fromBlock == blockNumber) {
                return cp.votes;
            } else if (cp.fromBlock < blockNumber) {
                lower = center;
            } else {
                upper = center - 1;
            }
        }
        return checkpoints[account][lower].votes;
    }

    function getMyDelegatee() external view override returns (address) {
        return delegates[msg.sender];
    }

    function getDelegatee(address account) external view override returns (address) {
        return delegates[account];
    }

    function getCheckpoints(address account, uint32 id)
        external
        view
        override
        returns (uint32 fromBlock, uint96 votes)
    {
        Checkpoint storage getCheckpoint = checkpoints[account][id];
        return (getCheckpoint.fromBlock, getCheckpoint.votes);
    }

    function getNumberOfCheckpoints(address account) external view override returns (uint32) {
        return numCheckpoints[account];
    }

    /* ============ Internal Only Function ============ */

    /**
     * GOVERNANCE FUNCTION. Make a delegation
     *
     * @dev Internal function to delegate voting power to an account
     * @param delegator The address of the account delegating votes from
     * @param delegatee The address to delegate votes to
     */

    function _delegate(address delegator, address delegatee) internal {
        address currentDelegate = delegates[delegator];
        uint96 delegatorBalance = safe96(_balanceOf(delegator), 'VoteToken::_delegate: uint96 overflow');
        delegates[delegator] = delegatee;

        emit DelegateChanged(delegator, currentDelegate, delegatee);

        _moveDelegates(currentDelegate, delegatee, delegatorBalance);
    }

    function _balanceOf(address account) internal view virtual returns (uint256) {
        return balanceOf(account);
    }

    /**
     * GOVERNANCE FUNCTION. Move the delegates
     *
     * @dev Internal function to move delegates between accounts
     * @param srcRep The address of the account delegating votes from
     * @param dstRep The address of the account delegating votes to
     * @param amount The voting power to move
     */
    function _moveDelegates(
        address srcRep,
        address dstRep,
        uint96 amount
    ) internal {
        if (srcRep != dstRep && amount > 0) {
            // It must not revert but do nothing in cases of address(0) being part of the move
            // Sub voting amount to source in case it is not the zero address (e.g. transfers)
            if (srcRep != address(0)) {
                uint32 srcRepNum = numCheckpoints[srcRep];
                uint96 srcRepOld = srcRepNum > 0 ? checkpoints[srcRep][srcRepNum - 1].votes : 0;
                uint96 srcRepNew = sub96(srcRepOld, amount, 'VoteToken::_moveDelegates: vote amount underflows');
                _writeCheckpoint(srcRep, srcRepNum, srcRepOld, srcRepNew);
            }
            if (dstRep != address(0)) {
                // Add it to destination in case it is not the zero address (e.g. any transfer of tokens or delegations except a first mint to a specific address)
                uint32 dstRepNum = numCheckpoints[dstRep];
                uint96 dstRepOld = dstRepNum > 0 ? checkpoints[dstRep][dstRepNum - 1].votes : 0;
                uint96 dstRepNew = add96(dstRepOld, amount, 'VoteToken::_moveDelegates: vote amount overflows');
                _writeCheckpoint(dstRep, dstRepNum, dstRepOld, dstRepNew);
            }
        }
    }

    /**
     * GOVERNANCE FUNCTION. Internal function to write a checkpoint for voting power
     *
     * @dev internal function to write a checkpoint for voting power
     * @param delegatee The address of the account delegating votes to
     * @param nCheckpoints The num checkpoint
     * @param oldVotes The previous voting power
     * @param newVotes The new voting power
     */
    function _writeCheckpoint(
        address delegatee,
        uint32 nCheckpoints,
        uint96 oldVotes,
        uint96 newVotes
    ) internal {
        uint32 blockNumber = safe32(block.number, 'VoteToken::_writeCheckpoint: block number exceeds 32 bits');

        if (nCheckpoints > 0 && checkpoints[delegatee][nCheckpoints - 1].fromBlock == blockNumber) {
            checkpoints[delegatee][nCheckpoints - 1].votes = newVotes;
        } else {
            checkpoints[delegatee][nCheckpoints] = Checkpoint(blockNumber, newVotes);
            numCheckpoints[delegatee] = nCheckpoints + 1;
        }

        emit DelegateVotesChanged(delegatee, oldVotes, newVotes);
    }

    /**
     * INTERNAL FUNCTION. Internal function to convert from uint256 to uint32
     *
     * @dev internal function to convert from uint256 to uint32
     */
    function safe32(uint256 n, string memory errorMessage) internal pure returns (uint32) {
        require(n < 2**32, errorMessage);
        return uint32(n);
    }

    /**
     * INTERNAL FUNCTION. Internal function to convert from uint256 to uint96
     *
     * @dev internal function to convert from uint256 to uint96
     */
    function safe96(uint256 n, string memory errorMessage) internal pure returns (uint96) {
        require(n < 2**96, errorMessage);
        return uint96(n);
    }

    /**
     * INTERNAL FUNCTION. Internal function to add two uint96 numbers
     *
     * @dev internal safe math function to add two uint96 numbers
     */
    function add96(
        uint96 a,
        uint96 b,
        string memory errorMessage
    ) internal pure returns (uint96) {
        uint96 c = a + b;
        require(c >= a, errorMessage);
        return c;
    }

    /**
     * INTERNAL FUNCTION. Internal function to subtract two uint96 numbers
     *
     * @dev internal safe math function to subtract two uint96 numbers
     */
    function sub96(
        uint96 a,
        uint96 b,
        string memory errorMessage
    ) internal pure returns (uint96) {
        require(b <= a, errorMessage);
        return a - b;
    }

    /**
     * INTERNAL FUNCTION. Internal function to get chain ID
     *
     * @dev internal function to get chain ID
     */
    function getChainId() internal pure returns (uint256) {
        uint256 chainId;
        assembly {
            chainId := chainid()
        }
        return chainId;
    }
}

File 10 of 18 : Ownable.sol
// SPDX-License-Identifier: MIT

pragma solidity >=0.6.0 <0.8.0;

import "../utils/Context.sol";
/**
 * @dev Contract module which provides a basic access control mechanism, where
 * there is an account (an owner) that can be granted exclusive access to
 * specific functions.
 *
 * By default, the owner account will be the one that deploys the contract. This
 * can later be changed with {transferOwnership}.
 *
 * This module is used through inheritance. It will make available the modifier
 * `onlyOwner`, which can be applied to your functions to restrict their use to
 * the owner.
 */
abstract contract Ownable is Context {
    address private _owner;

    event OwnershipTransferred(address indexed previousOwner, address indexed newOwner);

    /**
     * @dev Initializes the contract setting the deployer as the initial owner.
     */
    constructor () internal {
        address msgSender = _msgSender();
        _owner = msgSender;
        emit OwnershipTransferred(address(0), msgSender);
    }

    /**
     * @dev Returns the address of the current owner.
     */
    function owner() public view virtual returns (address) {
        return _owner;
    }

    /**
     * @dev Throws if called by any account other than the owner.
     */
    modifier onlyOwner() {
        require(owner() == _msgSender(), "Ownable: caller is not the owner");
        _;
    }

    /**
     * @dev Leaves the contract without owner. It will not be possible to call
     * `onlyOwner` functions anymore. Can only be called by the current owner.
     *
     * NOTE: Renouncing ownership will leave the contract without an owner,
     * thereby removing any functionality that is only available to the owner.
     */
    function renounceOwnership() public virtual onlyOwner {
        emit OwnershipTransferred(_owner, address(0));
        _owner = address(0);
    }

    /**
     * @dev Transfers ownership of the contract to a new account (`newOwner`).
     * Can only be called by the current owner.
     */
    function transferOwnership(address newOwner) public virtual onlyOwner {
        require(newOwner != address(0), "Ownable: new owner is the zero address");
        emit OwnershipTransferred(_owner, newOwner);
        _owner = newOwner;
    }
}

File 11 of 18 : BabylonErrors.sol
/*
    Original version by Synthetix.io
    https://docs.synthetix.io/contracts/source/libraries/safedecimalmath

    Adapted by Babylon Finance.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at
    http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.

    SPDX-License-Identifier: Apache License, Version 2.0
*/

pragma solidity 0.7.6;

// solhint-disable

/**
 * @notice Forked from https://github.com/balancer-labs/balancer-core-v2/blob/master/contracts/lib/helpers/BalancerErrors.sol
 * @dev Reverts if `condition` is false, with a revert reason containing `errorCode`. Only codes up to 999 are
 * supported.
 */
function _require(bool condition, uint256 errorCode) pure {
    if (!condition) _revert(errorCode);
}

/**
 * @dev Reverts with a revert reason containing `errorCode`. Only codes up to 999 are supported.
 */
function _revert(uint256 errorCode) pure {
    // We're going to dynamically create a revert string based on the error code, with the following format:
    // 'BAB#{errorCode}'
    // where the code is left-padded with zeroes to three digits (so they range from 000 to 999).
    //
    // We don't have revert strings embedded in the contract to save bytecode size: it takes much less space to store a
    // number (8 to 16 bits) than the individual string characters.
    //
    // The dynamic string creation algorithm that follows could be implemented in Solidity, but assembly allows for a
    // much denser implementation, again saving bytecode size. Given this function unconditionally reverts, this is a
    // safe place to rely on it without worrying about how its usage might affect e.g. memory contents.
    assembly {
        // First, we need to compute the ASCII representation of the error code. We assume that it is in the 0-999
        // range, so we only need to convert three digits. To convert the digits to ASCII, we add 0x30, the value for
        // the '0' character.

        let units := add(mod(errorCode, 10), 0x30)

        errorCode := div(errorCode, 10)
        let tenths := add(mod(errorCode, 10), 0x30)

        errorCode := div(errorCode, 10)
        let hundreds := add(mod(errorCode, 10), 0x30)

        // With the individual characters, we can now construct the full string. The "BAB#" part is a known constant
        // (0x42414223): we simply shift this by 24 (to provide space for the 3 bytes of the error code), and add the
        // characters to it, each shifted by a multiple of 8.
        // The revert reason is then shifted left by 200 bits (256 minus the length of the string, 7 characters * 8 bits
        // per character = 56) to locate it in the most significant part of the 256 slot (the beginning of a byte
        // array).

        let revertReason := shl(200, add(0x42414223000000, add(add(units, shl(8, tenths)), shl(16, hundreds))))

        // We can now encode the reason in memory, which can be safely overwritten as we're about to revert. The encoded
        // message will have the following layout:
        // [ revert reason identifier ] [ string location offset ] [ string length ] [ string contents ]

        // The Solidity revert reason identifier is 0x08c739a0, the function selector of the Error(string) function. We
        // also write zeroes to the next 28 bytes of memory, but those are about to be overwritten.
        mstore(0x0, 0x08c379a000000000000000000000000000000000000000000000000000000000)
        // Next is the offset to the location of the string, which will be placed immediately after (20 bytes away).
        mstore(0x04, 0x0000000000000000000000000000000000000000000000000000000000000020)
        // The string length is fixed: 7 characters.
        mstore(0x24, 7)
        // Finally, the string itself is stored.
        mstore(0x44, revertReason)

        // Even if the string is only 7 bytes long, we need to return a full 32 byte slot containing it. The length of
        // the encoded message is therefore 4 + 32 + 32 + 32 = 100.
        revert(0, 100)
    }
}

library Errors {
    // Max deposit limit needs to be under the limit
    uint256 internal constant MAX_DEPOSIT_LIMIT = 0;
    // Creator needs to deposit
    uint256 internal constant MIN_CONTRIBUTION = 1;
    // Min Garden token supply >= 0
    uint256 internal constant MIN_TOKEN_SUPPLY = 2;
    // Deposit hardlock needs to be at least 1 block
    uint256 internal constant DEPOSIT_HARDLOCK = 3;
    // Needs to be at least the minimum
    uint256 internal constant MIN_LIQUIDITY = 4;
    // _reserveAssetQuantity is not equal to msg.value
    uint256 internal constant MSG_VALUE_DO_NOT_MATCH = 5;
    // Withdrawal amount has to be equal or less than msg.sender balance
    uint256 internal constant MSG_SENDER_TOKENS_DO_NOT_MATCH = 6;
    // Tokens are staked
    uint256 internal constant TOKENS_STAKED = 7;
    // Balance too low
    uint256 internal constant BALANCE_TOO_LOW = 8;
    // msg.sender doesn't have enough tokens
    uint256 internal constant MSG_SENDER_TOKENS_TOO_LOW = 9;
    //  There is an open redemption window already
    uint256 internal constant REDEMPTION_OPENED_ALREADY = 10;
    // Cannot request twice in the same window
    uint256 internal constant ALREADY_REQUESTED = 11;
    // Rewards and profits already claimed
    uint256 internal constant ALREADY_CLAIMED = 12;
    // Value have to be greater than zero
    uint256 internal constant GREATER_THAN_ZERO = 13;
    // Must be reserve asset
    uint256 internal constant MUST_BE_RESERVE_ASSET = 14;
    // Only contributors allowed
    uint256 internal constant ONLY_CONTRIBUTOR = 15;
    // Only controller allowed
    uint256 internal constant ONLY_CONTROLLER = 16;
    // Only creator allowed
    uint256 internal constant ONLY_CREATOR = 17;
    // Only keeper allowed
    uint256 internal constant ONLY_KEEPER = 18;
    // Fee is too high
    uint256 internal constant FEE_TOO_HIGH = 19;
    // Only strategy allowed
    uint256 internal constant ONLY_STRATEGY = 20;
    // Only active allowed
    uint256 internal constant ONLY_ACTIVE = 21;
    // Only inactive allowed
    uint256 internal constant ONLY_INACTIVE = 22;
    // Address should be not zero address
    uint256 internal constant ADDRESS_IS_ZERO = 23;
    // Not within range
    uint256 internal constant NOT_IN_RANGE = 24;
    // Value is too low
    uint256 internal constant VALUE_TOO_LOW = 25;
    // Value is too high
    uint256 internal constant VALUE_TOO_HIGH = 26;
    // Only strategy or protocol allowed
    uint256 internal constant ONLY_STRATEGY_OR_CONTROLLER = 27;
    // Normal withdraw possible
    uint256 internal constant NORMAL_WITHDRAWAL_POSSIBLE = 28;
    // User does not have permissions to join garden
    uint256 internal constant USER_CANNOT_JOIN = 29;
    // User does not have permissions to add strategies in garden
    uint256 internal constant USER_CANNOT_ADD_STRATEGIES = 30;
    // Only Protocol or garden
    uint256 internal constant ONLY_PROTOCOL_OR_GARDEN = 31;
    // Only Strategist
    uint256 internal constant ONLY_STRATEGIST = 32;
    // Only Integration
    uint256 internal constant ONLY_INTEGRATION = 33;
    // Only garden and data not set
    uint256 internal constant ONLY_GARDEN_AND_DATA_NOT_SET = 34;
    // Only active garden
    uint256 internal constant ONLY_ACTIVE_GARDEN = 35;
    // Contract is not a garden
    uint256 internal constant NOT_A_GARDEN = 36;
    // Not enough tokens
    uint256 internal constant STRATEGIST_TOKENS_TOO_LOW = 37;
    // Stake is too low
    uint256 internal constant STAKE_HAS_TO_AT_LEAST_ONE = 38;
    // Duration must be in range
    uint256 internal constant DURATION_MUST_BE_IN_RANGE = 39;
    // Max Capital Requested
    uint256 internal constant MAX_CAPITAL_REQUESTED = 41;
    // Votes are already resolved
    uint256 internal constant VOTES_ALREADY_RESOLVED = 42;
    // Voting window is closed
    uint256 internal constant VOTING_WINDOW_IS_OVER = 43;
    // Strategy needs to be active
    uint256 internal constant STRATEGY_NEEDS_TO_BE_ACTIVE = 44;
    // Max capital reached
    uint256 internal constant MAX_CAPITAL_REACHED = 45;
    // Capital is less then rebalance
    uint256 internal constant CAPITAL_IS_LESS_THAN_REBALANCE = 46;
    // Strategy is in cooldown period
    uint256 internal constant STRATEGY_IN_COOLDOWN = 47;
    // Strategy is not executed
    uint256 internal constant STRATEGY_IS_NOT_EXECUTED = 48;
    // Strategy is not over yet
    uint256 internal constant STRATEGY_IS_NOT_OVER_YET = 49;
    // Strategy is already finalized
    uint256 internal constant STRATEGY_IS_ALREADY_FINALIZED = 50;
    // No capital to unwind
    uint256 internal constant STRATEGY_NO_CAPITAL_TO_UNWIND = 51;
    // Strategy needs to be inactive
    uint256 internal constant STRATEGY_NEEDS_TO_BE_INACTIVE = 52;
    // Duration needs to be less
    uint256 internal constant DURATION_NEEDS_TO_BE_LESS = 53;
    // Can't sweep reserve asset
    uint256 internal constant CANNOT_SWEEP_RESERVE_ASSET = 54;
    // Voting window is opened
    uint256 internal constant VOTING_WINDOW_IS_OPENED = 55;
    // Strategy is executed
    uint256 internal constant STRATEGY_IS_EXECUTED = 56;
    // Min Rebalance Capital
    uint256 internal constant MIN_REBALANCE_CAPITAL = 57;
    // Not a valid strategy NFT
    uint256 internal constant NOT_STRATEGY_NFT = 58;
    // Garden Transfers Disabled
    uint256 internal constant GARDEN_TRANSFERS_DISABLED = 59;
    // Tokens are hardlocked
    uint256 internal constant TOKENS_HARDLOCKED = 60;
    // Max contributors reached
    uint256 internal constant MAX_CONTRIBUTORS = 61;
    // BABL Transfers Disabled
    uint256 internal constant BABL_TRANSFERS_DISABLED = 62;
    // Strategy duration range error
    uint256 internal constant DURATION_RANGE = 63;
    // Checks the min amount of voters
    uint256 internal constant MIN_VOTERS_CHECK = 64;
    // Ge contributor power error
    uint256 internal constant CONTRIBUTOR_POWER_CHECK_WINDOW = 65;
    // Not enough reserve set aside
    uint256 internal constant NOT_ENOUGH_RESERVE = 66;
    // Garden is already public
    uint256 internal constant GARDEN_ALREADY_PUBLIC = 67;
    // Withdrawal with penalty
    uint256 internal constant WITHDRAWAL_WITH_PENALTY = 68;
    // Withdrawal with penalty
    uint256 internal constant ONLY_MINING_ACTIVE = 69;
    // Overflow in supply
    uint256 internal constant OVERFLOW_IN_SUPPLY = 70;
    // Overflow in power
    uint256 internal constant OVERFLOW_IN_POWER = 71;
    // Not a system contract
    uint256 internal constant NOT_A_SYSTEM_CONTRACT = 72;
    // Strategy vs Garden mismatch
    uint256 internal constant STRATEGY_GARDEN_MISMATCH = 73;
    // Minimum quarters is 1
    uint256 internal constant QUARTERS_MIN_1 = 74;
    // Too many strategy operations
    uint256 internal constant TOO_MANY_OPS = 75;
    // Only operations
    uint256 internal constant ONLY_OPERATION = 76;
    // Strat params wrong length
    uint256 internal constant STRAT_PARAMS_LENGTH = 77;
    // Garden params wrong length
    uint256 internal constant GARDEN_PARAMS_LENGTH = 78;
    // Token names too long
    uint256 internal constant NAME_TOO_LONG = 79;
    // Contributor power overflows over garden power
    uint256 internal constant CONTRIBUTOR_POWER_OVERFLOW = 80;
    // Contributor power window out of bounds
    uint256 internal constant CONTRIBUTOR_POWER_CHECK_DEPOSITS = 81;
    // Contributor power window out of bounds
    uint256 internal constant NO_REWARDS_TO_CLAIM = 82;
    // Pause guardian paused this operation
    uint256 internal constant ONLY_UNPAUSED = 83;
    // Reentrant intent
    uint256 internal constant REENTRANT_CALL = 84;
    // Reserve asset not supported
    uint256 internal constant RESERVE_ASSET_NOT_SUPPORTED = 85;
    // Withdrawal/Deposit check min amount received
    uint256 internal constant RECEIVE_MIN_AMOUNT = 86;
    // Total Votes has to be positive
    uint256 internal constant TOTAL_VOTES_HAVE_TO_BE_POSITIVE = 87;
    // Signer has to be valid
    uint256 internal constant INVALID_SIGNER = 88;
    // Nonce has to be valid
    uint256 internal constant INVALID_NONCE = 89;
    // Garden is not public
    uint256 internal constant GARDEN_IS_NOT_PUBLIC = 90;
    // Setting max contributors
    uint256 internal constant MAX_CONTRIBUTORS_SET = 91;
    // Profit sharing mismatch for customized gardens
    uint256 internal constant PROFIT_SHARING_MISMATCH = 92;
}

File 12 of 18 : SafeERC20.sol
// SPDX-License-Identifier: MIT

pragma solidity >=0.6.0 <0.8.0;

import "./IERC20.sol";
import "../../math/SafeMath.sol";
import "../../utils/Address.sol";

/**
 * @title SafeERC20
 * @dev Wrappers around ERC20 operations that throw on failure (when the token
 * contract returns false). Tokens that return no value (and instead revert or
 * throw on failure) are also supported, non-reverting calls are assumed to be
 * successful.
 * To use this library you can add a `using SafeERC20 for IERC20;` statement to your contract,
 * which allows you to call the safe operations as `token.safeTransfer(...)`, etc.
 */
library SafeERC20 {
    using SafeMath for uint256;
    using Address for address;

    function safeTransfer(IERC20 token, address to, uint256 value) internal {
        _callOptionalReturn(token, abi.encodeWithSelector(token.transfer.selector, to, value));
    }

    function safeTransferFrom(IERC20 token, address from, address to, uint256 value) internal {
        _callOptionalReturn(token, abi.encodeWithSelector(token.transferFrom.selector, from, to, value));
    }

    /**
     * @dev Deprecated. This function has issues similar to the ones found in
     * {IERC20-approve}, and its usage is discouraged.
     *
     * Whenever possible, use {safeIncreaseAllowance} and
     * {safeDecreaseAllowance} instead.
     */
    function safeApprove(IERC20 token, address spender, uint256 value) internal {
        // safeApprove should only be called when setting an initial allowance,
        // or when resetting it to zero. To increase and decrease it, use
        // 'safeIncreaseAllowance' and 'safeDecreaseAllowance'
        // solhint-disable-next-line max-line-length
        require((value == 0) || (token.allowance(address(this), spender) == 0),
            "SafeERC20: approve from non-zero to non-zero allowance"
        );
        _callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, value));
    }

    function safeIncreaseAllowance(IERC20 token, address spender, uint256 value) internal {
        uint256 newAllowance = token.allowance(address(this), spender).add(value);
        _callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, newAllowance));
    }

    function safeDecreaseAllowance(IERC20 token, address spender, uint256 value) internal {
        uint256 newAllowance = token.allowance(address(this), spender).sub(value, "SafeERC20: decreased allowance below zero");
        _callOptionalReturn(token, abi.encodeWithSelector(token.approve.selector, spender, newAllowance));
    }

    /**
     * @dev Imitates a Solidity high-level call (i.e. a regular function call to a contract), relaxing the requirement
     * on the return value: the return value is optional (but if data is returned, it must not be false).
     * @param token The token targeted by the call.
     * @param data The call data (encoded using abi.encode or one of its variants).
     */
    function _callOptionalReturn(IERC20 token, bytes memory data) private {
        // We need to perform a low level call here, to bypass Solidity's return data size checking mechanism, since
        // we're implementing it ourselves. We use {Address.functionCall} to perform this call, which verifies that
        // the target address contains contract code and also asserts for success in the low-level call.

        bytes memory returndata = address(token).functionCall(data, "SafeERC20: low-level call failed");
        if (returndata.length > 0) { // Return data is optional
            // solhint-disable-next-line max-line-length
            require(abi.decode(returndata, (bool)), "SafeERC20: ERC20 operation did not succeed");
        }
    }
}

File 13 of 18 : IERC20.sol
// SPDX-License-Identifier: MIT

pragma solidity >=0.6.0 <0.8.0;

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

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

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

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

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

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

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

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

File 14 of 18 : AddressArrayUtils.sol
/*
    Copyright 2020 Set Labs Inc.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

    http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.

    SPDX-License-Identifier: Apache License, Version 2.0
*/

pragma solidity 0.7.6;

/**
 * @title AddressArrayUtils
 * @author Set Protocol
 *
 * Utility functions to handle Address Arrays
 */
library AddressArrayUtils {
    /**
     * Finds the index of the first occurrence of the given element.
     * @param A The input array to search
     * @param a The value to find
     * @return Returns (index and isIn) for the first occurrence starting from index 0
     */
    function indexOf(address[] memory A, address a) internal pure returns (uint256, bool) {
        uint256 length = A.length;
        for (uint256 i = 0; i < length; i++) {
            if (A[i] == a) {
                return (i, true);
            }
        }
        return (uint256(-1), false);
    }

    /**
     * Returns true if the value is present in the list. Uses indexOf internally.
     * @param A The input array to search
     * @param a The value to find
     * @return Returns isIn for the first occurrence starting from index 0
     */
    function contains(address[] memory A, address a) internal pure returns (bool) {
        (, bool isIn) = indexOf(A, a);
        return isIn;
    }

    /**
     * Returns true if there are 2 elements that are the same in an array
     * @param A The input array to search
     * @return Returns boolean for the first occurrence of a duplicate
     */
    function hasDuplicate(address[] memory A) internal pure returns (bool) {
        require(A.length > 0, 'A is empty');

        for (uint256 i = 0; i < A.length - 1; i++) {
            address current = A[i];
            for (uint256 j = i + 1; j < A.length; j++) {
                if (current == A[j]) {
                    return true;
                }
            }
        }
        return false;
    }

    /**
     * @param A The input array to search
     * @param a The address to remove
     * @return Returns the array with the object removed.
     */
    function remove(address[] memory A, address a) internal pure returns (address[] memory) {
        (uint256 index, bool isIn) = indexOf(A, a);
        if (!isIn) {
            revert('Address not in array.');
        } else {
            (address[] memory _A, ) = pop(A, index);
            return _A;
        }
    }

    /**
     * Removes specified index from array
     * @param A The input array to search
     * @param index The index to remove
     * @return Returns the new array and the removed entry
     */
    function pop(address[] memory A, uint256 index) internal pure returns (address[] memory, address) {
        uint256 length = A.length;
        require(index < A.length, 'Index must be < A length');
        address[] memory newAddresses = new address[](length - 1);
        for (uint256 i = 0; i < index; i++) {
            newAddresses[i] = A[i];
        }
        for (uint256 j = index + 1; j < length; j++) {
            newAddresses[j - 1] = A[j];
        }
        return (newAddresses, A[index]);
    }

    /**
     * Returns the combination of the two arrays
     * @param A The first array
     * @param B The second array
     * @return Returns A extended by B
     */
    function extend(address[] memory A, address[] memory B) internal pure returns (address[] memory) {
        uint256 aLength = A.length;
        uint256 bLength = B.length;
        address[] memory newAddresses = new address[](aLength + bLength);
        for (uint256 i = 0; i < aLength; i++) {
            newAddresses[i] = A[i];
        }
        for (uint256 j = 0; j < bLength; j++) {
            newAddresses[aLength + j] = B[j];
        }
        return newAddresses;
    }
}

File 15 of 18 : Context.sol
// SPDX-License-Identifier: MIT

pragma solidity >=0.6.0 <0.8.0;

/*
 * @dev Provides information about the current execution context, including the
 * sender of the transaction and its data. While these are generally available
 * via msg.sender and msg.data, they should not be accessed in such a direct
 * manner, since when dealing with 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 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 16 of 18 : SafeMath.sol
// SPDX-License-Identifier: MIT

pragma solidity >=0.6.0 <0.8.0;

/**
 * @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, with an overflow flag.
     *
     * _Available since v3.4._
     */
    function tryAdd(uint256 a, uint256 b) internal pure returns (bool, uint256) {
        uint256 c = a + b;
        if (c < a) return (false, 0);
        return (true, c);
    }

    /**
     * @dev Returns the substraction of two unsigned integers, with an overflow flag.
     *
     * _Available since v3.4._
     */
    function trySub(uint256 a, uint256 b) internal pure returns (bool, uint256) {
        if (b > a) return (false, 0);
        return (true, a - b);
    }

    /**
     * @dev Returns the multiplication of two unsigned integers, with an overflow flag.
     *
     * _Available since v3.4._
     */
    function tryMul(uint256 a, uint256 b) internal pure returns (bool, 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 (true, 0);
        uint256 c = a * b;
        if (c / a != b) return (false, 0);
        return (true, c);
    }

    /**
     * @dev Returns the division of two unsigned integers, with a division by zero flag.
     *
     * _Available since v3.4._
     */
    function tryDiv(uint256 a, uint256 b) internal pure returns (bool, uint256) {
        if (b == 0) return (false, 0);
        return (true, a / b);
    }

    /**
     * @dev Returns the remainder of dividing two unsigned integers, with a division by zero flag.
     *
     * _Available since v3.4._
     */
    function tryMod(uint256 a, uint256 b) internal pure returns (bool, uint256) {
        if (b == 0) return (false, 0);
        return (true, a % b);
    }

    /**
     * @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) {
        require(b <= a, "SafeMath: subtraction overflow");
        return a - b;
    }

    /**
     * @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) {
        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, reverting 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) {
        require(b > 0, "SafeMath: division by zero");
        return a / b;
    }

    /**
     * @dev Returns the remainder of dividing two unsigned integers. (unsigned integer modulo),
     * reverting 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) {
        require(b > 0, "SafeMath: modulo by zero");
        return a % b;
    }

    /**
     * @dev Returns the subtraction of two unsigned integers, reverting with custom message on
     * overflow (when the result is negative).
     *
     * CAUTION: This function is deprecated because it requires allocating memory for the error
     * message unnecessarily. For custom revert reasons use {trySub}.
     *
     * Counterpart to Solidity's `-` operator.
     *
     * Requirements:
     *
     * - Subtraction cannot overflow.
     */
    function sub(uint256 a, uint256 b, string memory errorMessage) internal pure returns (uint256) {
        require(b <= a, errorMessage);
        return a - b;
    }

    /**
     * @dev Returns the integer division of two unsigned integers, reverting with custom message on
     * division by zero. The result is rounded towards zero.
     *
     * CAUTION: This function is deprecated because it requires allocating memory for the error
     * message unnecessarily. For custom revert reasons use {tryDiv}.
     *
     * 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, string memory errorMessage) internal pure returns (uint256) {
        require(b > 0, errorMessage);
        return a / b;
    }

    /**
     * @dev Returns the remainder of dividing two unsigned integers. (unsigned integer modulo),
     * reverting with custom message when dividing by zero.
     *
     * CAUTION: This function is deprecated because it requires allocating memory for the error
     * message unnecessarily. For custom revert reasons use {tryMod}.
     *
     * 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, string memory errorMessage) internal pure returns (uint256) {
        require(b > 0, errorMessage);
        return a % b;
    }
}

File 17 of 18 : ERC20.sol
// SPDX-License-Identifier: MIT

pragma solidity >=0.6.0 <0.8.0;

import "../../utils/Context.sol";
import "./IERC20.sol";
import "../../math/SafeMath.sol";

/**
 * @dev Implementation of the {IERC20} interface.
 *
 * This implementation is agnostic to the way tokens are created. This means
 * that a supply mechanism has to be added in a derived contract using {_mint}.
 * For a generic mechanism see {ERC20PresetMinterPauser}.
 *
 * TIP: For a detailed writeup see our guide
 * https://forum.zeppelin.solutions/t/how-to-implement-erc20-supply-mechanisms/226[How
 * to implement supply mechanisms].
 *
 * We have followed general OpenZeppelin guidelines: functions revert instead
 * of returning `false` on failure. This behavior is nonetheless conventional
 * and does not conflict with the expectations of ERC20 applications.
 *
 * Additionally, an {Approval} event is emitted on calls to {transferFrom}.
 * This allows applications to reconstruct the allowance for all accounts just
 * by listening to said events. Other implementations of the EIP may not emit
 * these events, as it isn't required by the specification.
 *
 * Finally, the non-standard {decreaseAllowance} and {increaseAllowance}
 * functions have been added to mitigate the well-known issues around setting
 * allowances. See {IERC20-approve}.
 */
contract ERC20 is Context, IERC20 {
    using SafeMath for uint256;

    mapping (address => uint256) private _balances;

    mapping (address => mapping (address => uint256)) private _allowances;

    uint256 private _totalSupply;

    string private _name;
    string private _symbol;
    uint8 private _decimals;

    /**
     * @dev Sets the values for {name} and {symbol}, initializes {decimals} with
     * a default value of 18.
     *
     * To select a different value for {decimals}, use {_setupDecimals}.
     *
     * All three of these values are immutable: they can only be set once during
     * construction.
     */
    constructor (string memory name_, string memory symbol_) public {
        _name = name_;
        _symbol = symbol_;
        _decimals = 18;
    }

    /**
     * @dev Returns the name of the token.
     */
    function name() public view virtual returns (string memory) {
        return _name;
    }

    /**
     * @dev Returns the symbol of the token, usually a shorter version of the
     * name.
     */
    function symbol() public view virtual returns (string memory) {
        return _symbol;
    }

    /**
     * @dev Returns the number of decimals used to get its user representation.
     * For example, if `decimals` equals `2`, a balance of `505` tokens should
     * be displayed to a user as `5,05` (`505 / 10 ** 2`).
     *
     * Tokens usually opt for a value of 18, imitating the relationship between
     * Ether and Wei. This is the value {ERC20} uses, unless {_setupDecimals} is
     * called.
     *
     * NOTE: This information is only used for _display_ purposes: it in
     * no way affects any of the arithmetic of the contract, including
     * {IERC20-balanceOf} and {IERC20-transfer}.
     */
    function decimals() public view virtual returns (uint8) {
        return _decimals;
    }

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

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

    /**
     * @dev See {IERC20-transfer}.
     *
     * Requirements:
     *
     * - `recipient` cannot be the zero address.
     * - the caller must have a balance of at least `amount`.
     */
    function transfer(address recipient, uint256 amount) public virtual override returns (bool) {
        _transfer(_msgSender(), recipient, amount);
        return true;
    }

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

    /**
     * @dev See {IERC20-approve}.
     *
     * Requirements:
     *
     * - `spender` cannot be the zero address.
     */
    function approve(address spender, uint256 amount) public virtual override returns (bool) {
        _approve(_msgSender(), spender, amount);
        return true;
    }

    /**
     * @dev See {IERC20-transferFrom}.
     *
     * Emits an {Approval} event indicating the updated allowance. This is not
     * required by the EIP. See the note at the beginning of {ERC20}.
     *
     * Requirements:
     *
     * - `sender` and `recipient` cannot be the zero address.
     * - `sender` must have a balance of at least `amount`.
     * - the caller must have allowance for ``sender``'s tokens of at least
     * `amount`.
     */
    function transferFrom(address sender, address recipient, uint256 amount) public virtual override returns (bool) {
        _transfer(sender, recipient, amount);
        _approve(sender, _msgSender(), _allowances[sender][_msgSender()].sub(amount, "ERC20: transfer amount exceeds allowance"));
        return true;
    }

    /**
     * @dev Atomically increases the allowance granted to `spender` by the caller.
     *
     * This is an alternative to {approve} that can be used as a mitigation for
     * problems described in {IERC20-approve}.
     *
     * Emits an {Approval} event indicating the updated allowance.
     *
     * Requirements:
     *
     * - `spender` cannot be the zero address.
     */
    function increaseAllowance(address spender, uint256 addedValue) public virtual returns (bool) {
        _approve(_msgSender(), spender, _allowances[_msgSender()][spender].add(addedValue));
        return true;
    }

    /**
     * @dev Atomically decreases the allowance granted to `spender` by the caller.
     *
     * This is an alternative to {approve} that can be used as a mitigation for
     * problems described in {IERC20-approve}.
     *
     * Emits an {Approval} event indicating the updated allowance.
     *
     * Requirements:
     *
     * - `spender` cannot be the zero address.
     * - `spender` must have allowance for the caller of at least
     * `subtractedValue`.
     */
    function decreaseAllowance(address spender, uint256 subtractedValue) public virtual returns (bool) {
        _approve(_msgSender(), spender, _allowances[_msgSender()][spender].sub(subtractedValue, "ERC20: decreased allowance below zero"));
        return true;
    }

    /**
     * @dev Moves tokens `amount` from `sender` to `recipient`.
     *
     * This is internal function is equivalent to {transfer}, and can be used to
     * e.g. implement automatic token fees, slashing mechanisms, etc.
     *
     * Emits a {Transfer} event.
     *
     * Requirements:
     *
     * - `sender` cannot be the zero address.
     * - `recipient` cannot be the zero address.
     * - `sender` must have a balance of at least `amount`.
     */
    function _transfer(address sender, address recipient, uint256 amount) internal virtual {
        require(sender != address(0), "ERC20: transfer from the zero address");
        require(recipient != address(0), "ERC20: transfer to the zero address");

        _beforeTokenTransfer(sender, recipient, amount);

        _balances[sender] = _balances[sender].sub(amount, "ERC20: transfer amount exceeds balance");
        _balances[recipient] = _balances[recipient].add(amount);
        emit Transfer(sender, recipient, amount);
    }

    /** @dev Creates `amount` tokens and assigns them to `account`, increasing
     * the total supply.
     *
     * Emits a {Transfer} event with `from` set to the zero address.
     *
     * Requirements:
     *
     * - `to` cannot be the zero address.
     */
    function _mint(address account, uint256 amount) internal virtual {
        require(account != address(0), "ERC20: mint to the zero address");

        _beforeTokenTransfer(address(0), account, amount);

        _totalSupply = _totalSupply.add(amount);
        _balances[account] = _balances[account].add(amount);
        emit Transfer(address(0), account, amount);
    }

    /**
     * @dev Destroys `amount` tokens from `account`, reducing the
     * total supply.
     *
     * Emits a {Transfer} event with `to` set to the zero address.
     *
     * Requirements:
     *
     * - `account` cannot be the zero address.
     * - `account` must have at least `amount` tokens.
     */
    function _burn(address account, uint256 amount) internal virtual {
        require(account != address(0), "ERC20: burn from the zero address");

        _beforeTokenTransfer(account, address(0), amount);

        _balances[account] = _balances[account].sub(amount, "ERC20: burn amount exceeds balance");
        _totalSupply = _totalSupply.sub(amount);
        emit Transfer(account, address(0), amount);
    }

    /**
     * @dev Sets `amount` as the allowance of `spender` over the `owner` s tokens.
     *
     * This internal function is equivalent to `approve`, and can be used to
     * e.g. set automatic allowances for certain subsystems, etc.
     *
     * Emits an {Approval} event.
     *
     * Requirements:
     *
     * - `owner` cannot be the zero address.
     * - `spender` cannot be the zero address.
     */
    function _approve(address owner, address spender, uint256 amount) internal virtual {
        require(owner != address(0), "ERC20: approve from the zero address");
        require(spender != address(0), "ERC20: approve to the zero address");

        _allowances[owner][spender] = amount;
        emit Approval(owner, spender, amount);
    }

    /**
     * @dev Sets {decimals} to a value other than the default one of 18.
     *
     * WARNING: This function should only be called from the constructor. Most
     * applications that interact with token contracts will not expect
     * {decimals} to ever change, and may work incorrectly if it does.
     */
    function _setupDecimals(uint8 decimals_) internal virtual {
        _decimals = decimals_;
    }

    /**
     * @dev Hook that is called before any transfer of tokens. This includes
     * minting and burning.
     *
     * Calling conditions:
     *
     * - when `from` and `to` are both non-zero, `amount` of ``from``'s tokens
     * will be to transferred to `to`.
     * - when `from` is zero, `amount` tokens will be minted for `to`.
     * - when `to` is zero, `amount` of ``from``'s tokens will be burned.
     * - `from` and `to` are never both zero.
     *
     * To learn more about hooks, head to xref:ROOT:extending-contracts.adoc#using-hooks[Using Hooks].
     */
    function _beforeTokenTransfer(address from, address to, uint256 amount) internal virtual { }
}

File 18 of 18 : IVoteToken.sol
/*
    Copyright 2021 Babylon Finance.

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at
    http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.

    SPDX-License-Identifier: Apache License, Version 2.0
*/

pragma solidity 0.7.6;

import {IERC20} from '@openzeppelin/contracts/token/ERC20/IERC20.sol';

interface IVoteToken {
    function delegate(address delegatee) external;

    function delegateBySig(
        address delegatee,
        uint256 nonce,
        uint256 expiry,
        uint8 v,
        bytes32 r,
        bytes32 s,
        bool prefix
    ) external;

    function getCurrentVotes(address account) external view returns (uint96);

    function getPriorVotes(address account, uint256 blockNumber) external view returns (uint96);

    function getMyDelegatee() external view returns (address);

    function getDelegatee(address account) external view returns (address);

    function getCheckpoints(address account, uint32 id) external view returns (uint32 fromBlock, uint96 votes);

    function getNumberOfCheckpoints(address account) external view returns (uint32);
}

interface IVoteTokenWithERC20 is IVoteToken, IERC20 {}

Settings
{
  "optimizer": {
    "enabled": true,
    "runs": 999
  },
  "outputSelection": {
    "*": {
      "*": [
        "evm.bytecode",
        "evm.deployedBytecode",
        "abi"
      ]
    }
  },
  "metadata": {
    "useLiteralContent": true
  },
  "libraries": {}
}

Contract Security Audit

Contract ABI

[{"inputs":[{"internalType":"contract IBabController","name":"newController","type":"address"}],"stateMutability":"nonpayable","type":"constructor"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"owner","type":"address"},{"indexed":true,"internalType":"address","name":"spender","type":"address"},{"indexed":false,"internalType":"uint256","name":"value","type":"uint256"}],"name":"Approval","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"account","type":"address"},{"indexed":false,"internalType":"uint256","name":"amount","type":"uint256"}],"name":"Cancel","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"_receiver","type":"address"},{"indexed":false,"internalType":"uint256","name":"amount","type":"uint256"}],"name":"Claim","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"delegator","type":"address"},{"indexed":true,"internalType":"address","name":"fromDelegate","type":"address"},{"indexed":true,"internalType":"address","name":"toDelegate","type":"address"}],"name":"DelegateChanged","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"delegate","type":"address"},{"indexed":false,"internalType":"uint256","name":"previousBalance","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"newBalance","type":"uint256"}],"name":"DelegateVotesChanged","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"_account","type":"address"},{"indexed":false,"internalType":"uint256","name":"amount","type":"uint256"}],"name":"LockedBalance","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"previousAllowedAfterValue","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"newAllowedAfterValue","type":"uint256"}],"name":"MaxSupplyAllowedAfterChanged","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"previousMaxValue","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"newMaxValue","type":"uint256"}],"name":"MaxSupplyChanged","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"account","type":"address"},{"indexed":false,"internalType":"uint256","name":"tokensminted","type":"uint256"}],"name":"MintedNewTokens","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"account","type":"address"},{"indexed":false,"internalType":"uint256","name":"tokenslocked","type":"uint256"},{"indexed":false,"internalType":"bool","name":"isTeamOrAdvisor","type":"bool"},{"indexed":false,"internalType":"uint256","name":"startingVesting","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"endingVesting","type":"uint256"}],"name":"NewLockout","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"previousAddress","type":"address"},{"indexed":false,"internalType":"address","name":"newAddress","type":"address"}],"name":"NewRewardsDistributorRegistration","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"previousAddress","type":"address"},{"indexed":false,"internalType":"address","name":"newAddress","type":"address"}],"name":"NewTimeLockRegistration","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"previousOwner","type":"address"},{"indexed":true,"internalType":"address","name":"newOwner","type":"address"}],"name":"OwnershipTransferred","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"from","type":"address"},{"indexed":true,"internalType":"address","name":"to","type":"address"},{"indexed":false,"internalType":"uint256","name":"value","type":"uint256"}],"name":"Transfer","type":"event"},{"inputs":[],"name":"BABLTokenDeploymentTimestamp","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"DELEGATION_TYPEHASH","outputs":[{"internalType":"bytes32","name":"","type":"bytes32"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"DOMAIN_TYPEHASH","outputs":[{"internalType":"bytes32","name":"","type":"bytes32"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"MAX_SUPPLY_CAP","outputs":[{"internalType":"uint8","name":"","type":"uint8"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"MINT_CAP","outputs":[{"internalType":"uint8","name":"","type":"uint8"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"owner","type":"address"},{"internalType":"address","name":"spender","type":"address"}],"name":"allowance","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"spender","type":"address"},{"internalType":"uint256","name":"rawAmount","type":"uint256"}],"name":"approve","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"account","type":"address"}],"name":"balanceOf","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"lockedAccount","type":"address"}],"name":"cancelVestedTokens","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"newMaxSupply","type":"uint256"},{"internalType":"uint256","name":"newMaxSupplyAllowedAfter","type":"uint256"}],"name":"changeMaxSupply","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"},{"internalType":"uint32","name":"","type":"uint32"}],"name":"checkpoints","outputs":[{"internalType":"uint32","name":"fromBlock","type":"uint32"},{"internalType":"uint96","name":"votes","type":"uint96"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"claimMyTokens","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"controller","outputs":[{"internalType":"contract IBabController","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"decimals","outputs":[{"internalType":"uint8","name":"","type":"uint8"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"spender","type":"address"},{"internalType":"uint256","name":"subtractedValue","type":"uint256"}],"name":"decreaseAllowance","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"delegatee","type":"address"}],"name":"delegate","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"delegatee","type":"address"},{"internalType":"uint256","name":"nonce","type":"uint256"},{"internalType":"uint256","name":"expiry","type":"uint256"},{"internalType":"uint8","name":"v","type":"uint8"},{"internalType":"bytes32","name":"r","type":"bytes32"},{"internalType":"bytes32","name":"s","type":"bytes32"},{"internalType":"bool","name":"prefix","type":"bool"}],"name":"delegateBySig","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"delegates","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"disableTokensTransfers","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"enableTokensTransfers","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"account","type":"address"},{"internalType":"uint32","name":"id","type":"uint32"}],"name":"getCheckpoints","outputs":[{"internalType":"uint32","name":"fromBlock","type":"uint32"},{"internalType":"uint96","name":"votes","type":"uint96"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"account","type":"address"}],"name":"getCurrentVotes","outputs":[{"internalType":"uint96","name":"","type":"uint96"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"account","type":"address"}],"name":"getDelegatee","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"getMyDelegatee","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"account","type":"address"}],"name":"getNumberOfCheckpoints","outputs":[{"internalType":"uint32","name":"","type":"uint32"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"account","type":"address"},{"internalType":"uint256","name":"blockNumber","type":"uint256"}],"name":"getPriorVotes","outputs":[{"internalType":"uint96","name":"","type":"uint96"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"getTimeLockRegistry","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"spender","type":"address"},{"internalType":"uint256","name":"addedValue","type":"uint256"}],"name":"increaseAllowance","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"account","type":"address"}],"name":"lockedBalance","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"maxSupply","outputs":[{"internalType":"uint96","name":"","type":"uint96"},{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"maxSupplyAllowed","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"maxSupplyAllowedAfter","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_to","type":"address"},{"internalType":"uint256","name":"_amount","type":"uint256"}],"name":"mint","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"mintingAllowedAfter","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"name","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"nonces","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"numCheckpoints","outputs":[{"internalType":"uint32","name":"","type":"uint32"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"owner","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_receiver","type":"address"},{"internalType":"uint256","name":"_amount","type":"uint256"},{"internalType":"bool","name":"_profile","type":"bool"},{"internalType":"uint256","name":"_vestingBegin","type":"uint256"},{"internalType":"uint256","name":"_vestingEnd","type":"uint256"},{"internalType":"uint256","name":"_lastClaim","type":"uint256"}],"name":"registerLockup","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"renounceOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"rewardsDistributor","outputs":[{"internalType":"contract IRewardsDistributor","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"contract IRewardsDistributor","name":"newRewardsDistributor","type":"address"}],"name":"setRewardsDistributor","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"contract TimeLockRegistry","name":"newTimeLockRegistry","type":"address"}],"name":"setTimeLockRegistry","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"symbol","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"timeLockRegistry","outputs":[{"internalType":"contract TimeLockRegistry","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"totalSupply","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"recipient","type":"address"},{"internalType":"uint256","name":"amount","type":"uint256"}],"name":"transfer","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"sender","type":"address"},{"internalType":"address","name":"recipient","type":"address"},{"internalType":"uint256","name":"amount","type":"uint256"}],"name":"transferFrom","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"newOwner","type":"address"}],"name":"transferOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"account","type":"address"}],"name":"unlockedBalance","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"vestedToken","outputs":[{"internalType":"bool","name":"teamOrAdvisor","type":"bool"},{"internalType":"uint256","name":"vestingBegin","type":"uint256"},{"internalType":"uint256","name":"vestingEnd","type":"uint256"},{"internalType":"uint256","name":"lastClaim","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"account","type":"address"}],"name":"viewLockedBalance","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"}]

608060405269d3c21bcecceda10000006010553480156200001f57600080fd5b506040516200503438038062005034833981810160405260208110156200004557600080fd5b5051604080518082018252600f81526e426162796c6f6e2e46696e616e636560881b6020828101918252835180850190945260048452631090509360e21b9084015281519192918391839183918391620000a29160039162000458565b508051620000b890600490602084019062000458565b50506005805460ff19166012179055506000620000d4620001d0565b60058054610100600160a81b0319166101006001600160a01b03841690810291909117909155604051919250906000907f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e0908290a3505060016006555050600f8054600160a01b60ff60a01b19909116179055504260138190556200016a90630f099c00620001d4602090811b62002e7917901c565b601155620001833369d3c21bcecceda1000000620001eb565b620001a7630f099c0063ffffffff1642620001d460201b62002e791790919060201c565b601255600d80546001600160a01b0319166001600160a01b039290921691909117905562000504565b3390565b80820182811015620001e557600080fd5b92915050565b6001600160a01b03821662000247576040805162461bcd60e51b815260206004820152601f60248201527f45524332303a206d696e7420746f20746865207a65726f206164647265737300604482015290519081900360640190fd5b6200025560008383620002fa565b62000271816002546200038e60201b62002e891790919060201c565b6002556001600160a01b03821660009081526020818152604090912054620002a491839062002e896200038e821b17901c565b6001600160a01b0383166000818152602081815260408083209490945583518581529351929391927fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef9281900390910190a35050565b620003128383836200038960201b62002ee31760201c565b620003896001600160a01b03841615806200033a5750600e546001600160a01b038581169116145b80620003535750600f546001600160a01b038581169116145b806200036c5750600e546001600160a01b038481169116145b80620003815750600f54600160a01b900460ff165b603e620003f0565b505050565b600082820183811015620003e9576040805162461bcd60e51b815260206004820152601b60248201527f536166654d6174683a206164646974696f6e206f766572666c6f770000000000604482015290519081900360640190fd5b9392505050565b816200040157620004018162000405565b5050565b62461bcd60e51b6000908152602060045260076024526642414223000030600a808404818106603090810160081b95839006959095019082900491820690940160101b939093010160c81b604452606490fd5b828054600181600116156101000203166002900490600052602060002090601f016020900481019282620004905760008555620004db565b82601f10620004ab57805160ff1916838001178555620004db565b82800160010185558215620004db579182015b82811115620004db578251825591602001919060010190620004be565b50620004e9929150620004ed565b5090565b5b80821115620004e95760008155600101620004ee565b614b2080620005146000396000f3fe608060405234801561001057600080fd5b506004361061034c5760003560e01c806382e012b1116101bd578063b23ddce7116100f9578063d5e62855116100a2578063e7a324dc1161007c578063e7a324dc14610a00578063f1127ed814610a08578063f2fde38b14610a3a578063f77c479114610a605761034c565b8063d5e62855146109c2578063d6a73a54146109ca578063dd62ed3e146109d25761034c565b8063c7de902d116100d3578063c7de902d1461094b578063d1f5c33b14610971578063d5abeb01146109975761034c565b8063b23ddce71461087c578063b4838cb4146108d6578063b4b5ea57146109255761034c565b8063962399e2116101665780639b4e735f116101405780639b4e735f146107f6578063a15d39d41461081c578063a457c2d714610824578063a9059cbb146108505761034c565b8063962399e2146107a257806398f1312e146107c85780639ae697bf146107d05761034c565b80638da5cb5b116101975780638da5cb5b1461078a578063937f2e331461079257806395d89b411461079a5761034c565b806382e012b1146106ee57806386cb9498146106f65780638cec2923146107445761034c565b806340c10f191161028c57806360caae5811610235578063715018a61161020f578063715018a61461067057806376ee403814610678578063782d6fe1146106805780637ecebe00146106c85761034c565b806360caae581461061c5780636fcfff451461062457806370a082311461064a5761034c565b8063587cde1e11610266578063587cde1e146105a85780635c19a95c146105ce5780635e0fac2e146105f65761034c565b806340c10f191461051757806342061268146105435780634f205087146105825761034c565b806323b872dd116102f9578063313ce567116102d3578063313ce567146104bd57806339509351146104db5780633dd5a83e146105075780633f2a55401461050f5761034c565b806323b872dd1461045c578063265e7bde1461049257806330b36cef146104b55761034c565b806318160ddd1161032a57806318160ddd1461042857806320606b7014610430578063238189a8146104385761034c565b806306fdde0314610351578063095ea7b3146103ce5780631523a3711461040e575b600080fd5b610359610a68565b6040805160208082528351818301528351919283929083019185019080838360005b8381101561039357818101518382015260200161037b565b50505050905090810190601f1680156103c05780820380516001836020036101000a031916815260200191505b509250505060405180910390f35b6103fa600480360360408110156103e457600080fd5b506001600160a01b038135169060200135610afe565b604080519115158252519081900360200190f35b610416610d0d565b60408051918252519081900360200190f35b610416610d13565b610416610d19565b610440610d3d565b604080516001600160a01b039092168252519081900360200190f35b6103fa6004803603606081101561047257600080fd5b506001600160a01b03813581169160208101359091169060400135610d4c565b6103fa600480360360408110156104a857600080fd5b5080359060200135610dd3565b6104166110c0565b6104c56110c6565b6040805160ff9092168252519081900360200190f35b6103fa600480360360408110156104f157600080fd5b506001600160a01b0381351690602001356110cf565b61044061124d565b61044061125c565b6103fa6004803603604081101561052d57600080fd5b506001600160a01b03813516906020013561126b565b6105696004803603602081101561055957600080fd5b50356001600160a01b03166115a6565b6040805163ffffffff9092168252519081900360200190f35b6104166004803603602081101561059857600080fd5b50356001600160a01b03166115cb565b610440600480360360208110156105be57600080fd5b50356001600160a01b0316611620565b6105f4600480360360208110156105e457600080fd5b50356001600160a01b031661163b565b005b6104166004803603602081101561060c57600080fd5b50356001600160a01b0316611648565b6104c5611665565b6105696004803603602081101561063a57600080fd5b50356001600160a01b031661166a565b6104166004803603602081101561066057600080fd5b50356001600160a01b0316611682565b6105f461169d565b61044061176e565b6106ac6004803603604081101561069657600080fd5b506001600160a01b03813516906020013561178a565b604080516001600160601b039092168252519081900360200190f35b610416600480360360208110156106de57600080fd5b50356001600160a01b03166119b5565b6105f46119c7565b61071c6004803603602081101561070c57600080fd5b50356001600160a01b0316611a6b565b6040805194151585526020850193909352838301919091526060830152519081900360800190f35b6103fa600480360360c081101561075a57600080fd5b506001600160a01b0381351690602081013590604081013515159060608101359060808101359060a00135611a96565b610440611d7c565b6105f4611d90565b610359611e97565b6103fa600480360360208110156107b857600080fd5b50356001600160a01b0316611ef8565b6104c5612162565b610416600480360360208110156107e657600080fd5b50356001600160a01b0316612167565b6104406004803603602081101561080c57600080fd5b50356001600160a01b0316612242565b610416612260565b6103fa6004803603604081101561083a57600080fd5b506001600160a01b038135169060200135612266565b6103fa6004803603604081101561086657600080fd5b506001600160a01b0381351690602001356123fa565b6108ae6004803603604081101561089257600080fd5b5080356001600160a01b0316906020013563ffffffff16612417565b6040805163ffffffff90931683526001600160601b0390911660208301528051918290030190f35b6105f4600480360360e08110156108ec57600080fd5b506001600160a01b038135169060208101359060408101359060ff6060820135169060808101359060a08101359060c00135151561245d565b6106ac6004803603602081101561093b57600080fd5b50356001600160a01b0316612803565b6104166004803603602081101561096157600080fd5b50356001600160a01b0316612875565b6103fa6004803603602081101561098757600080fd5b50356001600160a01b031661296d565b61099f612b79565b604080516001600160601b03909316835260208301919091528051918290030190f35b6105f4612baf565b610416612cb2565b610416600480360360408110156109e857600080fd5b506001600160a01b0381358116916020013516612cb8565b610416612ce3565b6108ae60048036036040811015610a1e57600080fd5b5080356001600160a01b0316906020013563ffffffff16612d07565b6105f460048036036020811015610a5057600080fd5b50356001600160a01b0316612d3d565b610440612e6a565b60038054604080516020601f6002600019610100600188161502019095169490940493840181900481028201810190925282815260609390929091830182828015610af45780601f10610ac957610100808354040283529160200191610af4565b820191906000526020600020905b815481529060010190602001808311610ad757829003601f168201915b5050505050905090565b600060026006541415610b58576040805162461bcd60e51b815260206004820152601f60248201527f5265656e7472616e637947756172643a207265656e7472616e742063616c6c00604482015290519081900360640190fd5b60026006556001600160a01b038316610ba25760405162461bcd60e51b815260040180806020018281038252603881526020018061425b6038913960400191505060405180910390fd5b6001600160a01b038316331415610bea5760405162461bcd60e51b815260040180806020018281038252603a81526020018061488a603a913960400191505060405180910390fd5b6000600019831415610bff5750600019610c24565b610c218360405180606001604052806030815260200161415660309139612ee8565b90505b600e546001600160a01b038581169116148015610c5f5750600e54610c539033906001600160a01b0316612cb8565b816001600160601b0316105b15610ca057600e54610c9d90610c7f9033906001600160a01b0316612cb8565b60405180608001604052806047815260200161492f60479139612ee8565b90505b610cb43385836001600160601b0316612f8b565b604080516001600160601b038316815290516001600160a01b0386169133917f8c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b9259181900360200190a36001915050600160065592915050565b60135481565b60025490565b7f8cad95687ba82c2ce50e74f7b754645e5117c3a5bec8151c0726d5857980a86681565b600e546001600160a01b031681565b6000610d59848484613077565b610dc984610d656132d6565b610dc48560405180606001604052806028815260200161441d602891396001600160a01b038a16600090815260016020526040812090610da36132d6565b6001600160a01b0316815260208101919091526040016000205491906132da565b612f8b565b5060019392505050565b6000610ddd6132d6565b6001600160a01b0316610dee611d7c565b6001600160a01b031614610e49576040805162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572604482015290519081900360640190fd5b601354610e5a90630f099c00612e79565b421015610e985760405162461bcd60e51b81526004018080602001828103825260638152602001806144a56063913960800191505060405180910390fd5b601154421015610ed95760405162461bcd60e51b81526004018080602001828103825260488152602001806146396048913960600191505060405180910390fd5b6010548311610f195760405162461bcd60e51b815260040180806020018281038252605081526020018061420b6050913960600191505060405180910390fd5b6000610f4b610f426064610f3c600560ff1660105461333490919063ffffffff16565b90613358565b60105490612e79565b905080841115610f8c5760405162461bcd60e51b81526004018080602001828103825260368152602001806143486036913960400191505060405180910390fd5b601054604080519182526020820186905280517fa626f93317e2474b716846a09cf0e28aacb96052643b252fe97704208fecf9b59281900390910190a1610feb846040518060600160405280604081526020016148c460409139612ee8565b6001600160601b03166010556000611007426301e13380612e79565b9050808410156110485760405162461bcd60e51b81526004018080602001828103825260608152602001806144456060913960600191505060405180910390fd5b601154604080519182526020820186905280517f63abb52e09389994840613a4780aa50f65baa62888e3cdcfcb088ac2de0b7db59281900390910190a16110a7846040518060800160405280604881526020016141c360489139612ee8565b6001600160601b03166011555060019150505b92915050565b60125481565b60055460ff1690565b600060026006541415611129576040805162461bcd60e51b815260206004820152601f60248201527f5265656e7472616e637947756172643a207265656e7472616e742063616c6c00604482015290519081900360640190fd5b60026006556111428261113c3386612cb8565b90612e79565b61114b33611648565b1015806111655750600e546001600160a01b038481169116145b6111a05760405162461bcd60e51b815260040180806020018281038252603d8152602001806149c6603d913960400191505060405180910390fd5b6001600160a01b0383166111e55760405162461bcd60e51b81526004018080602001828103825260418152602001806143ac6041913960600191505060405180910390fd5b6001600160a01b03831633141561122d5760405162461bcd60e51b8152600401808060200182810382526043815260200180614aa86043913960600191505060405180910390fd5b6112403384610dc48561113c3389612cb8565b5060018060065592915050565b600e546001600160a01b031690565b600f546001600160a01b031681565b60006112756132d6565b6001600160a01b0316611286611d7c565b6001600160a01b0316146112e1576040805162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572604482015290519081900360640190fd5b6010546112f08361113c610d13565b111561132d5760405162461bcd60e51b81526004018080602001828103825260248152602001806145946024913960400191505060405180910390fd5b60135461133e90630f099c00612e79565b42101561137c5760405162461bcd60e51b8152600401808060200182810382526055815260200180614a036055913960600191505060405180910390fd5b600082116113bb5760405162461bcd60e51b81526004018080602001828103825260308152602001806143ed6030913960400191505060405180910390fd5b6012544210156113fc5760405162461bcd60e51b81526004018080602001828103825260448152602001806142c46044913960600191505060405180910390fd5b6001600160a01b0383166114415760405162461bcd60e51b81526004018080602001828103825260348152602001806145606034913960400191505060405180910390fd5b6001600160a01b0383163014156114895760405162461bcd60e51b815260040180806020018281038252603c815260200180614079603c913960400191505060405180910390fd5b611497426301e13380612e79565b60128190555060006114c1836040518060600160405280602781526020016147b260279139612ee8565b90506114db6064610f3c60026114d5610d13565b90613334565b816001600160601b031611156115225760405162461bcd60e51b81526004018080602001828103825260388152602001806147006038913960400191505060405180910390fd5b61153584826001600160601b03166133bf565b604080516001600160a01b03861681526001600160601b038316602082015281517f51c018f451551bf4e92356d59bc5f1fcd317ef0f884d97807cae6dd395da7fe0929181900390910190a16001600160a01b03808516600090815260076020526040812054610dc99216836134af565b6001600160a01b03811660009081526009602052604090205463ffffffff165b919050565b600e546000906001600160a01b031633146116175760405162461bcd60e51b8152600401808060200182810382526050815260200180613fab6050913960600191505060405180910390fd5b6110ba8261364b565b6007602052600090815260409020546001600160a01b031681565b6116453382613868565b50565b60006110ba61165683612167565b61165f84611682565b9061392b565b600581565b60096020526000908152604090205463ffffffff1681565b6001600160a01b031660009081526020819052604090205490565b6116a56132d6565b6001600160a01b03166116b6611d7c565b6001600160a01b031614611711576040805162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572604482015290519081900360640190fd5b60055460405160009161010090046001600160a01b0316907f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e0908390a36005805474ffffffffffffffffffffffffffffffffffffffff0019169055565b336000908152600760205260409020546001600160a01b031690565b60004382106117ca5760405162461bcd60e51b815260040180806020018281038252602c8152602001806147d9602c913960400191505060405180910390fd5b6001600160a01b03831660009081526009602052604090205463ffffffff16806117f85760009150506110ba565b6001600160a01b038416600090815260086020908152604080832063ffffffff600019860181168552925290912054168310611875576001600160a01b03841660009081526008602090815260408083206000199490940163ffffffff168352929052205464010000000090046001600160601b031690506110ba565b6001600160a01b038416600090815260086020908152604080832083805290915290205463ffffffff168310156118b05760009150506110ba565b600060001982015b8163ffffffff168163ffffffff16111561196f576000600263ffffffff848403166001600160a01b038916600090815260086020908152604080832094909304860363ffffffff8181168452948252918390208351808501909452549384168084526401000000009094046001600160601b03169083015292509087141561194a576020015194506110ba9350505050565b805163ffffffff1687111561196157819350611968565b6001820392505b50506118b8565b506001600160a01b038516600090815260086020908152604080832063ffffffff909416835292905220546001600160601b036401000000009091041691505092915050565b600a6020526000908152604090205481565b6119cf6132d6565b6001600160a01b03166119e0611d7c565b6001600160a01b031614611a3b576040805162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572604482015290519081900360640190fd5b600f80547fffffffffffffffffffffff00ffffffffffffffffffffffffffffffffffffffff16600160a01b179055565b600c60205260009081526040902080546001820154600283015460039093015460ff90921692909184565b600e546000906001600160a01b03163314611ae25760405162461bcd60e51b8152600401808060200182810382526050815260200180613fab6050913960600191505060405180910390fd5b85611aec33611682565b1015611b3f576040805162461bcd60e51b815260206004820152601460248201527f696e73756666696369656e742062616c616e6365000000000000000000000000604482015290519081900360640190fd5b6001600160a01b038716611b9a576040805162461bcd60e51b815260206004820152601660248201527f63616e6e6f74206265207a65726f206164647265737300000000000000000000604482015290519081900360640190fd5b6001600160a01b038716301415611bf8576040805162461bcd60e51b815260206004820152601760248201527f63616e6e6f74206265207468697320636f6e7472616374000000000000000000604482015290519081900360640190fd5b600e546001600160a01b0388811691161415611c455760405162461bcd60e51b815260040180806020018281038252602e81526020018061437e602e913960400191505060405180910390fd5b6001600160a01b038716331415611ca3576040805162461bcd60e51b815260206004820152601e60248201527f746865206f776e65722063616e6e6f74206c6f636b757020697473656c660000604482015290519081900360640190fd5b6001600160a01b0387166000908152600b6020526040902054611cc69087612e79565b6001600160a01b0388166000908152600b6020908152604080832093909355600c905220805460ff1916861515178155600181018590556002810184905560038101839055611d16338989613077565b604080516001600160a01b038a1681526020810189905287151581830152606081018790526080810186905290517f7ec8d498ce0e4d11d52cf0588976b9a8b30679d7135f9e82de77e2e318a1ea409181900360a00190a1506001979650505050505050565b60055461010090046001600160a01b031690565b600e54604080517f1e83409a00000000000000000000000000000000000000000000000000000000815233600482015290516000926001600160a01b031691631e83409a91602480830192602092919082900301818787803b158015611df557600080fd5b505af1158015611e09573d6000803e3d6000fd5b505050506040513d6020811015611e1f57600080fd5b5051336000908152600c602052604090205490915060ff16151560011415611e5957600e54611e57906001600160a01b031682610afe565b505b604080513381526020810183905281517f47cee97cb7acd717b3c0aa1435d004cd5b3c8c57d70dbceb4e4458bbd60e39d4929181900390910190a150565b60048054604080516020601f6002600019610100600188161502019095169490940493840181900481028201810190925282815260609390929091830182828015610af45780601f10610ac957610100808354040283529160200191610af4565b600e546000906001600160a01b031615611fcc57600e60009054906101000a90046001600160a01b03166001600160a01b0316638da5cb5b6040518163ffffffff1660e01b815260040160206040518083038186803b158015611f5a57600080fd5b505afa158015611f6e573d6000803e3d6000fd5b505050506040513d6020811015611f8457600080fd5b50516001600160a01b03163314611fcc5760405162461bcd60e51b815260040180806020018281038252605a815260200180613f24605a913960600191505060405180910390fd5b6001600160a01b038216612027576040805162461bcd60e51b815260206004820152601660248201527f63616e6e6f74206265207a65726f206164647265737300000000000000000000604482015290519081900360640190fd5b6001600160a01b038216301415612085576040805162461bcd60e51b815260206004820152601760248201527f63616e6e6f74206265207468697320636f6e7472616374000000000000000000604482015290519081900360640190fd5b600e546001600160a01b03838116911614156120e8576040805162461bcd60e51b815260206004820152601c60248201527f6d757374206265206e65772054696d654c6f636b526567697374727900000000604482015290519081900360640190fd5b600e54604080516001600160a01b039283168152918416602083015280517f1b4fa7e35ece4dcbbb914f91d8c3539c828d43c0b8a700237373966a27a9693a9281900390910190a150600e80546001600160a01b03831673ffffffffffffffffffffffffffffffffffffffff199091161790556001919050565b600281565b60008061217383612875565b6001600160a01b0384166000908152600c602052604090206002015490915042108015906121a95750336001600160a01b038416145b80156121b3575080155b80156121d957506001600160a01b0383166000908152600c602052604090206002015415155b156121f8576001600160a01b0383166000908152600b60205260408120555b604080516001600160a01b03851681526020810183905281517febfb816221eaf798e256b970818c54755242a19e3e1a2b5b2060f9d22ecbf46d929181900390910190a192915050565b6001600160a01b039081166000908152600760205260409020541690565b60105481565b6000600260065414156122c0576040805162461bcd60e51b815260206004820152601f60248201527f5265656e7472616e637947756172643a207265656e7472616e742063616c6c00604482015290519081900360640190fd5b60026006556001600160a01b03831661230a5760405162461bcd60e51b81526004018080602001828103825260418152602001806145b86041913960600191505060405180910390fd5b6001600160a01b0383163314156123525760405162461bcd60e51b81526004018080602001828103825260438152602001806148476043913960600191505060405180910390fd5b8161235d3385612cb8565b101561239a5760405162461bcd60e51b81526004018080602001828103825260368152602001806140436036913960400191505060405180910390fd5b600e546001600160a01b03848116911614156123e75760405162461bcd60e51b81526004018080602001828103825260508152602001806141066050913960600191505060405180910390fd5b6112403384610dc48561165f3389612cb8565b600061240e6124076132d6565b8484613077565b50600192915050565b6001600160a01b0391909116600090815260086020908152604080832063ffffffff9485168452909152902054908116916401000000009091046001600160601b031690565b6000807f8cad95687ba82c2ce50e74f7b754645e5117c3a5bec8151c0726d5857980a866612489610a68565b8051906020012061249861393b565b60408051602080820195909552808201939093526060830191909152306080808401919091528151808403909101815260a0830182528051908401207fe48329057bfd03d55e49b547132e39cffd9c1820ad7b9d4c5307691425d15adf60c08401526001600160a01b038d1660e084015261010083018c90526101208084018c905282518085039091018152610140840183528051908501207f19010000000000000000000000000000000000000000000000000000000000006101608501526101628401829052610182808501829052835180860390910181526101a29094019092528251929093019190912091925090841561264557604080517f19457468657265756d205369676e6564204d6573736167653a0a333200000000602080830191909152603c80830185905283518084039091018152605c83018085528151918301919091206000909152607c830180855281905260ff8c16609c84015260bc83018b905260dc83018a9052925160019260fc8082019392601f1981019281900390910190855afa158015612633573d6000803e3d6000fd5b505050602060405103519450506126ac565b60018189898960405160008152602001604052604051808581526020018460ff1681526020018381526020018281526020019450505050506020604051602081039080840390855afa15801561269f573d6000803e3d6000fd5b5050506020604051035193505b60006126b785611682565b116126f35760405162461bcd60e51b815260040180806020018281038252602b8152602001806140b5602b913960400191505060405180910390fd5b6001600160a01b0384166127385760405162461bcd60e51b815260040180806020018281038252602b815260200180614904602b913960400191505060405180910390fd5b6001600160a01b0384166000908152600a60205260409020548a1461278e5760405162461bcd60e51b8152600401808060200182810382526027815260200180613ecd6027913960400191505060405180910390fd5b6001600160a01b0384166000908152600a6020526040902080546001019055428910156127ec5760405162461bcd60e51b815260040180806020018281038252602b815260200180614a7d602b913960400191505060405180910390fd5b6127f6848c613868565b5050505050505050505050565b6001600160a01b03811660009081526009602052604081205463ffffffff168061282e57600061286e565b6001600160a01b0383166000908152600860209081526040808320600019850163ffffffff16845290915290205464010000000090046001600160601b03165b9392505050565b6001600160a01b0381166000908152600b6020908152604080832054600c909252822060010154819042906128ae906301e13380612e79565b1180156128ba57508115155b156128c85791506115c69050565b6001600160a01b0384166000908152600c6020526040902060020154421015806128f0575081155b156128fd5750600061286e565b811561286e576001600160a01b0384166000908152600c60205260409020600181015460029091015461296591612934919061392b565b6001600160a01b0386166000908152600c6020526040902060020154610f3c9061295e904261392b565b8590613334565b949350505050565b60006129776132d6565b6001600160a01b0316612988611d7c565b6001600160a01b0316146129e3576040805162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572604482015290519081900360640190fd5b6001600160a01b038216612a3e576040805162461bcd60e51b815260206004820152601660248201527f63616e6e6f74206265207a65726f206164647265737300000000000000000000604482015290519081900360640190fd5b6001600160a01b038216301415612a9c576040805162461bcd60e51b815260206004820152601760248201527f63616e6e6f74206265207468697320636f6e7472616374000000000000000000604482015290519081900360640190fd5b600f546001600160a01b0383811691161415612aff576040805162461bcd60e51b815260206004820152601f60248201527f6d757374206265206e65772052657761726473204469737472696275746f7200604482015290519081900360640190fd5b600f54604080516001600160a01b039283168152918416602083015280517fd8732c543edde4e60b54e1136459ee3a610f1243749765d9dbdbcf9fe76a7bd59281900390910190a150600f80546001600160a01b03831673ffffffffffffffffffffffffffffffffffffffff199091161790556001919050565b6000806000612ba26010546040518060600160405280603d8152602001613e6d603d9139612ee8565b6011549093509150509091565b612bb76132d6565b6001600160a01b0316612bc8611d7c565b6001600160a01b031614612c23576040805162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572604482015290519081900360640190fd5b600f54600160a81b900460ff1615612c82576040805162461bcd60e51b815260206004820152600e60248201527f4241424c206d75737420666c6f77000000000000000000000000000000000000604482015290519081900360640190fd5b600f80547fffffffffffffffffffff0000ffffffffffffffffffffffffffffffffffffffff16600160a81b179055565b60115481565b6001600160a01b03918216600090815260016020908152604080832093909416825291909152205490565b7fe48329057bfd03d55e49b547132e39cffd9c1820ad7b9d4c5307691425d15adf81565b600860209081526000928352604080842090915290825290205463ffffffff81169064010000000090046001600160601b031682565b612d456132d6565b6001600160a01b0316612d56611d7c565b6001600160a01b031614612db1576040805162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572604482015290519081900360640190fd5b6001600160a01b038116612df65760405162461bcd60e51b8152600401808060200182810382526026815260200180613ffb6026913960400191505060405180910390fd5b6005546040516001600160a01b0380841692610100900416907f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e090600090a3600580546001600160a01b039092166101000274ffffffffffffffffffffffffffffffffffffffff0019909216919091179055565b600d546001600160a01b031681565b808201828110156110ba57600080fd5b60008282018381101561286e576040805162461bcd60e51b815260206004820152601b60248201527f536166654d6174683a206164646974696f6e206f766572666c6f770000000000604482015290519081900360640190fd5b505050565b6000816c010000000000000000000000008410612f835760405162461bcd60e51b81526004018080602001828103825283818151815260200191508051906020019080838360005b83811015612f48578181015183820152602001612f30565b50505050905090810190601f168015612f755780820380516001836020036101000a031916815260200191505b509250505060405180910390fd5b509192915050565b6001600160a01b038316612fd05760405162461bcd60e51b815260040180806020018281038252602481526020018061478e6024913960400191505060405180910390fd5b6001600160a01b0382166130155760405162461bcd60e51b81526004018080602001828103825260228152602001806140216022913960400191505060405180910390fd5b6001600160a01b03808416600081815260016020908152604080832094871680845294825291829020859055815185815291517f8c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b9259281900390910190a3505050565b600d54604080517f5b14f1830000000000000000000000000000000000000000000000000000000081523060048201529051613110926001600160a01b031691635b14f183916024808301926020929190829003018186803b1580156130dc57600080fd5b505afa1580156130f0573d6000803e3d6000fd5b505050506040513d602081101561310657600080fd5b505115605361393f565b6001600160a01b0383166131555760405162461bcd60e51b81526004018080602001828103825260428152602001806148056042913960600191505060405180910390fd5b6001600160a01b03821661319a5760405162461bcd60e51b81526004018080602001828103825260408152602001806143086040913960400191505060405180910390fd5b6001600160a01b0382163014156131e25760405162461bcd60e51b81526004018080602001828103825260508152602001806149766050913960600191505060405180910390fd5b806131ec84611682565b10156132295760405162461bcd60e51b815260040180806020018281038252603181526020018061475d6031913960400191505060405180910390fd5b8061323384611648565b10156132705760405162461bcd60e51b81526004018080602001828103825260408152602001806145f96040913960400191505060405180910390fd5b61327b838383613951565b6001600160a01b0380841660009081526007602090815260408083205486851684529281902054815160608101909252602c808352612ee3959485169491909116926132d1928792909161450890830139612ee8565b6134af565b3390565b6000818484111561332c5760405162461bcd60e51b8152602060048201818152835160248401528351909283926044909101919085019080838360008315612f48578181015183820152602001612f30565b505050900390565b600082158061334f5750508181028183828161334c57fe5b04145b6110ba57600080fd5b60008082116133ae576040805162461bcd60e51b815260206004820152601a60248201527f536166654d6174683a206469766973696f6e206279207a65726f000000000000604482015290519081900360640190fd5b8183816133b757fe5b049392505050565b6001600160a01b03821661341a576040805162461bcd60e51b815260206004820152601f60248201527f45524332303a206d696e7420746f20746865207a65726f206164647265737300604482015290519081900360640190fd5b61342660008383613aac565b6002546134339082612e89565b6002556001600160a01b0382166000908152602081905260409020546134599082612e89565b6001600160a01b0383166000818152602081815260408083209490945583518581529351929391927fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef9281900390910190a35050565b816001600160a01b0316836001600160a01b0316141580156134da57506000816001600160601b0316115b15612ee3576001600160a01b03831615613593576001600160a01b03831660009081526009602052604081205463ffffffff16908161351a57600061355a565b6001600160a01b0385166000908152600860209081526040808320600019860163ffffffff16845290915290205464010000000090046001600160601b03165b90506000613581828560405180606001604052806031815260200161429360319139613b28565b905061358f86848484613b8d565b5050505b6001600160a01b03821615612ee3576001600160a01b03821660009081526009602052604081205463ffffffff1690816135ce57600061360e565b6001600160a01b0384166000908152600860209081526040808320600019860163ffffffff16845290915290205464010000000090046001600160601b03165b905060006136358285604051806060016040528060308152602001613ef460309139613d4e565b905061364385848484613b8d565b505050505050565b600e546000906001600160a01b031633146136975760405162461bcd60e51b8152600401808060200182810382526050815260200180613fab6050913960600191505060405180910390fd5b6001600160a01b0382166000908152600b60205260409020546136eb5760405162461bcd60e51b815260040180806020018281038252602c815260200180614534602c913960400191505060405180910390fd5b60006136f683612167565b9050600081116137375760405162461bcd60e51b815260040180806020018281038252603d815260200180614186603d913960400191505060405180910390fd5b6001600160a01b0383166000908152600c602052604090205460ff1615156001146137935760405162461bcd60e51b81526004018080602001828103825260468152602001806146ba6046913960600191505060405180910390fd5b6001600160a01b038084166000908152600b60209081526040808320839055600c9091528120805460ff19168155600181018290556002810182905560030155600e546137e39185911683610d4c565b61381e5760405162461bcd60e51b815260040180806020018281038252602d815260200180613f7e602d913960400191505060405180910390fd5b604080516001600160a01b03851681526020810183905281517f27f83af92b39768b17fe0c8d6922452702717efb8626d97e7a754e0b27d4f6d2929181900390910190a192915050565b6001600160a01b03808316600090815260076020526040812054909116906138b061389285613db8565b604051806060016040528060258152602001614a5860259139612ee8565b6001600160a01b03858116600081815260076020526040808220805473ffffffffffffffffffffffffffffffffffffffff1916898616908117909155905194955093928616927f3134e8a2e6d97e929a7e54011ea5485d7d196dd5f0ba4d4ef95803e8e3fc257f9190a46139258284836134af565b50505050565b808203828111156110ba57600080fd5b4690565b8161394d5761394d81613dc3565b5050565b6001600160a01b0383166139965760405162461bcd60e51b81526004018080602001828103825260258152602001806147386025913960400191505060405180910390fd5b6001600160a01b0382166139db5760405162461bcd60e51b8152600401808060200182810382526023815260200180613eaa6023913960400191505060405180910390fd5b6139e6838383613aac565b613a23816040518060600160405280602681526020016140e0602691396001600160a01b03861660009081526020819052604090205491906132da565b6001600160a01b038085166000908152602081905260408082209390935590841681522054613a529082612e89565b6001600160a01b038084166000818152602081815260409182902094909455805185815290519193928716927fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef92918290030190a3505050565b613ab7838383612ee3565b612ee36001600160a01b0384161580613add5750600e546001600160a01b038581169116145b80613af55750600f546001600160a01b038581169116145b80613b0d5750600e546001600160a01b038481169116145b80613b215750600f54600160a01b900460ff165b603e61393f565b6000836001600160601b0316836001600160601b03161115829061332c5760405162461bcd60e51b8152602060048201818152835160248401528351909283926044909101919085019080838360008315612f48578181015183820152602001612f30565b6000613bb14360405180606001604052806039815260200161468160399139613e16565b905060008463ffffffff16118015613bfa57506001600160a01b038516600090815260086020908152604080832063ffffffff6000198901811685529252909120548282169116145b15613c5a576001600160a01b0385166000908152600860209081526040808320600019880163ffffffff168452909152902080546fffffffffffffffffffffffff0000000019166401000000006001600160601b03851602179055613cfa565b60408051808201825263ffffffff80841682526001600160601b0380861660208085019182526001600160a01b038b166000818152600883528781208c871682528352878120965187549451909516640100000000026fffffffffffffffffffffffff000000001995871663ffffffff19958616179590951694909417909555938252600990935292909220805460018801909316929091169190911790555b604080516001600160601b0380861682528416602082015281516001600160a01b038816927fdec2bacdd2f05b59de34da9b523dff8be42e5e38e818c82fdb0bae774387a724928290030190a25050505050565b6000838301826001600160601b038087169083161015613daf5760405162461bcd60e51b8152602060048201818152835160248401528351909283926044909101919085019080838360008315612f48578181015183820152602001612f30565b50949350505050565b60006110ba82611682565b62461bcd60e51b6000908152602060045260076024526642414223000030600a808404818106603090810160081b95839006959095019082900491820690940160101b939093010160c81b604452606490fd5b6000816401000000008410612f835760405162461bcd60e51b8152602060048201818152835160248401528351909283926044909101919085019080838360008315612f48578181015183820152602001612f3056fe4241424c546f6b656e3a3a6d6178537570706c79416c6c6f7765643a206d6178537570706c79416c6c6f7765642065786365656473203936206269747345524332303a207472616e7366657220746f20746865207a65726f2061646472657373566f7465546f6b656e3a3a64656c656761746542795369673a20696e76616c6964206e6f6e6365566f7465546f6b656e3a3a5f6d6f766544656c6567617465733a20766f746520616d6f756e74206f766572666c6f777354696d654c6f636b6564546f6b656e3a3a206f6e6c7954696d654c6f636b4f776e65723a2063616e206f6e6c7920626520657865637574656420627920746865206f776e6572206f662054696d654c6f636b526567697374727954696d654c6f636b6564546f6b656e3a3a63616e63656c546f6b656e733a5472616e73666572206661696c656454696d654c6f636b6564546f6b656e3a3a206f6e6c7954696d654c6f636b52656769737472793a2063616e206f6e6c792062652065786563757465642062792054696d654c6f636b52656769737472794f776e61626c653a206e6577206f776e657220697320746865207a65726f206164647265737345524332303a20617070726f766520746f20746865207a65726f206164647265737354696d654c6f636b6564546f6b656e3a3a6465637265617365416c6c6f77616e63653a556e646572666c6f7720636f6e646974696f6e4241424c546f6b656e3a3a6d696e743a2063616e6e6f74206d696e7420746f207468652061646472657373206f66207468697320636f6e7472616374566f7465546f6b656e3a3a64656c656761746542795369673a20696e76616c69642064656c656761746f7245524332303a207472616e7366657220616d6f756e7420657863656564732062616c616e636554696d654c6f636b6564546f6b656e3a3a6465637265617365416c6c6f77616e63653a63616e6e6f7420646563726561736520616c6c6f77616e636520746f2074696d654c6f636b526567697374727954696d654c6f636b6564546f6b656e3a3a617070726f76653a20616d6f756e742065786365656473203936206269747354696d654c6f636b6564546f6b656e3a3a63616e63656c546f6b656e733a546865726520617265206e6f206d6f7265206c6f636b656420746f6b656e734241424c546f6b656e3a3a6368616e67654d6178537570706c793a206e6577206e65774d6178537570706c79416c6c6f7765644166746572206578636565647320393620626974734241424c546f6b656e3a3a6368616e67654d6178537570706c793a206368616e67654d6178537570706c792073686f756c6420626520686967686572207468616e2070726576696f75732076616c756554696d654c6f636b6564546f6b656e3a3a617070726f76653a207370656e6465722063616e6e6f74206265207a65726f2061646472657373566f7465546f6b656e3a3a5f6d6f766544656c6567617465733a20766f746520616d6f756e7420756e646572666c6f77734241424c546f6b656e3a3a6d696e743a206d696e74696e67206e6f7420616c6c6f776564207965742062656361757365206d696e74696e67416c6c6f776564416674657254696d654c6f636b6564546f6b656e3a3a205f7472616e736665723a2063616e6e6f74207472616e7366657220746f20746865207a65726f20616464726573734241424c546f6b656e3a3a6368616e67654d6178537570706c793a206578636565646564206f6620616c6c6f7765642035252063617063616e6e6f74206265207468652054696d654c6f636b526567697374727920636f6e747261637420697473656c6654696d654c6f636b6564546f6b656e3a3a696e637265617365416c6c6f77616e63653a5370656e6465722063616e6e6f74206265207a65726f20616464726573734241424c546f6b656e3a3a6d696e743a206d696e742073686f756c6420626520686967686572207468616e207a65726f45524332303a207472616e7366657220616d6f756e74206578636565647320616c6c6f77616e63654241424c546f6b656e3a3a6368616e67654d6178537570706c793a20746865206e65774d6178537570706c79416c6c6f77656441667465722073686f756c64206265206174206c656173742031207965617220696e20746865206675747572654241424c546f6b656e3a3a6368616e67654d6178537570706c793a2061206368616e6765206f6e206d6178537570706c79416c6c6f776564206e6f7420616c6c6f77656420756e74696c2038207965617273206166746572206465706c6f796d656e7454696d654c6f636b6564546f6b656e3a3a205f7472616e736665723a2075696e743936206f766572666c6f7754696d654c6f636b6564546f6b656e3a3a63616e63656c546f6b656e733a4e6f7420726567697374657265644241424c546f6b656e3a3a6d696e743a2063616e6e6f74207472616e7366657220746f20746865207a65726f20616464726573734241424c546f6b656e3a3a6d696e743a206d617820737570706c7920657863656564656454696d654c6f636b6564546f6b656e3a3a6465637265617365416c6c6f77616e63653a5370656e6465722063616e6e6f74206265207a65726f206164647265737354696d654c6f636b6564546f6b656e3a3a205f7472616e736665723a20617474656d7074696e6720746f207472616e73666572206c6f636b65642066756e64734241424c546f6b656e3a3a6368616e67654d6178537570706c793a2061206368616e6765206f6e206d6178537570706c79416c6c6f776564206e6f7420616c6c6f77656420796574566f7465546f6b656e3a3a5f7772697465436865636b706f696e743a20626c6f636b206e756d6265722065786365656473203332206269747354696d654c6f636b6564546f6b656e3a3a63616e63656c546f6b656e733a63616e6e6f742063616e63656c206c6f636b656420746f6b656e7320746f20496e766573746f72734241424c546f6b656e3a3a6d696e743a206578636565646564206d696e7420636170206f66203225206f6620746f74616c20737570706c7945524332303a207472616e736665722066726f6d20746865207a65726f206164647265737354696d654c6f636b6564546f6b656e3a3a205f7472616e736665723a20696e73756666696369656e742062616c616e636545524332303a20617070726f76652066726f6d20746865207a65726f20616464726573734241424c546f6b656e3a3a6d696e743a20616d6f756e74206578636565647320393620626974734241424c546f6b656e3a3a6765745072696f72566f7465733a206e6f74207965742064657465726d696e656454696d654c6f636b6564546f6b656e3a3a205f7472616e736665723a2063616e6e6f74207472616e736665722066726f6d20746865207a65726f206164647265737354696d654c6f636b6564546f6b656e3a3a6465637265617365416c6c6f77616e63653a5370656e6465722063616e6e6f7420626520746865206d73672e73656e64657254696d654c6f636b6564546f6b656e3a3a617070726f76653a207370656e6465722063616e6e6f7420626520746865206d73672e73656e6465724241424c546f6b656e3a3a6368616e67654d6178537570706c793a20706f74656e7469616c206d617820616d6f756e7420657863656564732039362062697473566f7465546f6b656e3a3a64656c656761746542795369673a20696e76616c6964207369676e617475726554696d654c6f636b6564546f6b656e3a3a617070726f76653a2063616e6e6f7420646563726561736520616c6c6f77616e636520746f2074696d656c6f636b726567697374727954696d654c6f636b6564546f6b656e3a3a205f7472616e736665723a20646f206e6f74207472616e7366657220746f6b656e7320746f2074686520746f6b656e20636f6e747261637420697473656c6654696d654c6f636b6564546f6b656e3a3a696e637265617365416c6c6f77616e63653a4e6f7420656e6f75676820756e6c6f636b656420746f6b656e734241424c546f6b656e3a3a6d696e743a206d696e74696e67206e6f7420616c6c6f776564206166746572207468652046495253545f45504f43485f4d494e542068617320706173736564203e3d2038207965617273566f7465546f6b656e3a3a5f64656c65676174653a2075696e743936206f766572666c6f77566f7465546f6b656e3a3a64656c656761746542795369673a207369676e6174757265206578706972656454696d654c6f636b6564546f6b656e3a3a696e637265617365416c6c6f77616e63653a5370656e6465722063616e6e6f7420626520746865206d73672e73656e646572a26469706673582212207dbb8d9a484e41c9dbe9780f75d2898dcd8e889e8f43b61c88fa4cd51fb1d5f764736f6c63430007060033000000000000000000000000d4a5b5fcb561daf3adf86f8477555b92fba43b5f

Deployed Bytecode

0x608060405234801561001057600080fd5b506004361061034c5760003560e01c806382e012b1116101bd578063b23ddce7116100f9578063d5e62855116100a2578063e7a324dc1161007c578063e7a324dc14610a00578063f1127ed814610a08578063f2fde38b14610a3a578063f77c479114610a605761034c565b8063d5e62855146109c2578063d6a73a54146109ca578063dd62ed3e146109d25761034c565b8063c7de902d116100d3578063c7de902d1461094b578063d1f5c33b14610971578063d5abeb01146109975761034c565b8063b23ddce71461087c578063b4838cb4146108d6578063b4b5ea57146109255761034c565b8063962399e2116101665780639b4e735f116101405780639b4e735f146107f6578063a15d39d41461081c578063a457c2d714610824578063a9059cbb146108505761034c565b8063962399e2146107a257806398f1312e146107c85780639ae697bf146107d05761034c565b80638da5cb5b116101975780638da5cb5b1461078a578063937f2e331461079257806395d89b411461079a5761034c565b806382e012b1146106ee57806386cb9498146106f65780638cec2923146107445761034c565b806340c10f191161028c57806360caae5811610235578063715018a61161020f578063715018a61461067057806376ee403814610678578063782d6fe1146106805780637ecebe00146106c85761034c565b806360caae581461061c5780636fcfff451461062457806370a082311461064a5761034c565b8063587cde1e11610266578063587cde1e146105a85780635c19a95c146105ce5780635e0fac2e146105f65761034c565b806340c10f191461051757806342061268146105435780634f205087146105825761034c565b806323b872dd116102f9578063313ce567116102d3578063313ce567146104bd57806339509351146104db5780633dd5a83e146105075780633f2a55401461050f5761034c565b806323b872dd1461045c578063265e7bde1461049257806330b36cef146104b55761034c565b806318160ddd1161032a57806318160ddd1461042857806320606b7014610430578063238189a8146104385761034c565b806306fdde0314610351578063095ea7b3146103ce5780631523a3711461040e575b600080fd5b610359610a68565b6040805160208082528351818301528351919283929083019185019080838360005b8381101561039357818101518382015260200161037b565b50505050905090810190601f1680156103c05780820380516001836020036101000a031916815260200191505b509250505060405180910390f35b6103fa600480360360408110156103e457600080fd5b506001600160a01b038135169060200135610afe565b604080519115158252519081900360200190f35b610416610d0d565b60408051918252519081900360200190f35b610416610d13565b610416610d19565b610440610d3d565b604080516001600160a01b039092168252519081900360200190f35b6103fa6004803603606081101561047257600080fd5b506001600160a01b03813581169160208101359091169060400135610d4c565b6103fa600480360360408110156104a857600080fd5b5080359060200135610dd3565b6104166110c0565b6104c56110c6565b6040805160ff9092168252519081900360200190f35b6103fa600480360360408110156104f157600080fd5b506001600160a01b0381351690602001356110cf565b61044061124d565b61044061125c565b6103fa6004803603604081101561052d57600080fd5b506001600160a01b03813516906020013561126b565b6105696004803603602081101561055957600080fd5b50356001600160a01b03166115a6565b6040805163ffffffff9092168252519081900360200190f35b6104166004803603602081101561059857600080fd5b50356001600160a01b03166115cb565b610440600480360360208110156105be57600080fd5b50356001600160a01b0316611620565b6105f4600480360360208110156105e457600080fd5b50356001600160a01b031661163b565b005b6104166004803603602081101561060c57600080fd5b50356001600160a01b0316611648565b6104c5611665565b6105696004803603602081101561063a57600080fd5b50356001600160a01b031661166a565b6104166004803603602081101561066057600080fd5b50356001600160a01b0316611682565b6105f461169d565b61044061176e565b6106ac6004803603604081101561069657600080fd5b506001600160a01b03813516906020013561178a565b604080516001600160601b039092168252519081900360200190f35b610416600480360360208110156106de57600080fd5b50356001600160a01b03166119b5565b6105f46119c7565b61071c6004803603602081101561070c57600080fd5b50356001600160a01b0316611a6b565b6040805194151585526020850193909352838301919091526060830152519081900360800190f35b6103fa600480360360c081101561075a57600080fd5b506001600160a01b0381351690602081013590604081013515159060608101359060808101359060a00135611a96565b610440611d7c565b6105f4611d90565b610359611e97565b6103fa600480360360208110156107b857600080fd5b50356001600160a01b0316611ef8565b6104c5612162565b610416600480360360208110156107e657600080fd5b50356001600160a01b0316612167565b6104406004803603602081101561080c57600080fd5b50356001600160a01b0316612242565b610416612260565b6103fa6004803603604081101561083a57600080fd5b506001600160a01b038135169060200135612266565b6103fa6004803603604081101561086657600080fd5b506001600160a01b0381351690602001356123fa565b6108ae6004803603604081101561089257600080fd5b5080356001600160a01b0316906020013563ffffffff16612417565b6040805163ffffffff90931683526001600160601b0390911660208301528051918290030190f35b6105f4600480360360e08110156108ec57600080fd5b506001600160a01b038135169060208101359060408101359060ff6060820135169060808101359060a08101359060c00135151561245d565b6106ac6004803603602081101561093b57600080fd5b50356001600160a01b0316612803565b6104166004803603602081101561096157600080fd5b50356001600160a01b0316612875565b6103fa6004803603602081101561098757600080fd5b50356001600160a01b031661296d565b61099f612b79565b604080516001600160601b03909316835260208301919091528051918290030190f35b6105f4612baf565b610416612cb2565b610416600480360360408110156109e857600080fd5b506001600160a01b0381358116916020013516612cb8565b610416612ce3565b6108ae60048036036040811015610a1e57600080fd5b5080356001600160a01b0316906020013563ffffffff16612d07565b6105f460048036036020811015610a5057600080fd5b50356001600160a01b0316612d3d565b610440612e6a565b60038054604080516020601f6002600019610100600188161502019095169490940493840181900481028201810190925282815260609390929091830182828015610af45780601f10610ac957610100808354040283529160200191610af4565b820191906000526020600020905b815481529060010190602001808311610ad757829003601f168201915b5050505050905090565b600060026006541415610b58576040805162461bcd60e51b815260206004820152601f60248201527f5265656e7472616e637947756172643a207265656e7472616e742063616c6c00604482015290519081900360640190fd5b60026006556001600160a01b038316610ba25760405162461bcd60e51b815260040180806020018281038252603881526020018061425b6038913960400191505060405180910390fd5b6001600160a01b038316331415610bea5760405162461bcd60e51b815260040180806020018281038252603a81526020018061488a603a913960400191505060405180910390fd5b6000600019831415610bff5750600019610c24565b610c218360405180606001604052806030815260200161415660309139612ee8565b90505b600e546001600160a01b038581169116148015610c5f5750600e54610c539033906001600160a01b0316612cb8565b816001600160601b0316105b15610ca057600e54610c9d90610c7f9033906001600160a01b0316612cb8565b60405180608001604052806047815260200161492f60479139612ee8565b90505b610cb43385836001600160601b0316612f8b565b604080516001600160601b038316815290516001600160a01b0386169133917f8c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b9259181900360200190a36001915050600160065592915050565b60135481565b60025490565b7f8cad95687ba82c2ce50e74f7b754645e5117c3a5bec8151c0726d5857980a86681565b600e546001600160a01b031681565b6000610d59848484613077565b610dc984610d656132d6565b610dc48560405180606001604052806028815260200161441d602891396001600160a01b038a16600090815260016020526040812090610da36132d6565b6001600160a01b0316815260208101919091526040016000205491906132da565b612f8b565b5060019392505050565b6000610ddd6132d6565b6001600160a01b0316610dee611d7c565b6001600160a01b031614610e49576040805162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572604482015290519081900360640190fd5b601354610e5a90630f099c00612e79565b421015610e985760405162461bcd60e51b81526004018080602001828103825260638152602001806144a56063913960800191505060405180910390fd5b601154421015610ed95760405162461bcd60e51b81526004018080602001828103825260488152602001806146396048913960600191505060405180910390fd5b6010548311610f195760405162461bcd60e51b815260040180806020018281038252605081526020018061420b6050913960600191505060405180910390fd5b6000610f4b610f426064610f3c600560ff1660105461333490919063ffffffff16565b90613358565b60105490612e79565b905080841115610f8c5760405162461bcd60e51b81526004018080602001828103825260368152602001806143486036913960400191505060405180910390fd5b601054604080519182526020820186905280517fa626f93317e2474b716846a09cf0e28aacb96052643b252fe97704208fecf9b59281900390910190a1610feb846040518060600160405280604081526020016148c460409139612ee8565b6001600160601b03166010556000611007426301e13380612e79565b9050808410156110485760405162461bcd60e51b81526004018080602001828103825260608152602001806144456060913960600191505060405180910390fd5b601154604080519182526020820186905280517f63abb52e09389994840613a4780aa50f65baa62888e3cdcfcb088ac2de0b7db59281900390910190a16110a7846040518060800160405280604881526020016141c360489139612ee8565b6001600160601b03166011555060019150505b92915050565b60125481565b60055460ff1690565b600060026006541415611129576040805162461bcd60e51b815260206004820152601f60248201527f5265656e7472616e637947756172643a207265656e7472616e742063616c6c00604482015290519081900360640190fd5b60026006556111428261113c3386612cb8565b90612e79565b61114b33611648565b1015806111655750600e546001600160a01b038481169116145b6111a05760405162461bcd60e51b815260040180806020018281038252603d8152602001806149c6603d913960400191505060405180910390fd5b6001600160a01b0383166111e55760405162461bcd60e51b81526004018080602001828103825260418152602001806143ac6041913960600191505060405180910390fd5b6001600160a01b03831633141561122d5760405162461bcd60e51b8152600401808060200182810382526043815260200180614aa86043913960600191505060405180910390fd5b6112403384610dc48561113c3389612cb8565b5060018060065592915050565b600e546001600160a01b031690565b600f546001600160a01b031681565b60006112756132d6565b6001600160a01b0316611286611d7c565b6001600160a01b0316146112e1576040805162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572604482015290519081900360640190fd5b6010546112f08361113c610d13565b111561132d5760405162461bcd60e51b81526004018080602001828103825260248152602001806145946024913960400191505060405180910390fd5b60135461133e90630f099c00612e79565b42101561137c5760405162461bcd60e51b8152600401808060200182810382526055815260200180614a036055913960600191505060405180910390fd5b600082116113bb5760405162461bcd60e51b81526004018080602001828103825260308152602001806143ed6030913960400191505060405180910390fd5b6012544210156113fc5760405162461bcd60e51b81526004018080602001828103825260448152602001806142c46044913960600191505060405180910390fd5b6001600160a01b0383166114415760405162461bcd60e51b81526004018080602001828103825260348152602001806145606034913960400191505060405180910390fd5b6001600160a01b0383163014156114895760405162461bcd60e51b815260040180806020018281038252603c815260200180614079603c913960400191505060405180910390fd5b611497426301e13380612e79565b60128190555060006114c1836040518060600160405280602781526020016147b260279139612ee8565b90506114db6064610f3c60026114d5610d13565b90613334565b816001600160601b031611156115225760405162461bcd60e51b81526004018080602001828103825260388152602001806147006038913960400191505060405180910390fd5b61153584826001600160601b03166133bf565b604080516001600160a01b03861681526001600160601b038316602082015281517f51c018f451551bf4e92356d59bc5f1fcd317ef0f884d97807cae6dd395da7fe0929181900390910190a16001600160a01b03808516600090815260076020526040812054610dc99216836134af565b6001600160a01b03811660009081526009602052604090205463ffffffff165b919050565b600e546000906001600160a01b031633146116175760405162461bcd60e51b8152600401808060200182810382526050815260200180613fab6050913960600191505060405180910390fd5b6110ba8261364b565b6007602052600090815260409020546001600160a01b031681565b6116453382613868565b50565b60006110ba61165683612167565b61165f84611682565b9061392b565b600581565b60096020526000908152604090205463ffffffff1681565b6001600160a01b031660009081526020819052604090205490565b6116a56132d6565b6001600160a01b03166116b6611d7c565b6001600160a01b031614611711576040805162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572604482015290519081900360640190fd5b60055460405160009161010090046001600160a01b0316907f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e0908390a36005805474ffffffffffffffffffffffffffffffffffffffff0019169055565b336000908152600760205260409020546001600160a01b031690565b60004382106117ca5760405162461bcd60e51b815260040180806020018281038252602c8152602001806147d9602c913960400191505060405180910390fd5b6001600160a01b03831660009081526009602052604090205463ffffffff16806117f85760009150506110ba565b6001600160a01b038416600090815260086020908152604080832063ffffffff600019860181168552925290912054168310611875576001600160a01b03841660009081526008602090815260408083206000199490940163ffffffff168352929052205464010000000090046001600160601b031690506110ba565b6001600160a01b038416600090815260086020908152604080832083805290915290205463ffffffff168310156118b05760009150506110ba565b600060001982015b8163ffffffff168163ffffffff16111561196f576000600263ffffffff848403166001600160a01b038916600090815260086020908152604080832094909304860363ffffffff8181168452948252918390208351808501909452549384168084526401000000009094046001600160601b03169083015292509087141561194a576020015194506110ba9350505050565b805163ffffffff1687111561196157819350611968565b6001820392505b50506118b8565b506001600160a01b038516600090815260086020908152604080832063ffffffff909416835292905220546001600160601b036401000000009091041691505092915050565b600a6020526000908152604090205481565b6119cf6132d6565b6001600160a01b03166119e0611d7c565b6001600160a01b031614611a3b576040805162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572604482015290519081900360640190fd5b600f80547fffffffffffffffffffffff00ffffffffffffffffffffffffffffffffffffffff16600160a01b179055565b600c60205260009081526040902080546001820154600283015460039093015460ff90921692909184565b600e546000906001600160a01b03163314611ae25760405162461bcd60e51b8152600401808060200182810382526050815260200180613fab6050913960600191505060405180910390fd5b85611aec33611682565b1015611b3f576040805162461bcd60e51b815260206004820152601460248201527f696e73756666696369656e742062616c616e6365000000000000000000000000604482015290519081900360640190fd5b6001600160a01b038716611b9a576040805162461bcd60e51b815260206004820152601660248201527f63616e6e6f74206265207a65726f206164647265737300000000000000000000604482015290519081900360640190fd5b6001600160a01b038716301415611bf8576040805162461bcd60e51b815260206004820152601760248201527f63616e6e6f74206265207468697320636f6e7472616374000000000000000000604482015290519081900360640190fd5b600e546001600160a01b0388811691161415611c455760405162461bcd60e51b815260040180806020018281038252602e81526020018061437e602e913960400191505060405180910390fd5b6001600160a01b038716331415611ca3576040805162461bcd60e51b815260206004820152601e60248201527f746865206f776e65722063616e6e6f74206c6f636b757020697473656c660000604482015290519081900360640190fd5b6001600160a01b0387166000908152600b6020526040902054611cc69087612e79565b6001600160a01b0388166000908152600b6020908152604080832093909355600c905220805460ff1916861515178155600181018590556002810184905560038101839055611d16338989613077565b604080516001600160a01b038a1681526020810189905287151581830152606081018790526080810186905290517f7ec8d498ce0e4d11d52cf0588976b9a8b30679d7135f9e82de77e2e318a1ea409181900360a00190a1506001979650505050505050565b60055461010090046001600160a01b031690565b600e54604080517f1e83409a00000000000000000000000000000000000000000000000000000000815233600482015290516000926001600160a01b031691631e83409a91602480830192602092919082900301818787803b158015611df557600080fd5b505af1158015611e09573d6000803e3d6000fd5b505050506040513d6020811015611e1f57600080fd5b5051336000908152600c602052604090205490915060ff16151560011415611e5957600e54611e57906001600160a01b031682610afe565b505b604080513381526020810183905281517f47cee97cb7acd717b3c0aa1435d004cd5b3c8c57d70dbceb4e4458bbd60e39d4929181900390910190a150565b60048054604080516020601f6002600019610100600188161502019095169490940493840181900481028201810190925282815260609390929091830182828015610af45780601f10610ac957610100808354040283529160200191610af4565b600e546000906001600160a01b031615611fcc57600e60009054906101000a90046001600160a01b03166001600160a01b0316638da5cb5b6040518163ffffffff1660e01b815260040160206040518083038186803b158015611f5a57600080fd5b505afa158015611f6e573d6000803e3d6000fd5b505050506040513d6020811015611f8457600080fd5b50516001600160a01b03163314611fcc5760405162461bcd60e51b815260040180806020018281038252605a815260200180613f24605a913960600191505060405180910390fd5b6001600160a01b038216612027576040805162461bcd60e51b815260206004820152601660248201527f63616e6e6f74206265207a65726f206164647265737300000000000000000000604482015290519081900360640190fd5b6001600160a01b038216301415612085576040805162461bcd60e51b815260206004820152601760248201527f63616e6e6f74206265207468697320636f6e7472616374000000000000000000604482015290519081900360640190fd5b600e546001600160a01b03838116911614156120e8576040805162461bcd60e51b815260206004820152601c60248201527f6d757374206265206e65772054696d654c6f636b526567697374727900000000604482015290519081900360640190fd5b600e54604080516001600160a01b039283168152918416602083015280517f1b4fa7e35ece4dcbbb914f91d8c3539c828d43c0b8a700237373966a27a9693a9281900390910190a150600e80546001600160a01b03831673ffffffffffffffffffffffffffffffffffffffff199091161790556001919050565b600281565b60008061217383612875565b6001600160a01b0384166000908152600c602052604090206002015490915042108015906121a95750336001600160a01b038416145b80156121b3575080155b80156121d957506001600160a01b0383166000908152600c602052604090206002015415155b156121f8576001600160a01b0383166000908152600b60205260408120555b604080516001600160a01b03851681526020810183905281517febfb816221eaf798e256b970818c54755242a19e3e1a2b5b2060f9d22ecbf46d929181900390910190a192915050565b6001600160a01b039081166000908152600760205260409020541690565b60105481565b6000600260065414156122c0576040805162461bcd60e51b815260206004820152601f60248201527f5265656e7472616e637947756172643a207265656e7472616e742063616c6c00604482015290519081900360640190fd5b60026006556001600160a01b03831661230a5760405162461bcd60e51b81526004018080602001828103825260418152602001806145b86041913960600191505060405180910390fd5b6001600160a01b0383163314156123525760405162461bcd60e51b81526004018080602001828103825260438152602001806148476043913960600191505060405180910390fd5b8161235d3385612cb8565b101561239a5760405162461bcd60e51b81526004018080602001828103825260368152602001806140436036913960400191505060405180910390fd5b600e546001600160a01b03848116911614156123e75760405162461bcd60e51b81526004018080602001828103825260508152602001806141066050913960600191505060405180910390fd5b6112403384610dc48561165f3389612cb8565b600061240e6124076132d6565b8484613077565b50600192915050565b6001600160a01b0391909116600090815260086020908152604080832063ffffffff9485168452909152902054908116916401000000009091046001600160601b031690565b6000807f8cad95687ba82c2ce50e74f7b754645e5117c3a5bec8151c0726d5857980a866612489610a68565b8051906020012061249861393b565b60408051602080820195909552808201939093526060830191909152306080808401919091528151808403909101815260a0830182528051908401207fe48329057bfd03d55e49b547132e39cffd9c1820ad7b9d4c5307691425d15adf60c08401526001600160a01b038d1660e084015261010083018c90526101208084018c905282518085039091018152610140840183528051908501207f19010000000000000000000000000000000000000000000000000000000000006101608501526101628401829052610182808501829052835180860390910181526101a29094019092528251929093019190912091925090841561264557604080517f19457468657265756d205369676e6564204d6573736167653a0a333200000000602080830191909152603c80830185905283518084039091018152605c83018085528151918301919091206000909152607c830180855281905260ff8c16609c84015260bc83018b905260dc83018a9052925160019260fc8082019392601f1981019281900390910190855afa158015612633573d6000803e3d6000fd5b505050602060405103519450506126ac565b60018189898960405160008152602001604052604051808581526020018460ff1681526020018381526020018281526020019450505050506020604051602081039080840390855afa15801561269f573d6000803e3d6000fd5b5050506020604051035193505b60006126b785611682565b116126f35760405162461bcd60e51b815260040180806020018281038252602b8152602001806140b5602b913960400191505060405180910390fd5b6001600160a01b0384166127385760405162461bcd60e51b815260040180806020018281038252602b815260200180614904602b913960400191505060405180910390fd5b6001600160a01b0384166000908152600a60205260409020548a1461278e5760405162461bcd60e51b8152600401808060200182810382526027815260200180613ecd6027913960400191505060405180910390fd5b6001600160a01b0384166000908152600a6020526040902080546001019055428910156127ec5760405162461bcd60e51b815260040180806020018281038252602b815260200180614a7d602b913960400191505060405180910390fd5b6127f6848c613868565b5050505050505050505050565b6001600160a01b03811660009081526009602052604081205463ffffffff168061282e57600061286e565b6001600160a01b0383166000908152600860209081526040808320600019850163ffffffff16845290915290205464010000000090046001600160601b03165b9392505050565b6001600160a01b0381166000908152600b6020908152604080832054600c909252822060010154819042906128ae906301e13380612e79565b1180156128ba57508115155b156128c85791506115c69050565b6001600160a01b0384166000908152600c6020526040902060020154421015806128f0575081155b156128fd5750600061286e565b811561286e576001600160a01b0384166000908152600c60205260409020600181015460029091015461296591612934919061392b565b6001600160a01b0386166000908152600c6020526040902060020154610f3c9061295e904261392b565b8590613334565b949350505050565b60006129776132d6565b6001600160a01b0316612988611d7c565b6001600160a01b0316146129e3576040805162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572604482015290519081900360640190fd5b6001600160a01b038216612a3e576040805162461bcd60e51b815260206004820152601660248201527f63616e6e6f74206265207a65726f206164647265737300000000000000000000604482015290519081900360640190fd5b6001600160a01b038216301415612a9c576040805162461bcd60e51b815260206004820152601760248201527f63616e6e6f74206265207468697320636f6e7472616374000000000000000000604482015290519081900360640190fd5b600f546001600160a01b0383811691161415612aff576040805162461bcd60e51b815260206004820152601f60248201527f6d757374206265206e65772052657761726473204469737472696275746f7200604482015290519081900360640190fd5b600f54604080516001600160a01b039283168152918416602083015280517fd8732c543edde4e60b54e1136459ee3a610f1243749765d9dbdbcf9fe76a7bd59281900390910190a150600f80546001600160a01b03831673ffffffffffffffffffffffffffffffffffffffff199091161790556001919050565b6000806000612ba26010546040518060600160405280603d8152602001613e6d603d9139612ee8565b6011549093509150509091565b612bb76132d6565b6001600160a01b0316612bc8611d7c565b6001600160a01b031614612c23576040805162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572604482015290519081900360640190fd5b600f54600160a81b900460ff1615612c82576040805162461bcd60e51b815260206004820152600e60248201527f4241424c206d75737420666c6f77000000000000000000000000000000000000604482015290519081900360640190fd5b600f80547fffffffffffffffffffff0000ffffffffffffffffffffffffffffffffffffffff16600160a81b179055565b60115481565b6001600160a01b03918216600090815260016020908152604080832093909416825291909152205490565b7fe48329057bfd03d55e49b547132e39cffd9c1820ad7b9d4c5307691425d15adf81565b600860209081526000928352604080842090915290825290205463ffffffff81169064010000000090046001600160601b031682565b612d456132d6565b6001600160a01b0316612d56611d7c565b6001600160a01b031614612db1576040805162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572604482015290519081900360640190fd5b6001600160a01b038116612df65760405162461bcd60e51b8152600401808060200182810382526026815260200180613ffb6026913960400191505060405180910390fd5b6005546040516001600160a01b0380841692610100900416907f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e090600090a3600580546001600160a01b039092166101000274ffffffffffffffffffffffffffffffffffffffff0019909216919091179055565b600d546001600160a01b031681565b808201828110156110ba57600080fd5b60008282018381101561286e576040805162461bcd60e51b815260206004820152601b60248201527f536166654d6174683a206164646974696f6e206f766572666c6f770000000000604482015290519081900360640190fd5b505050565b6000816c010000000000000000000000008410612f835760405162461bcd60e51b81526004018080602001828103825283818151815260200191508051906020019080838360005b83811015612f48578181015183820152602001612f30565b50505050905090810190601f168015612f755780820380516001836020036101000a031916815260200191505b509250505060405180910390fd5b509192915050565b6001600160a01b038316612fd05760405162461bcd60e51b815260040180806020018281038252602481526020018061478e6024913960400191505060405180910390fd5b6001600160a01b0382166130155760405162461bcd60e51b81526004018080602001828103825260228152602001806140216022913960400191505060405180910390fd5b6001600160a01b03808416600081815260016020908152604080832094871680845294825291829020859055815185815291517f8c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b9259281900390910190a3505050565b600d54604080517f5b14f1830000000000000000000000000000000000000000000000000000000081523060048201529051613110926001600160a01b031691635b14f183916024808301926020929190829003018186803b1580156130dc57600080fd5b505afa1580156130f0573d6000803e3d6000fd5b505050506040513d602081101561310657600080fd5b505115605361393f565b6001600160a01b0383166131555760405162461bcd60e51b81526004018080602001828103825260428152602001806148056042913960600191505060405180910390fd5b6001600160a01b03821661319a5760405162461bcd60e51b81526004018080602001828103825260408152602001806143086040913960400191505060405180910390fd5b6001600160a01b0382163014156131e25760405162461bcd60e51b81526004018080602001828103825260508152602001806149766050913960600191505060405180910390fd5b806131ec84611682565b10156132295760405162461bcd60e51b815260040180806020018281038252603181526020018061475d6031913960400191505060405180910390fd5b8061323384611648565b10156132705760405162461bcd60e51b81526004018080602001828103825260408152602001806145f96040913960400191505060405180910390fd5b61327b838383613951565b6001600160a01b0380841660009081526007602090815260408083205486851684529281902054815160608101909252602c808352612ee3959485169491909116926132d1928792909161450890830139612ee8565b6134af565b3390565b6000818484111561332c5760405162461bcd60e51b8152602060048201818152835160248401528351909283926044909101919085019080838360008315612f48578181015183820152602001612f30565b505050900390565b600082158061334f5750508181028183828161334c57fe5b04145b6110ba57600080fd5b60008082116133ae576040805162461bcd60e51b815260206004820152601a60248201527f536166654d6174683a206469766973696f6e206279207a65726f000000000000604482015290519081900360640190fd5b8183816133b757fe5b049392505050565b6001600160a01b03821661341a576040805162461bcd60e51b815260206004820152601f60248201527f45524332303a206d696e7420746f20746865207a65726f206164647265737300604482015290519081900360640190fd5b61342660008383613aac565b6002546134339082612e89565b6002556001600160a01b0382166000908152602081905260409020546134599082612e89565b6001600160a01b0383166000818152602081815260408083209490945583518581529351929391927fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef9281900390910190a35050565b816001600160a01b0316836001600160a01b0316141580156134da57506000816001600160601b0316115b15612ee3576001600160a01b03831615613593576001600160a01b03831660009081526009602052604081205463ffffffff16908161351a57600061355a565b6001600160a01b0385166000908152600860209081526040808320600019860163ffffffff16845290915290205464010000000090046001600160601b03165b90506000613581828560405180606001604052806031815260200161429360319139613b28565b905061358f86848484613b8d565b5050505b6001600160a01b03821615612ee3576001600160a01b03821660009081526009602052604081205463ffffffff1690816135ce57600061360e565b6001600160a01b0384166000908152600860209081526040808320600019860163ffffffff16845290915290205464010000000090046001600160601b03165b905060006136358285604051806060016040528060308152602001613ef460309139613d4e565b905061364385848484613b8d565b505050505050565b600e546000906001600160a01b031633146136975760405162461bcd60e51b8152600401808060200182810382526050815260200180613fab6050913960600191505060405180910390fd5b6001600160a01b0382166000908152600b60205260409020546136eb5760405162461bcd60e51b815260040180806020018281038252602c815260200180614534602c913960400191505060405180910390fd5b60006136f683612167565b9050600081116137375760405162461bcd60e51b815260040180806020018281038252603d815260200180614186603d913960400191505060405180910390fd5b6001600160a01b0383166000908152600c602052604090205460ff1615156001146137935760405162461bcd60e51b81526004018080602001828103825260468152602001806146ba6046913960600191505060405180910390fd5b6001600160a01b038084166000908152600b60209081526040808320839055600c9091528120805460ff19168155600181018290556002810182905560030155600e546137e39185911683610d4c565b61381e5760405162461bcd60e51b815260040180806020018281038252602d815260200180613f7e602d913960400191505060405180910390fd5b604080516001600160a01b03851681526020810183905281517f27f83af92b39768b17fe0c8d6922452702717efb8626d97e7a754e0b27d4f6d2929181900390910190a192915050565b6001600160a01b03808316600090815260076020526040812054909116906138b061389285613db8565b604051806060016040528060258152602001614a5860259139612ee8565b6001600160a01b03858116600081815260076020526040808220805473ffffffffffffffffffffffffffffffffffffffff1916898616908117909155905194955093928616927f3134e8a2e6d97e929a7e54011ea5485d7d196dd5f0ba4d4ef95803e8e3fc257f9190a46139258284836134af565b50505050565b808203828111156110ba57600080fd5b4690565b8161394d5761394d81613dc3565b5050565b6001600160a01b0383166139965760405162461bcd60e51b81526004018080602001828103825260258152602001806147386025913960400191505060405180910390fd5b6001600160a01b0382166139db5760405162461bcd60e51b8152600401808060200182810382526023815260200180613eaa6023913960400191505060405180910390fd5b6139e6838383613aac565b613a23816040518060600160405280602681526020016140e0602691396001600160a01b03861660009081526020819052604090205491906132da565b6001600160a01b038085166000908152602081905260408082209390935590841681522054613a529082612e89565b6001600160a01b038084166000818152602081815260409182902094909455805185815290519193928716927fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef92918290030190a3505050565b613ab7838383612ee3565b612ee36001600160a01b0384161580613add5750600e546001600160a01b038581169116145b80613af55750600f546001600160a01b038581169116145b80613b0d5750600e546001600160a01b038481169116145b80613b215750600f54600160a01b900460ff165b603e61393f565b6000836001600160601b0316836001600160601b03161115829061332c5760405162461bcd60e51b8152602060048201818152835160248401528351909283926044909101919085019080838360008315612f48578181015183820152602001612f30565b6000613bb14360405180606001604052806039815260200161468160399139613e16565b905060008463ffffffff16118015613bfa57506001600160a01b038516600090815260086020908152604080832063ffffffff6000198901811685529252909120548282169116145b15613c5a576001600160a01b0385166000908152600860209081526040808320600019880163ffffffff168452909152902080546fffffffffffffffffffffffff0000000019166401000000006001600160601b03851602179055613cfa565b60408051808201825263ffffffff80841682526001600160601b0380861660208085019182526001600160a01b038b166000818152600883528781208c871682528352878120965187549451909516640100000000026fffffffffffffffffffffffff000000001995871663ffffffff19958616179590951694909417909555938252600990935292909220805460018801909316929091169190911790555b604080516001600160601b0380861682528416602082015281516001600160a01b038816927fdec2bacdd2f05b59de34da9b523dff8be42e5e38e818c82fdb0bae774387a724928290030190a25050505050565b6000838301826001600160601b038087169083161015613daf5760405162461bcd60e51b8152602060048201818152835160248401528351909283926044909101919085019080838360008315612f48578181015183820152602001612f30565b50949350505050565b60006110ba82611682565b62461bcd60e51b6000908152602060045260076024526642414223000030600a808404818106603090810160081b95839006959095019082900491820690940160101b939093010160c81b604452606490fd5b6000816401000000008410612f835760405162461bcd60e51b8152602060048201818152835160248401528351909283926044909101919085019080838360008315612f48578181015183820152602001612f3056fe4241424c546f6b656e3a3a6d6178537570706c79416c6c6f7765643a206d6178537570706c79416c6c6f7765642065786365656473203936206269747345524332303a207472616e7366657220746f20746865207a65726f2061646472657373566f7465546f6b656e3a3a64656c656761746542795369673a20696e76616c6964206e6f6e6365566f7465546f6b656e3a3a5f6d6f766544656c6567617465733a20766f746520616d6f756e74206f766572666c6f777354696d654c6f636b6564546f6b656e3a3a206f6e6c7954696d654c6f636b4f776e65723a2063616e206f6e6c7920626520657865637574656420627920746865206f776e6572206f662054696d654c6f636b526567697374727954696d654c6f636b6564546f6b656e3a3a63616e63656c546f6b656e733a5472616e73666572206661696c656454696d654c6f636b6564546f6b656e3a3a206f6e6c7954696d654c6f636b52656769737472793a2063616e206f6e6c792062652065786563757465642062792054696d654c6f636b52656769737472794f776e61626c653a206e6577206f776e657220697320746865207a65726f206164647265737345524332303a20617070726f766520746f20746865207a65726f206164647265737354696d654c6f636b6564546f6b656e3a3a6465637265617365416c6c6f77616e63653a556e646572666c6f7720636f6e646974696f6e4241424c546f6b656e3a3a6d696e743a2063616e6e6f74206d696e7420746f207468652061646472657373206f66207468697320636f6e7472616374566f7465546f6b656e3a3a64656c656761746542795369673a20696e76616c69642064656c656761746f7245524332303a207472616e7366657220616d6f756e7420657863656564732062616c616e636554696d654c6f636b6564546f6b656e3a3a6465637265617365416c6c6f77616e63653a63616e6e6f7420646563726561736520616c6c6f77616e636520746f2074696d654c6f636b526567697374727954696d654c6f636b6564546f6b656e3a3a617070726f76653a20616d6f756e742065786365656473203936206269747354696d654c6f636b6564546f6b656e3a3a63616e63656c546f6b656e733a546865726520617265206e6f206d6f7265206c6f636b656420746f6b656e734241424c546f6b656e3a3a6368616e67654d6178537570706c793a206e6577206e65774d6178537570706c79416c6c6f7765644166746572206578636565647320393620626974734241424c546f6b656e3a3a6368616e67654d6178537570706c793a206368616e67654d6178537570706c792073686f756c6420626520686967686572207468616e2070726576696f75732076616c756554696d654c6f636b6564546f6b656e3a3a617070726f76653a207370656e6465722063616e6e6f74206265207a65726f2061646472657373566f7465546f6b656e3a3a5f6d6f766544656c6567617465733a20766f746520616d6f756e7420756e646572666c6f77734241424c546f6b656e3a3a6d696e743a206d696e74696e67206e6f7420616c6c6f776564207965742062656361757365206d696e74696e67416c6c6f776564416674657254696d654c6f636b6564546f6b656e3a3a205f7472616e736665723a2063616e6e6f74207472616e7366657220746f20746865207a65726f20616464726573734241424c546f6b656e3a3a6368616e67654d6178537570706c793a206578636565646564206f6620616c6c6f7765642035252063617063616e6e6f74206265207468652054696d654c6f636b526567697374727920636f6e747261637420697473656c6654696d654c6f636b6564546f6b656e3a3a696e637265617365416c6c6f77616e63653a5370656e6465722063616e6e6f74206265207a65726f20616464726573734241424c546f6b656e3a3a6d696e743a206d696e742073686f756c6420626520686967686572207468616e207a65726f45524332303a207472616e7366657220616d6f756e74206578636565647320616c6c6f77616e63654241424c546f6b656e3a3a6368616e67654d6178537570706c793a20746865206e65774d6178537570706c79416c6c6f77656441667465722073686f756c64206265206174206c656173742031207965617220696e20746865206675747572654241424c546f6b656e3a3a6368616e67654d6178537570706c793a2061206368616e6765206f6e206d6178537570706c79416c6c6f776564206e6f7420616c6c6f77656420756e74696c2038207965617273206166746572206465706c6f796d656e7454696d654c6f636b6564546f6b656e3a3a205f7472616e736665723a2075696e743936206f766572666c6f7754696d654c6f636b6564546f6b656e3a3a63616e63656c546f6b656e733a4e6f7420726567697374657265644241424c546f6b656e3a3a6d696e743a2063616e6e6f74207472616e7366657220746f20746865207a65726f20616464726573734241424c546f6b656e3a3a6d696e743a206d617820737570706c7920657863656564656454696d654c6f636b6564546f6b656e3a3a6465637265617365416c6c6f77616e63653a5370656e6465722063616e6e6f74206265207a65726f206164647265737354696d654c6f636b6564546f6b656e3a3a205f7472616e736665723a20617474656d7074696e6720746f207472616e73666572206c6f636b65642066756e64734241424c546f6b656e3a3a6368616e67654d6178537570706c793a2061206368616e6765206f6e206d6178537570706c79416c6c6f776564206e6f7420616c6c6f77656420796574566f7465546f6b656e3a3a5f7772697465436865636b706f696e743a20626c6f636b206e756d6265722065786365656473203332206269747354696d654c6f636b6564546f6b656e3a3a63616e63656c546f6b656e733a63616e6e6f742063616e63656c206c6f636b656420746f6b656e7320746f20496e766573746f72734241424c546f6b656e3a3a6d696e743a206578636565646564206d696e7420636170206f66203225206f6620746f74616c20737570706c7945524332303a207472616e736665722066726f6d20746865207a65726f206164647265737354696d654c6f636b6564546f6b656e3a3a205f7472616e736665723a20696e73756666696369656e742062616c616e636545524332303a20617070726f76652066726f6d20746865207a65726f20616464726573734241424c546f6b656e3a3a6d696e743a20616d6f756e74206578636565647320393620626974734241424c546f6b656e3a3a6765745072696f72566f7465733a206e6f74207965742064657465726d696e656454696d654c6f636b6564546f6b656e3a3a205f7472616e736665723a2063616e6e6f74207472616e736665722066726f6d20746865207a65726f206164647265737354696d654c6f636b6564546f6b656e3a3a6465637265617365416c6c6f77616e63653a5370656e6465722063616e6e6f7420626520746865206d73672e73656e64657254696d654c6f636b6564546f6b656e3a3a617070726f76653a207370656e6465722063616e6e6f7420626520746865206d73672e73656e6465724241424c546f6b656e3a3a6368616e67654d6178537570706c793a20706f74656e7469616c206d617820616d6f756e7420657863656564732039362062697473566f7465546f6b656e3a3a64656c656761746542795369673a20696e76616c6964207369676e617475726554696d654c6f636b6564546f6b656e3a3a617070726f76653a2063616e6e6f7420646563726561736520616c6c6f77616e636520746f2074696d656c6f636b726567697374727954696d654c6f636b6564546f6b656e3a3a205f7472616e736665723a20646f206e6f74207472616e7366657220746f6b656e7320746f2074686520746f6b656e20636f6e747261637420697473656c6654696d654c6f636b6564546f6b656e3a3a696e637265617365416c6c6f77616e63653a4e6f7420656e6f75676820756e6c6f636b656420746f6b656e734241424c546f6b656e3a3a6d696e743a206d696e74696e67206e6f7420616c6c6f776564206166746572207468652046495253545f45504f43485f4d494e542068617320706173736564203e3d2038207965617273566f7465546f6b656e3a3a5f64656c65676174653a2075696e743936206f766572666c6f77566f7465546f6b656e3a3a64656c656761746542795369673a207369676e6174757265206578706972656454696d654c6f636b6564546f6b656e3a3a696e637265617365416c6c6f77616e63653a5370656e6465722063616e6e6f7420626520746865206d73672e73656e646572a26469706673582212207dbb8d9a484e41c9dbe9780f75d2898dcd8e889e8f43b61c88fa4cd51fb1d5f764736f6c63430007060033

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

000000000000000000000000d4a5b5fcb561daf3adf86f8477555b92fba43b5f

-----Decoded View---------------
Arg [0] : newController (address): 0xD4a5b5fcB561dAF3aDF86F8477555B92FBa43b5F

-----Encoded View---------------
1 Constructor Arguments found :
Arg [0] : 000000000000000000000000d4a5b5fcb561daf3adf86f8477555b92fba43b5f


Loading...
Loading
Loading...
Loading
[ Download: CSV Export  ]
[ Download: CSV Export  ]

A token is a representation of an on-chain or off-chain asset. The token page shows information such as price, total supply, holders, transfers and social links. Learn more about this page in our Knowledge Base.