Token SafeBank

 

Overview [ERC-20]

Price
$0.00 @ 0.000000 Eth (-2.90%)
Fully Diluted Market Cap
Max Total Supply:
100,000,000,000,000 sBANK

Holders:
256

Transfers:
-

 
Loading
[ Download CSV Export  ] 
Loading
[ Download CSV Export  ] 
Loading

OVERVIEW

SafeBitcoin is a token with frictionless fee redistribution and liquidity-adds on transfers. It allows investors to farm the new sBank token at a high percentage yield.

Market

Volume (24H):$1,191.31
Market Capitalization:$0.00
Circulating Supply:0.00 sBANK
Market Data Source: Coinmarketcap

# Exchange Pair Price  24H Volume % Volume
Loading
This contract contains unverified libraries: IterableMapping

Contract Source Code Verified (Exact Match)

Contract Name:
SafeBank

Compiler Version
v0.7.6+commit.7338295f

Optimization Enabled:
Yes with 10000 runs

Other Settings:
default evmVersion, None license

Contract Source Code (Solidity)

/**
 *Submitted for verification at Etherscan.io on 2021-07-18
*/

/**
 *Submitted for verification at BscScan.com on 2021-07-17
*/

/**
//Contract Developed by Snipe.Finance for SafeBTC team

 $$$$$$\           $$$$$$\         $$$$$$$\                   $$\       
$$  __$$\         $$  __$$\        $$  __$$\                  $$ |      
$$ /  \__|$$$$$$\ $$ /  \__$$$$$$\ $$ |  $$ |$$$$$$\ $$$$$$$\ $$ |  $$\ 
\$$$$$$\  \____$$\$$$$\   $$  __$$\$$$$$$$\ |\____$$\$$  __$$\$$ | $$  |
 \____$$\ $$$$$$$ $$  _|  $$$$$$$$ $$  __$$\ $$$$$$$ $$ |  $$ $$$$$$  / 
$$\   $$ $$  __$$ $$ |    $$   ____$$ |  $$ $$  __$$ $$ |  $$ $$  _$$<  
\$$$$$$  \$$$$$$$ $$ |    \$$$$$$$\$$$$$$$  \$$$$$$$ $$ |  $$ $$ | \$$\ 
 \______/ \_______\__|     \_______\_______/ \_______\__|  \__\__|  \__|

 */

pragma solidity ^0.7.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: contracts/Context.sol

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: contracts/IUniswapV2Router01.sol

pragma solidity >=0.6.2;

interface IUniswapV2Router01 {
    function factory() external pure returns (address);
    function WETH() external pure returns (address);

    function addLiquidity(
        address tokenA,
        address tokenB,
        uint amountADesired,
        uint amountBDesired,
        uint amountAMin,
        uint amountBMin,
        address to,
        uint deadline
    ) external returns (uint amountA, uint amountB, uint liquidity);
    function addLiquidityETH(
        address token,
        uint amountTokenDesired,
        uint amountTokenMin,
        uint amountETHMin,
        address to,
        uint deadline
    ) external payable returns (uint amountToken, uint amountETH, uint liquidity);
    function removeLiquidity(
        address tokenA,
        address tokenB,
        uint liquidity,
        uint amountAMin,
        uint amountBMin,
        address to,
        uint deadline
    ) external returns (uint amountA, uint amountB);
    function removeLiquidityETH(
        address token,
        uint liquidity,
        uint amountTokenMin,
        uint amountETHMin,
        address to,
        uint deadline
    ) external returns (uint amountToken, uint amountETH);
    function removeLiquidityWithPermit(
        address tokenA,
        address tokenB,
        uint liquidity,
        uint amountAMin,
        uint amountBMin,
        address to,
        uint deadline,
        bool approveMax, uint8 v, bytes32 r, bytes32 s
    ) external returns (uint amountA, uint amountB);
    function removeLiquidityETHWithPermit(
        address token,
        uint liquidity,
        uint amountTokenMin,
        uint amountETHMin,
        address to,
        uint deadline,
        bool approveMax, uint8 v, bytes32 r, bytes32 s
    ) external returns (uint amountToken, uint amountETH);
    function swapExactTokensForTokens(
        uint amountIn,
        uint amountOutMin,
        address[] calldata path,
        address to,
        uint deadline
    ) external returns (uint[] memory amounts);
    function swapTokensForExactTokens(
        uint amountOut,
        uint amountInMax,
        address[] calldata path,
        address to,
        uint deadline
    ) external returns (uint[] memory amounts);
    function swapExactETHForTokens(uint amountOutMin, address[] calldata path, address to, uint deadline)
        external
        payable
        returns (uint[] memory amounts);
    function swapTokensForExactETH(uint amountOut, uint amountInMax, address[] calldata path, address to, uint deadline)
        external
        returns (uint[] memory amounts);
    function swapExactTokensForETH(uint amountIn, uint amountOutMin, address[] calldata path, address to, uint deadline)
        external
        returns (uint[] memory amounts);
    function swapETHForExactTokens(uint amountOut, address[] calldata path, address to, uint deadline)
        external
        payable
        returns (uint[] memory amounts);

    function quote(uint amountA, uint reserveA, uint reserveB) external pure returns (uint amountB);
    function getAmountOut(uint amountIn, uint reserveIn, uint reserveOut) external pure returns (uint amountOut);
    function getAmountIn(uint amountOut, uint reserveIn, uint reserveOut) external pure returns (uint amountIn);
    function getAmountsOut(uint amountIn, address[] calldata path) external view returns (uint[] memory amounts);
    function getAmountsIn(uint amountOut, address[] calldata path) external view returns (uint[] memory amounts);
}
// File: contracts/IUniswapV2Router02.sol

pragma solidity >=0.6.2;


interface IUniswapV2Router02 is IUniswapV2Router01 {
    function removeLiquidityETHSupportingFeeOnTransferTokens(
        address token,
        uint liquidity,
        uint amountTokenMin,
        uint amountETHMin,
        address to,
        uint deadline
    ) external returns (uint amountETH);
    function removeLiquidityETHWithPermitSupportingFeeOnTransferTokens(
        address token,
        uint liquidity,
        uint amountTokenMin,
        uint amountETHMin,
        address to,
        uint deadline,
        bool approveMax, uint8 v, bytes32 r, bytes32 s
    ) external returns (uint amountETH);

    function swapExactTokensForTokensSupportingFeeOnTransferTokens(
        uint amountIn,
        uint amountOutMin,
        address[] calldata path,
        address to,
        uint deadline
    ) external;
    function swapExactETHForTokensSupportingFeeOnTransferTokens(
        uint amountOutMin,
        address[] calldata path,
        address to,
        uint deadline
    ) external payable;
    function swapExactTokensForETHSupportingFeeOnTransferTokens(
        uint amountIn,
        uint amountOutMin,
        address[] calldata path,
        address to,
        uint deadline
    ) external;
}
// File: contracts/IUniswapV2Factory.sol

pragma solidity >=0.5.0;

interface IUniswapV2Factory {
    event PairCreated(address indexed token0, address indexed token1, address pair, uint);

    function feeTo() external view returns (address);
    function feeToSetter() external view returns (address);

    function getPair(address tokenA, address tokenB) external view returns (address pair);
    function allPairs(uint) external view returns (address pair);
    function allPairsLength() external view returns (uint);

    function createPair(address tokenA, address tokenB) external returns (address pair);

    function setFeeTo(address) external;
    function setFeeToSetter(address) external;
}
// File: contracts/IUniswapV2Pair.sol

pragma solidity >=0.5.0;

interface IUniswapV2Pair {
    event Approval(address indexed owner, address indexed spender, uint value);
    event Transfer(address indexed from, address indexed to, uint value);

    function name() external pure returns (string memory);
    function symbol() external pure returns (string memory);
    function decimals() external pure returns (uint8);
    function totalSupply() external view returns (uint);
    function balanceOf(address owner) external view returns (uint);
    function allowance(address owner, address spender) external view returns (uint);

    function approve(address spender, uint value) external returns (bool);
    function transfer(address to, uint value) external returns (bool);
    function transferFrom(address from, address to, uint value) external returns (bool);

    function DOMAIN_SEPARATOR() external view returns (bytes32);
    function PERMIT_TYPEHASH() external pure returns (bytes32);
    function nonces(address owner) external view returns (uint);

    function permit(address owner, address spender, uint value, uint deadline, uint8 v, bytes32 r, bytes32 s) external;

    event Mint(address indexed sender, uint amount0, uint amount1);
    event Burn(address indexed sender, uint amount0, uint amount1, address indexed to);
    event Swap(
        address indexed sender,
        uint amount0In,
        uint amount1In,
        uint amount0Out,
        uint amount1Out,
        address indexed to
    );
    event Sync(uint112 reserve0, uint112 reserve1);

    function MINIMUM_LIQUIDITY() external pure returns (uint);
    function factory() external view returns (address);
    function token0() external view returns (address);
    function token1() external view returns (address);
    function getReserves() external view returns (uint112 reserve0, uint112 reserve1, uint32 blockTimestampLast);
    function price0CumulativeLast() external view returns (uint);
    function price1CumulativeLast() external view returns (uint);
    function kLast() external view returns (uint);

    function mint(address to) external returns (uint liquidity);
    function burn(address to) external returns (uint amount0, uint amount1);
    function swap(uint amount0Out, uint amount1Out, address to, bytes calldata data) external;
    function skim(address to) external;
    function sync() external;

    function initialize(address, address) external;
}
// File: contracts/IterableMapping.sol

// 
pragma solidity ^0.7.6;

library IterableMapping {
    // Iterable mapping from address to uint;
    struct Map {
        address[] keys;
        mapping(address => uint) values;
        mapping(address => uint) indexOf;
        mapping(address => bool) inserted;
    }

    function get(Map storage map, address key) public view returns (uint) {
        return map.values[key];
    }

    function getIndexOfKey(Map storage map, address key) public view returns (int) {
        if(!map.inserted[key]) {
            return -1;
        }
        return int(map.indexOf[key]);
    }

    function getKeyAtIndex(Map storage map, uint index) public view returns (address) {
        return map.keys[index];
    }



    function size(Map storage map) public view returns (uint) {
        return map.keys.length;
    }

    function set(Map storage map, address key, uint val) public {
        if (map.inserted[key]) {
            map.values[key] = val;
        } else {
            map.inserted[key] = true;
            map.values[key] = val;
            map.indexOf[key] = map.keys.length;
            map.keys.push(key);
        }
    }

    function remove(Map storage map, address key) public {
        if (!map.inserted[key]) {
            return;
        }

        delete map.inserted[key];
        delete map.values[key];

        uint index = map.indexOf[key];
        uint lastIndex = map.keys.length - 1;
        address lastKey = map.keys[lastIndex];

        map.indexOf[lastKey] = index;
        delete map.indexOf[key];

        map.keys[index] = lastKey;
        map.keys.pop();
    }
}
// File: contracts/Ownable.sol

// 

pragma solidity ^0.7.0;

/**
 * @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 () {
        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: contracts/IDividendPayingTokenOptional.sol

pragma solidity ^0.7.6;


/// @title Dividend-Paying Token Optional Interface
/// @author Roger Wu (https://github.com/roger-wu)
/// @dev OPTIONAL functions for a dividend-paying token contract.
interface IDividendPayingTokenOptional {
  /// @notice View the amount of dividend in wei that an address can withdraw.
  /// @param _owner The address of a token holder.
  /// @return The amount of dividend in wei that `_owner` can withdraw.
  function withdrawableDividendOf(address _owner) external view returns(uint256);

  /// @notice View the amount of dividend in wei that an address has withdrawn.
  /// @param _owner The address of a token holder.
  /// @return The amount of dividend in wei that `_owner` has withdrawn.
  function withdrawnDividendOf(address _owner) external view returns(uint256);

  /// @notice View the amount of dividend in wei that an address has earned in total.
  /// @dev accumulativeDividendOf(_owner) = withdrawableDividendOf(_owner) + withdrawnDividendOf(_owner)
  /// @param _owner The address of a token holder.
  /// @return The amount of dividend in wei that `_owner` has earned in total.
  function accumulativeDividendOf(address _owner) external view returns(uint256);
}
// File: contracts/IDividendPayingToken.sol

pragma solidity ^0.7.6;


/// @title Dividend-Paying Token Interface
/// @author Roger Wu (https://github.com/roger-wu)
/// @dev An interface for a dividend-paying token contract.
interface IDividendPayingToken {
  /// @notice View the amount of dividend in wei that an address can withdraw.
  /// @param _owner The address of a token holder.
  /// @return The amount of dividend in wei that `_owner` can withdraw.
  function dividendOf(address _owner) external view returns(uint256);

  /// @notice Distributes ether to token holders as dividends.
  /// @dev SHOULD distribute the paid ether to token holders as dividends.
  ///  SHOULD NOT directly transfer ether to token holders in this function.
  ///  MUST emit a `DividendsDistributed` event when the amount of distributed ether is greater than 0.
  function distributeDividends() external payable;

  /// @notice Withdraws the ether distributed to the sender.
  /// @dev SHOULD transfer `dividendOf(msg.sender)` wei to `msg.sender`, and `dividendOf(msg.sender)` SHOULD be 0 after the transfer.
  ///  MUST emit a `DividendWithdrawn` event if the amount of ether transferred is greater than 0.
  function withdrawDividend() external;

  /// @dev This event MUST emit when ether is distributed to token holders.
  /// @param from The address which sends ether to this contract.
  /// @param weiAmount The amount of distributed ether in wei.
  event DividendsDistributed(
    address indexed from,
    uint256 weiAmount
  );

  /// @dev This event MUST emit when an address withdraws their dividend.
  /// @param to The address which withdraws ether from this contract.
  /// @param weiAmount The amount of withdrawn ether in wei.
  event DividendWithdrawn(
    address indexed to,
    uint256 weiAmount,
    address indexed tokenWithdrawn
  );
}
// File: contracts/SafeMathInt.sol

pragma solidity ^0.7.6;

/**
 * @title SafeMathInt
 * @dev Math operations with safety checks that revert on error
 * @dev SafeMath adapted for int256
 * Based on code of  https://github.com/RequestNetwork/requestNetwork/blob/master/packages/requestNetworkSmartContracts/contracts/base/math/SafeMathInt.sol
 */
library SafeMathInt {
  function mul(int256 a, int256 b) internal pure returns (int256) {
    // Prevent overflow when multiplying INT256_MIN with -1
    // https://github.com/RequestNetwork/requestNetwork/issues/43
    require(!(a == - 2**255 && b == -1) && !(b == - 2**255 && a == -1));

    int256 c = a * b;
    require((b == 0) || (c / b == a));
    return c;
  }

  function div(int256 a, int256 b) internal pure returns (int256) {
    // Prevent overflow when dividing INT256_MIN by -1
    // https://github.com/RequestNetwork/requestNetwork/issues/43
    require(!(a == - 2**255 && b == -1) && (b > 0));

    return a / b;
  }

  function sub(int256 a, int256 b) internal pure returns (int256) {
    require((b >= 0 && a - b <= a) || (b < 0 && a - b > a));

    return a - b;
  }

  function add(int256 a, int256 b) internal pure returns (int256) {
    int256 c = a + b;
    require((b >= 0 && c >= a) || (b < 0 && c < a));
    return c;
  }

  function toUint256Safe(int256 a) internal pure returns (uint256) {
    require(a >= 0);
    return uint256(a);
  }
}
// File: contracts/SafeMathUint.sol

pragma solidity ^0.7.6;

/**
 * @title SafeMathUint
 * @dev Math operations with safety checks that revert on error
 */
library SafeMathUint {
  function toInt256Safe(uint256 a) internal pure returns (int256) {
    int256 b = int256(a);
    require(b >= 0);
    return b;
  }
}

// File: contracts/ERC20.sol

// 

pragma solidity ^0.7.0;




/**
 * @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_) {
        _name = name_;
        _symbol = symbol_;
        _decimals = 9;
    }

    /**
     * @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: contracts/SafeMath.sol

// 

pragma solidity ^0.7.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: contracts/SafeBankDividendPayingToken.sol

// 

pragma solidity ^0.7.6;









/// @title Dividend-Paying Token
/// @author Roger Wu (https://github.com/roger-wu)
/// @dev A mintable ERC20 token that allows anyone to pay and distribute ether
///  to token holders as dividends and allows token holders to withdraw their dividends.
///  Reference: the source code of PoWH3D: https://etherscan.io/address/0xB3775fB83F7D12A36E0475aBdD1FCA35c091efBe#code
contract DividendPayingToken is ERC20, IDividendPayingToken, IDividendPayingTokenOptional {
  using SafeMath for uint256;
  using SafeMathUint for uint256;
  using SafeMathInt for int256;

  // With `magnitude`, we can properly distribute dividends even if the amount of received ether is small.
  // For more discussion about choosing the value of `magnitude`,
  //  see https://github.com/ethereum/EIPs/issues/1726#issuecomment-472352728
  uint256 constant internal magnitude = 2**128;

  uint256 internal magnifiedDividendPerShare;
  uint256 internal lastAmount;
  
  address public DividendToken = address(0);

  // About dividendCorrection:
  // If the token balance of a `_user` is never changed, the dividend of `_user` can be computed with:
  //   `dividendOf(_user) = dividendPerShare * balanceOf(_user)`.
  // When `balanceOf(_user)` is changed (via minting/burning/transferring tokens),
  //   `dividendOf(_user)` should not be changed,
  //   but the computed value of `dividendPerShare * balanceOf(_user)` is changed.
  // To keep the `dividendOf(_user)` unchanged, we add a correction term:
  //   `dividendOf(_user) = dividendPerShare * balanceOf(_user) + dividendCorrectionOf(_user)`,
  //   where `dividendCorrectionOf(_user)` is updated whenever `balanceOf(_user)` is changed:
  //   `dividendCorrectionOf(_user) = dividendPerShare * (old balanceOf(_user)) - (new balanceOf(_user))`.
  // So now `dividendOf(_user)` returns the same value before and after `balanceOf(_user)` is changed.
  mapping(address => int256) internal magnifiedDividendCorrections;
  mapping(address => uint256) internal withdrawnDividends;

  uint256 public totalDividendsDistributed;

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

  }
  

  receive() external payable {
  }

  /// @notice Distributes ether to token holders as dividends.
  /// @dev It reverts if the total supply of tokens is 0.
  /// It emits the `DividendsDistributed` event if the amount of received ether is greater than 0.
  /// About undistributed ether:
  ///   In each distribution, there is a small amount of ether not distributed,
  ///     the magnified amount of which is
  ///     `(msg.value * magnitude) % totalSupply()`.
  ///   With a well-chosen `magnitude`, the amount of undistributed ether
  ///     (de-magnified) in a distribution can be less than 1 wei.
  ///   We can actually keep track of the undistributed ether in a distribution
  ///     and try to distribute it in the next distribution,
  ///     but keeping track of such data on-chain costs much more than
  ///     the saved ether, so we don't do that.
  function distributeDividends() public override payable {
    require(totalSupply() > 0);

    if (msg.value > 0) {
      magnifiedDividendPerShare = magnifiedDividendPerShare.add(
        (msg.value).mul(magnitude) / totalSupply()
      );
      emit DividendsDistributed(msg.sender, msg.value);

      totalDividendsDistributed = totalDividendsDistributed.add(msg.value);
    }
  }
  

  function distributeTokenDividends(uint256 amount) public {
    require(totalSupply() > 0);

    if (amount > 0) {
      magnifiedDividendPerShare = magnifiedDividendPerShare.add(
        (amount).mul(magnitude) / totalSupply()
      );
      emit DividendsDistributed(msg.sender, amount);

      totalDividendsDistributed = totalDividendsDistributed.add(amount);
    }
  }

  /// @notice Withdraws the ether distributed to the sender.
  /// @dev It emits a `DividendWithdrawn` event if the amount of withdrawn ether is greater than 0.
  function withdrawDividend() public virtual override {
    _withdrawDividendOfUser(msg.sender);
  }

  /// @notice Withdraws the ether distributed to the sender.
  /// @dev It emits a `DividendWithdrawn` event if the amount of withdrawn ether is greater than 0.
  /// modified to support BNB dividend
  function _withdrawDividendOfUser(address payable user) internal returns (uint256) {
    uint256 _withdrawableDividend = withdrawableDividendOf(user);
    if (_withdrawableDividend > 0) {
      withdrawnDividends[user] = withdrawnDividends[user].add(_withdrawableDividend);

        bool success = false;
        if(DividendToken == address(0)){
            (bool sent, bytes memory data) = user.call{value: _withdrawableDividend}("");
            success = sent;
            emit DividendWithdrawn(user, _withdrawableDividend, DividendToken);     
        }else{
            success = IERC20(DividendToken).transfer(user, _withdrawableDividend);
            emit DividendWithdrawn(user, _withdrawableDividend, DividendToken);
        }

      if(!success) {
        withdrawnDividends[user] = withdrawnDividends[user].sub(_withdrawableDividend);
        return 0;
      }

      return _withdrawableDividend;
    }

    return 0;
  }


  /// @notice View the amount of dividend in wei that an address can withdraw.
  /// @param _owner The address of a token holder.
  /// @return The amount of dividend in wei that `_owner` can withdraw.
  function dividendOf(address _owner) public view override returns(uint256) {
    return withdrawableDividendOf(_owner);
  }

  /// @notice View the amount of dividend in wei that an address can withdraw.
  /// @param _owner The address of a token holder.
  /// @return The amount of dividend in wei that `_owner` can withdraw.
  function withdrawableDividendOf(address _owner) public view override returns(uint256) {
    return accumulativeDividendOf(_owner).sub(withdrawnDividends[_owner]);
  }

  /// @notice View the amount of dividend in wei that an address has withdrawn.
  /// @param _owner The address of a token holder.
  /// @return The amount of dividend in wei that `_owner` has withdrawn.
  function withdrawnDividendOf(address _owner) public view override returns(uint256) {
    return withdrawnDividends[_owner];
  }


  /// @notice View the amount of dividend in wei that an address has earned in total.
  /// @dev accumulativeDividendOf(_owner) = withdrawableDividendOf(_owner) + withdrawnDividendOf(_owner)
  /// = (magnifiedDividendPerShare * balanceOf(_owner) + magnifiedDividendCorrections[_owner]) / magnitude
  /// @param _owner The address of a token holder.
  /// @return The amount of dividend in wei that `_owner` has earned in total.
  function accumulativeDividendOf(address _owner) public view override returns(uint256) {
    return magnifiedDividendPerShare.mul(balanceOf(_owner)).toInt256Safe()
      .add(magnifiedDividendCorrections[_owner]).toUint256Safe() / magnitude;
  }

  /// @dev Internal function that transfer tokens from one address to another.
  /// Update magnifiedDividendCorrections to keep dividends unchanged.
  /// @param from The address to transfer from.
  /// @param to The address to transfer to.
  /// @param value The amount to be transferred.
  function _transfer(address from, address to, uint256 value) internal virtual override {
    require(false);

    int256 _magCorrection = magnifiedDividendPerShare.mul(value).toInt256Safe();
    magnifiedDividendCorrections[from] = magnifiedDividendCorrections[from].add(_magCorrection);
    magnifiedDividendCorrections[to] = magnifiedDividendCorrections[to].sub(_magCorrection);
  }

  /// @dev Internal function that mints tokens to an account.
  /// Update magnifiedDividendCorrections to keep dividends unchanged.
  /// @param account The account that will receive the created tokens.
  /// @param value The amount that will be created.
  function _mint(address account, uint256 value) internal override {
    super._mint(account, value);

    magnifiedDividendCorrections[account] = magnifiedDividendCorrections[account]
      .sub( (magnifiedDividendPerShare.mul(value)).toInt256Safe() );
  }

  /// @dev Internal function that burns an amount of the token of a given account.
  /// Update magnifiedDividendCorrections to keep dividends unchanged.
  /// @param account The account whose tokens will be burnt.
  /// @param value The amount that will be burnt.
  function _burn(address account, uint256 value) internal override {
    super._burn(account, value);

    magnifiedDividendCorrections[account] = magnifiedDividendCorrections[account]
      .add( (magnifiedDividendPerShare.mul(value)).toInt256Safe() );
  }

  function _setBalance(address account, uint256 newBalance) internal {
    uint256 currentBalance = balanceOf(account);

    if(newBalance > currentBalance) {
      uint256 mintAmount = newBalance.sub(currentBalance);
      _mint(account, mintAmount);
    } else if(newBalance < currentBalance) {
      uint256 burnAmount = currentBalance.sub(newBalance);
      _burn(account, burnAmount);
    }
  }
}

// File: contracts/SafeBank.sol

// 

pragma solidity ^0.7.6;









contract SafeBank is ERC20, Ownable {
    using SafeMath for uint256;

    IUniswapV2Router02 public uniswapV2Router;
    address public immutable uniswapV2Pair;

    address public DividendToken = address(0);

    bool private swapping;

    SafeBankDividendTracker public dividendTracker;

    address public burnAddress;
    
    uint256 public maxBuyTranscationAmount = 100000000000000 * (10**9);
    uint256 public maxSellTransactionAmount = 100000000000000 * (10**9);
    uint256 public swapTokensAtAmount = 5000000 * (10**9);
    uint256 public _maxWalletToken = 100000000000000 * (10**9);

    // added
    address payable public wallet1Address;
    address payable public wallet2Address;
    address payable public wallet3Address;
    address public wallet1TokenAddressForFee;
    address public wallet2TokenAddressForFee;
    address public wallet3TokenAddressForFee;
    uint256 public wallet1Fee;
    uint256 public wallet2Fee;
    uint256 public wallet3Fee;
    uint256 public tokenRewardsFee;
    uint256 public liquidityFee;
    uint256 public totalFees;

    // fee aggiuntiva alla vendita del 20% 
    uint256 public sellFeeIncreaseFactor = 1200;

    // autoclaim gas to 400k
    uint256 public gasForProcessing = 400000;
    
    address public presaleAddress = address(0);

    // timestamp for when the token can be traded freely on PanackeSwap
    uint256 public tradingEnabledTimestamp = 1626624000;

    // exlcude from fees and max transaction amount
    mapping (address => bool) private _isExcludedFromFees;
    mapping (address => bool) public _isExcludedMaxSellTransactionAmount;

    // addresses that can make transfers before presale is over
    mapping (address => bool) private canTransferBeforeTradingIsEnabled;

    // store addresses that a automatic market maker pairs. Any transfer *to* these addresses
    // could be subject to a maximum transfer amount
    mapping (address => bool) public automatedMarketMakerPairs;

    event UpdateDividendTracker(address indexed newAddress, address indexed oldAddress);

    event UpdateUniswapV2Router(address indexed newAddress, address indexed oldAddress);
    // added
    event UpdateDividendToken(address indexed newAddress, address indexed oldAddress);
    //

    event ExcludeFromFees(address indexed account, bool isExcluded);
    event ExcludeMultipleAccountsFromFees(address[] accounts, bool isExcluded);
    event ExcludedMaxSellTransactionAmount(address indexed account, bool isExcluded);

    event SetAutomatedMarketMakerPair(address indexed pair, bool indexed value);

    event BurnWalletUpdated(address indexed newBurnWallet, address indexed oldBurnWallet);

    event GasForProcessingUpdated(uint256 indexed newValue, uint256 indexed oldValue);

    event FixedSaleBuy(address indexed account, uint256 indexed amount, bool indexed earlyParticipant, uint256 numberOfBuyers);

    event SwapAndLiquify(
        uint256 tokensSwapped,
        uint256 ethReceived,
        uint256 tokensIntoLiqudity
    );

    event SendDividends(
        uint256 tokensSwapped,
        uint256 amount
    );

    event ProcessedDividendTracker(
        uint256 iterations,
        uint256 claims,
        uint256 lastProcessedIndex,
        bool indexed automatic,
        uint256 gas,
        address indexed processor
    );

    constructor() public ERC20("SafeBank", "sBANK") {
        uint256 _tokenRewardsFee = 3;
        uint256 _liquidityFee = 2;
        uint256 _wallet1Fee = 1;
        uint256 _wallet2Fee = 2;
        uint256 _wallet3Fee = 2;

        tokenRewardsFee = _tokenRewardsFee;
        liquidityFee = _liquidityFee;
        wallet1Fee = _wallet1Fee;
        wallet2Fee = _wallet2Fee;
        wallet3Fee = _wallet3Fee;
        totalFees = _tokenRewardsFee.add(_liquidityFee).add(_wallet1Fee).add(_wallet2Fee).add(_wallet3Fee);
        


        dividendTracker = new SafeBankDividendTracker();

        burnAddress = address(0xdead);

        
        IUniswapV2Router02 _uniswapV2Router = IUniswapV2Router02(0x7a250d5630B4cF539739dF2C5dAcb4c659F2488D);
         // Create a uniswap pair for this new token
        address _uniswapV2Pair = IUniswapV2Factory(_uniswapV2Router.factory())
            .createPair(address(this), _uniswapV2Router.WETH());

        uniswapV2Router = _uniswapV2Router;
        uniswapV2Pair = _uniswapV2Pair;

        _setAutomatedMarketMakerPair(_uniswapV2Pair, true);

        // exclude from receiving dividends
        dividendTracker.excludeFromDividends(address(dividendTracker));
        dividendTracker.excludeFromDividends(address(this));
        dividendTracker.excludeFromDividends(address(_uniswapV2Router));
        dividendTracker.excludeFromDividends(owner());

        

        // exclude from paying fees or having max transaction amount
        excludeFromFees(burnAddress, true);
        excludeFromFees(address(this), true);
        excludeFromFees(owner(), true);


        // enable owner and fixed-sale wallet to send tokens before presales are over
        canTransferBeforeTradingIsEnabled[owner()] = true;
        /*
            _mint is an internal function in ERC20.sol that is only called here,
            and CANNOT be called ever again
        */
        _mint(owner(), 100000000000000 * (10**9));
    }

    receive() external payable {

    }
    
    // added
    function updateTradingEnabledTime (uint256 newTimeInEpoch) external onlyOwner {
        tradingEnabledTimestamp = newTimeInEpoch;
    }     

    function updateMinimumBalanceForDividends (uint256 newAmountNoDecimials) external onlyOwner {
        dividendTracker.updateMinimumBalanceForDividends(newAmountNoDecimials);
    }    

    function updateSellIncreaseFee (uint256 newFeeWholeNumber) external onlyOwner {
        sellFeeIncreaseFactor = newFeeWholeNumber;
    }
    
    function updateMaxWalletAmount(uint256 newAmountNoDecimials) external onlyOwner {
        _maxWalletToken = newAmountNoDecimials * (10**9);
    }     

    function updateSwapAtAmount(uint256 newAmountNoDecimials) external onlyOwner {
        swapTokensAtAmount = newAmountNoDecimials * (10**9);
    } 

    function updateTokenForDividend(address newAddress) external onlyOwner {
        dividendTracker.updateTokenForDividend(newAddress);
        DividendToken = newAddress;
        emit UpdateDividendToken(newAddress, address(DividendToken));
    }    

    function updateWallet1Address(address payable newAddress) external onlyOwner {
        wallet1Address = newAddress;
        excludeFromFees(wallet1Address, true);
        dividendTracker.excludeFromDividends(wallet1Address);     
    }    

    function updateWallet2Address(address payable newAddress) external onlyOwner {
        wallet2Address = newAddress;
        excludeFromFees(wallet2Address, true);
        dividendTracker.excludeFromDividends(wallet2Address);
    }       

    function updateWallet3Address(address payable newAddress) external onlyOwner {
        wallet3Address = newAddress;
        excludeFromFees(wallet3Address, true);
        dividendTracker.excludeFromDividends(wallet3Address);
    }    

    function updateWallet1TokenFeeAddress(address newAddress) external onlyOwner {
        wallet1TokenAddressForFee = newAddress;
    }    

    function updateWallet2TokenFeeAddress(address newAddress) external onlyOwner {
        wallet2TokenAddressForFee = newAddress;
    }    

    function updateWallet3TokenFeeAddress(address newAddress) external onlyOwner {
        wallet3TokenAddressForFee = newAddress;
    }

    function updateFees(uint256 _tokenRewardsFee, uint256 _liquidityFee, uint256 _wallet1Fee, uint256 _wallet2Fee, uint256 _wallet3Fee) external onlyOwner {
        tokenRewardsFee = _tokenRewardsFee;
        liquidityFee = _liquidityFee;
        wallet1Fee = _wallet1Fee;
        wallet2Fee = _wallet2Fee;
        wallet3Fee = _wallet3Fee;
        totalFees = tokenRewardsFee.add(liquidityFee).add(wallet1Fee).add(wallet2Fee).add(wallet3Fee);
    }
    //
    function whitelistDxSale(address _presaleAddress, address _routerAddress) external onlyOwner {
        presaleAddress = _presaleAddress;
        canTransferBeforeTradingIsEnabled[presaleAddress] = true;
        dividendTracker.excludeFromDividends(_presaleAddress);
        excludeFromFees(_presaleAddress, true);

        canTransferBeforeTradingIsEnabled[_routerAddress] = true;
        dividendTracker.excludeFromDividends(_routerAddress);
        excludeFromFees(_routerAddress, true);
    }

    function updateDividendTracker(address newAddress) external onlyOwner {
        require(newAddress != address(dividendTracker), "SafeBank: The dividend tracker already has that address");

        SafeBankDividendTracker newDividendTracker = SafeBankDividendTracker(payable(newAddress));

        require(newDividendTracker.owner() == address(this), "SafeBank: The new dividend tracker must be owned by the SafeBank token contract");

        newDividendTracker.excludeFromDividends(address(newDividendTracker));
        newDividendTracker.excludeFromDividends(address(this));
        newDividendTracker.excludeFromDividends(address(uniswapV2Router));

        emit UpdateDividendTracker(newAddress, address(dividendTracker));

        dividendTracker = newDividendTracker;
    }

    function updateUniswapV2Router(address newAddress) external onlyOwner {
        require(newAddress != address(uniswapV2Router), "SafeBank: The router already has that address");
        emit UpdateUniswapV2Router(newAddress, address(uniswapV2Router));
        uniswapV2Router = IUniswapV2Router02(newAddress);
    }

    function excludeFromFees(address account, bool excluded) public onlyOwner {
        //require(_isExcludedFromFees[account] != excluded, "SafeBank: Account is already the value of 'excluded'");
        _isExcludedFromFees[account] = excluded;

        emit ExcludeFromFees(account, excluded);
    }

    function excludeMultipleAccountsFromFees(address[] calldata accounts, bool excluded) external onlyOwner {
        for(uint256 i = 0; i < accounts.length; i++) {
            _isExcludedFromFees[accounts[i]] = excluded;
        }

        emit ExcludeMultipleAccountsFromFees(accounts, excluded);
    }

    function setAutomatedMarketMakerPair(address pair, bool value) external onlyOwner {
        require(pair != uniswapV2Pair, "SafeBank: The PancakeSwap pair cannot be removed from automatedMarketMakerPairs");

        _setAutomatedMarketMakerPair(pair, value);
    }

    function _setAutomatedMarketMakerPair(address pair, bool value) private {
        require(automatedMarketMakerPairs[pair] != value, "SafeBank: Automated market maker pair is already set to that value");
        automatedMarketMakerPairs[pair] = value;

        if(value) {
            dividendTracker.excludeFromDividends(pair);
        }

        emit SetAutomatedMarketMakerPair(pair, value);
    }

    function updateGasForProcessing(uint256 newValue) public onlyOwner {
        require(newValue >= 200000 && newValue <= 500000, "SafeBank: gasForProcessing must be between 200,000 and 500,000");
        require(newValue != gasForProcessing, "SafeBank: Cannot update gasForProcessing to same value");
        emit GasForProcessingUpdated(newValue, gasForProcessing);
        gasForProcessing = newValue;
    }

    function updateClaimWait(uint256 claimWait) external onlyOwner {
        dividendTracker.updateClaimWait(claimWait);
    }

    function getClaimWait() external view returns(uint256) {
        return dividendTracker.claimWait();
    }

    function getTotalDividendsDistributed() external view returns (uint256) {
        return dividendTracker.totalDividendsDistributed();
    }

    function isExcludedFromFees(address account) public view returns(bool) {
        return _isExcludedFromFees[account];
    }

    function withdrawableDividendOf(address account) public view returns(uint256) {
        return dividendTracker.withdrawableDividendOf(account);
    }

    function dividendTokenBalanceOf(address account) public view returns (uint256) {
        return dividendTracker.balanceOf(account);
    }

    function getAccountDividendsInfo(address account)
        external view returns (
            address,
            int256,
            int256,
            uint256,
            uint256,
            uint256,
            uint256,
            uint256) {
        return dividendTracker.getAccount(account);
    }

    function getAccountDividendsInfoAtIndex(uint256 index)
        external view returns (
            address,
            int256,
            int256,
            uint256,
            uint256,
            uint256,
            uint256,
            uint256) {
        return dividendTracker.getAccountAtIndex(index);
    }

    function processDividendTracker(uint256 gas) external {
        (uint256 iterations, uint256 claims, uint256 lastProcessedIndex) = dividendTracker.process(gas);
        emit ProcessedDividendTracker(iterations, claims, lastProcessedIndex, false, gas, tx.origin);
    }

    function claim() external {
        dividendTracker.processAccount(msg.sender, false);
    }

    function getLastProcessedIndex() external view returns(uint256) {
        return dividendTracker.getLastProcessedIndex();
    }

    function getNumberOfDividendTokenHolders() external view returns(uint256) {
        return dividendTracker.getNumberOfTokenHolders();
    }

    function getTradingIsEnabled() public view returns (bool) {
        return block.timestamp >= tradingEnabledTimestamp;
    }

    function _transfer(
        address from,
        address to,
        uint256 amount
    ) internal override {
        require(from != address(0), "ERC20: transfer from the zero address");
        require(to != address(0), "ERC20: transfer to the zero address");
        
            if (
            from != owner() &&
            to != owner() &&
            to != address(0) &&
            to != address(0xdead) &&
            to != uniswapV2Pair
        ) {
            require(
                amount <= maxBuyTranscationAmount,
                "Transfer amount exceeds the maxTxAmount."
            );
            
            uint256 contractBalanceRecepient = balanceOf(to);
            require(
                contractBalanceRecepient + amount <= _maxWalletToken,
                "Exceeds maximum wallet token amount."
            );
        }
        
        
        
        bool tradingIsEnabled = getTradingIsEnabled();

        // only whitelisted addresses can make transfers after the fixed-sale has started
        // and before the public presale is over
        if(!tradingIsEnabled) {
            require(canTransferBeforeTradingIsEnabled[from], "SafeBank: This account cannot send tokens until trading is enabled");
        }

        if(amount == 0) {
            super._transfer(from, to, 0);
            return;
        }

        if( 
            !swapping &&
            tradingIsEnabled &&
            automatedMarketMakerPairs[to] && // sells only by detecting transfer to automated market maker pair
            from != address(uniswapV2Router) && //router -> pair is removing liquidity which shouldn't have max
            !_isExcludedFromFees[to] //no max for those excluded from fees
        ) {
            require(amount <= maxSellTransactionAmount, "Sell transfer amount exceeds the maxSellTransactionAmount.");
        }

        uint256 contractTokenBalance = balanceOf(address(this));
        
        bool canSwap = contractTokenBalance >= swapTokensAtAmount;

        if(
            tradingIsEnabled && 
            canSwap &&
            !swapping &&
            !automatedMarketMakerPairs[from] &&
            from != burnAddress &&
            to != burnAddress
        ) {
            swapping = true;

            uint256 swapTokens = contractTokenBalance.mul(liquidityFee).div(totalFees);
            swapAndLiquify(swapTokens);

            uint256 sellTokens = balanceOf(address(this));
            swapAndSendDividends(sellTokens);

            swapping = false;
        }


        bool takeFee = tradingIsEnabled && !swapping;

        // if any account belongs to _isExcludedFromFee account then remove the fee
        if(_isExcludedFromFees[from] || _isExcludedFromFees[to]) {
            takeFee = false;
        }

        if(takeFee) {
            uint256 fees = amount.mul(totalFees).div(100);

            // if sell, multiply by 1.2
            if(automatedMarketMakerPairs[to]) {
                fees = fees.mul(sellFeeIncreaseFactor).div(100);
            }

            amount = amount.sub(fees);

            super._transfer(from, address(this), fees);
        }

        super._transfer(from, to, amount);

        try dividendTracker.setBalance(payable(from), balanceOf(from)) {} catch {}
        try dividendTracker.setBalance(payable(to), balanceOf(to)) {} catch {}

        if(!swapping) {
            uint256 gas = gasForProcessing;

            try dividendTracker.process(gas) returns (uint256 iterations, uint256 claims, uint256 lastProcessedIndex) {
                emit ProcessedDividendTracker(iterations, claims, lastProcessedIndex, true, gas, tx.origin);
            }
            catch {

            }
        }
    }
    
    // made public for testing
    function swapAndLiquify(uint256 tokens) public  {
        // split the contract balance into halves
        uint256 half = tokens.div(2);
        uint256 otherHalf = tokens.sub(half);

        // capture the contract's current ETH balance.
        // this is so that we can capture exactly the amount of ETH that the
        // swap creates, and not make the liquidity event include any ETH that
        // has been manually sent to the contract
        uint256 initialBalance = address(this).balance;

        // swap tokens for ETH
        swapTokensForEth(half); // <- this breaks the ETH -> HATE swap when swap+liquify is triggered

        // how much ETH did we just swap into?
        uint256 newBalance = address(this).balance.sub(initialBalance);
        
        // calculate the portions of the liquidity to add to wallet1fee
        uint256 wallet1feeBalance = newBalance.div(totalFees).mul(wallet1Fee);
        uint256 wallet1feePortion = otherHalf.div(totalFees).mul(wallet1Fee);        
        // calculate the portions of the liquidity to add to wallet2fee
        uint256 wallet2feeBalance = newBalance.div(totalFees).mul(wallet2Fee);
        uint256 wallet2feePortion = otherHalf.div(totalFees).mul(wallet2Fee);
        // calculate the portions of the liquidity to add to wallet3fee
        uint256 wallet3feeBalance = newBalance.div(totalFees).mul(wallet3Fee);
        uint256 wallet3feePortion = otherHalf.div(totalFees).mul(wallet3Fee);
        uint256 walletTotalBalance = wallet1feeBalance + wallet2feeBalance + wallet3feeBalance;
        uint256 walletTotalPortion = wallet1feePortion + wallet2feePortion + wallet3feePortion;
        uint256 finalBalance = newBalance.sub(walletTotalBalance);
        uint256 finalHalf = otherHalf.sub(walletTotalPortion);
        
        
        // added to manage receiving eth or any token 1
        if(wallet1TokenAddressForFee != address(0)){
            swapEthForTokens(wallet1feeBalance, wallet1TokenAddressForFee, wallet1Address);
            //_transfer(address(this), burnAddress, wallet1feePortion);
            //emit Transfer(address(this), burnAddress, wallet1feePortion);
        }else{
            (bool sent, bytes memory data) = wallet1Address.call{value: wallet1feeBalance}("");
            if(sent){
                //_transfer(address(this), burnAddress, wallet1feePortion);
                //emit Transfer(address(this), burnAddress, wallet1feePortion);
            } else {
                addLiquidity(wallet1feePortion, wallet1feeBalance);
            }
        }        

        // added to manage receiving eth or any token 2
        if(wallet2TokenAddressForFee != address(0)){
            swapEthForTokens(wallet2feeBalance, wallet2TokenAddressForFee, wallet2Address);
            //_transfer(address(this), burnAddress, wallet2feePortion);
            //emit Transfer(address(this), burnAddress, wallet2feePortion);
        }else{
            (bool sent, bytes memory data) = wallet2Address.call{value: wallet2feeBalance}("");
            if(sent){
                //_transfer(address(this), burnAddress, wallet2feePortion);
                //emit Transfer(address(this), burnAddress, wallet2feePortion);
            } else {
                addLiquidity(wallet2feePortion, wallet2feeBalance);
            }
        }        

        // added to manage receiving eth or any token 3
        if(wallet3TokenAddressForFee != address(0)){
            swapEthForTokens(wallet3feeBalance, wallet3TokenAddressForFee, wallet3Address);
            //_transfer(address(this), burnAddress, wallet3feePortion);
            //emit Transfer(address(this), burnAddress, wallet3feePortion);
        }else{
            (bool sent, bytes memory data) = wallet3Address.call{value: wallet3feeBalance}("");
            if(sent){
                //_transfer(address(this), burnAddress, wallet3feePortion);
                //emit Transfer(address(this), burnAddress, wallet3feePortion);
            } else {
                addLiquidity(wallet3feePortion, wallet3feeBalance);
            }
        }
        
        // add liquidity to uniswap
        addLiquidity(finalHalf, finalBalance);
        
        emit SwapAndLiquify(half, newBalance, otherHalf);
    }
    
    // added in order to sell eth for something else like usdt,busd,etc
    function swapEthForTokens(uint256 ethAmount, address tokenAddress, address receiver) private {
        // generate the uniswap pair path of weth -> token
        address[] memory path = new address[](2);
        path[0] = uniswapV2Router.WETH();
        path[1] = tokenAddress;

        // make the swap
        uniswapV2Router.swapExactETHForTokensSupportingFeeOnTransferTokens{value: ethAmount}(
            0, // accept any amount of ETH
            path,
            receiver,
            block.timestamp
        );
    }

    function swapTokensForEth(uint256 tokenAmount) private {

        
        // generate the uniswap pair path of token -> weth
        address[] memory path = new address[](2);
        path[0] = address(this);
        path[1] = uniswapV2Router.WETH();

        _approve(address(this), address(uniswapV2Router), tokenAmount);

        // make the swap
        uniswapV2Router.swapExactTokensForETHSupportingFeeOnTransferTokens(
            tokenAmount,
            0, // accept any amount of ETH
            path,
            address(this),
            block.timestamp
        );
        
    }


    function swapTokensForTokens(uint256 tokenAmount, address recipient) private {
       
        // generate the uniswap pair path of weth -> busd
        address[] memory path = new address[](3);
        path[0] = address(this);
        path[1] = uniswapV2Router.WETH();
        path[2] = DividendToken;

        _approve(address(this), address(uniswapV2Router), tokenAmount);

        // make the swap
        uniswapV2Router.swapExactTokensForTokensSupportingFeeOnTransferTokens(
            tokenAmount,
            0, // accept any amount of BUSD
            path,
            recipient,
            block.timestamp
        );
        
    }

    function addLiquidity(uint256 tokenAmount, uint256 ethAmount) public {
        
        // approve token transfer to cover all possible scenarios
        _approve(address(this), address(uniswapV2Router), tokenAmount);

        // add the liquidity
       uniswapV2Router.addLiquidityETH{value: ethAmount}(
            address(this),
            tokenAmount,
            0, // slippage is unavoidable
            0, // slippage is unavoidable
            burnAddress,
            block.timestamp
        );
        
    }

    // modified to support BNB dividend
    function swapAndSendDividends(uint256 tokens) private {
        address payable diviTracker = address(dividendTracker);
        bool success = false;
        uint256 dividends;
        if(DividendToken != address(0)){
            swapTokensForTokens(tokens, address(this));
            dividends = IERC20(DividendToken).balanceOf(address(this));
            success = IERC20(DividendToken).transfer(address(dividendTracker), dividends);
        }else{
            uint256 initialBalance = address(this).balance;
            swapTokensForEth(tokens);
            uint256 newBalance = address(this).balance.sub(initialBalance);
            dividends = newBalance;
            (bool sent, bytes memory data) = diviTracker.call{value: newBalance}("");
            success = sent;
        }
        if (success) {
            dividendTracker.distributeTokenDividends(dividends);
            emit SendDividends(tokens, dividends);
        }
    }
    
}

contract SafeBankDividendTracker is DividendPayingToken, Ownable {
    using SafeMath for uint256;
    using SafeMathInt for int256;
    using IterableMapping for IterableMapping.Map;

    IterableMapping.Map private tokenHoldersMap;
    uint256 public lastProcessedIndex;

    mapping (address => bool) public excludedFromDividends;

    mapping (address => uint256) public lastClaimTimes;

    uint256 public claimWait;
    uint256 public minimumTokenBalanceForDividends;

    event ExcludeFromDividends(address indexed account);
    event ClaimWaitUpdated(uint256 indexed newValue, uint256 indexed oldValue);

    event Claim(address indexed account, uint256 amount, bool indexed automatic);

    constructor() public DividendPayingToken("SafeBank_Dividend_Tracker", "SafeBank_Dividend_Tracker") {
        claimWait = 3600;
        minimumTokenBalanceForDividends = 1000000000 * (10**9); //min 1 billion
    }

    // added
    function updateMinimumBalanceForDividends(uint256 newAmountNoDecimials) external onlyOwner{
        minimumTokenBalanceForDividends = newAmountNoDecimials * (10**9);
    }

    function updateTokenForDividend(address newAddress) external onlyOwner {
        DividendToken = newAddress;
    }  
    // 
    function _transfer(address, address, uint256) internal override {
        require(false, "SafeBank_Dividend_Tracker: No transfers allowed");
    }

    function withdrawDividend() public override {
        require(false, "SafeBank_Dividend_Tracker: withdrawDividend disabled. Use the 'claim' function on the main SafeBank contract.");
   
    }

    function updateClaimWait(uint256 newClaimWait) external onlyOwner {
        require(newClaimWait >= 3600 && newClaimWait <= 86400, "SafeBank_Dividend_Tracker: claimWait must be updated to between 1 and 24 hours");
        require(newClaimWait != claimWait, "SafeBank_Dividend_Tracker: Cannot update claimWait to same value");
        emit ClaimWaitUpdated(newClaimWait, claimWait);
        claimWait = newClaimWait;
    }

    function getLastProcessedIndex() external view returns(uint256) {
        return lastProcessedIndex;
    }

    function getNumberOfTokenHolders() external view returns(uint256) {
        return tokenHoldersMap.keys.length;
    }


    function getAccount(address _account)
        public view returns (
            address account,
            int256 index,
            int256 iterationsUntilProcessed,
            uint256 withdrawableDividends,
            uint256 totalDividends,
            uint256 lastClaimTime,
            uint256 nextClaimTime,
            uint256 secondsUntilAutoClaimAvailable) {
        account = _account;

        index = tokenHoldersMap.getIndexOfKey(account);

        iterationsUntilProcessed = -1;

        if(index >= 0) {
            if(uint256(index) > lastProcessedIndex) {
                iterationsUntilProcessed = index.sub(int256(lastProcessedIndex));
            }
            else {
                uint256 processesUntilEndOfArray = tokenHoldersMap.keys.length > lastProcessedIndex ?
                                                        tokenHoldersMap.keys.length.sub(lastProcessedIndex) :
                                                        0;


                iterationsUntilProcessed = index.add(int256(processesUntilEndOfArray));
            }
        }


        withdrawableDividends = withdrawableDividendOf(account);
        totalDividends = accumulativeDividendOf(account);

        lastClaimTime = lastClaimTimes[account];

        nextClaimTime = lastClaimTime > 0 ?
                                    lastClaimTime.add(claimWait) :
                                    0;

        secondsUntilAutoClaimAvailable = nextClaimTime > block.timestamp ?
                                                    nextClaimTime.sub(block.timestamp) :
                                                    0;
    }

    function getAccountAtIndex(uint256 index)
        public view returns (
            address,
            int256,
            int256,
            uint256,
            uint256,
            uint256,
            uint256,
            uint256) {
        if(index >= tokenHoldersMap.size()) {
            return (0x0000000000000000000000000000000000000000, -1, -1, 0, 0, 0, 0, 0);
        }

        address account = tokenHoldersMap.getKeyAtIndex(index);

        return getAccount(account);

     }

    function excludeFromDividends(address account) external onlyOwner {
        require(!excludedFromDividends[account]);
        excludedFromDividends[account] = true;

        _setBalance(account, 0);
        tokenHoldersMap.remove(account);

        //emit ExcludeFromDividends(account);
        
    }

    function canAutoClaim(uint256 lastClaimTime) private view returns (bool) {
        if(lastClaimTime > block.timestamp)  {
            return false;
        }

        return block.timestamp.sub(lastClaimTime) >= claimWait;
    }

    function setBalance(address payable account, uint256 newBalance) external onlyOwner {
        if(excludedFromDividends[account]) {
            return;
        }

        if(newBalance >= minimumTokenBalanceForDividends) {
            _setBalance(account, newBalance);
            tokenHoldersMap.set(account, newBalance);
        }
        else {
            _setBalance(account, 0);
            tokenHoldersMap.remove(account);
        }

        processAccount(account, true);
    }

    function process(uint256 gas) public returns (uint256, uint256, uint256) {
        uint256 numberOfTokenHolders = tokenHoldersMap.keys.length;

        if(numberOfTokenHolders == 0) {
            return (0, 0, lastProcessedIndex);
        }

        uint256 _lastProcessedIndex = lastProcessedIndex;

        uint256 gasUsed = 0;

        uint256 gasLeft = gasleft();

        uint256 iterations = 0;
        uint256 claims = 0;

        while(gasUsed < gas && iterations < numberOfTokenHolders) {
            _lastProcessedIndex++;

            if(_lastProcessedIndex >= tokenHoldersMap.keys.length) {
                _lastProcessedIndex = 0;
            }

            address account = tokenHoldersMap.keys[_lastProcessedIndex];

            if(canAutoClaim(lastClaimTimes[account])) {
                if(processAccount(payable(account), true)) {
                    claims++;
                }
            }

            iterations++;

            uint256 newGasLeft = gasleft();

            if(gasLeft > newGasLeft) {
                gasUsed = gasUsed.add(gasLeft.sub(newGasLeft));
            }

            gasLeft = newGasLeft;
        }

        lastProcessedIndex = _lastProcessedIndex;

        return (iterations, claims, lastProcessedIndex);
    }

    function processAccount(address payable account, bool automatic) public onlyOwner returns (bool) {
        uint256 amount = _withdrawDividendOfUser(account);

        if(amount > 0) {
            lastClaimTimes[account] = block.timestamp;
            emit Claim(account, amount, automatic);
            return true;
        }

        return false;
    }
    
}

Contract Security Audit

Contract ABI

[{"inputs":[],"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":true,"internalType":"address","name":"newBurnWallet","type":"address"},{"indexed":true,"internalType":"address","name":"oldBurnWallet","type":"address"}],"name":"BurnWalletUpdated","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"account","type":"address"},{"indexed":false,"internalType":"bool","name":"isExcluded","type":"bool"}],"name":"ExcludeFromFees","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address[]","name":"accounts","type":"address[]"},{"indexed":false,"internalType":"bool","name":"isExcluded","type":"bool"}],"name":"ExcludeMultipleAccountsFromFees","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"account","type":"address"},{"indexed":false,"internalType":"bool","name":"isExcluded","type":"bool"}],"name":"ExcludedMaxSellTransactionAmount","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"account","type":"address"},{"indexed":true,"internalType":"uint256","name":"amount","type":"uint256"},{"indexed":true,"internalType":"bool","name":"earlyParticipant","type":"bool"},{"indexed":false,"internalType":"uint256","name":"numberOfBuyers","type":"uint256"}],"name":"FixedSaleBuy","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"uint256","name":"newValue","type":"uint256"},{"indexed":true,"internalType":"uint256","name":"oldValue","type":"uint256"}],"name":"GasForProcessingUpdated","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"previousOwner","type":"address"},{"indexed":true,"internalType":"address","name":"newOwner","type":"address"}],"name":"OwnershipTransferred","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"iterations","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"claims","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"lastProcessedIndex","type":"uint256"},{"indexed":true,"internalType":"bool","name":"automatic","type":"bool"},{"indexed":false,"internalType":"uint256","name":"gas","type":"uint256"},{"indexed":true,"internalType":"address","name":"processor","type":"address"}],"name":"ProcessedDividendTracker","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"tokensSwapped","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"amount","type":"uint256"}],"name":"SendDividends","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"pair","type":"address"},{"indexed":true,"internalType":"bool","name":"value","type":"bool"}],"name":"SetAutomatedMarketMakerPair","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256","name":"tokensSwapped","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"ethReceived","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"tokensIntoLiqudity","type":"uint256"}],"name":"SwapAndLiquify","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"from","type":"address"},{"indexed":true,"internalType":"address","name":"to","type":"address"},{"indexed":false,"internalType":"uint256","name":"value","type":"uint256"}],"name":"Transfer","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"newAddress","type":"address"},{"indexed":true,"internalType":"address","name":"oldAddress","type":"address"}],"name":"UpdateDividendToken","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"newAddress","type":"address"},{"indexed":true,"internalType":"address","name":"oldAddress","type":"address"}],"name":"UpdateDividendTracker","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"newAddress","type":"address"},{"indexed":true,"internalType":"address","name":"oldAddress","type":"address"}],"name":"UpdateUniswapV2Router","type":"event"},{"inputs":[],"name":"DividendToken","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"_isExcludedMaxSellTransactionAmount","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"_maxWalletToken","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"tokenAmount","type":"uint256"},{"internalType":"uint256","name":"ethAmount","type":"uint256"}],"name":"addLiquidity","outputs":[],"stateMutability":"nonpayable","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":"amount","type":"uint256"}],"name":"approve","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"automatedMarketMakerPairs","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"account","type":"address"}],"name":"balanceOf","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"burnAddress","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"claim","outputs":[],"stateMutability":"nonpayable","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":"account","type":"address"}],"name":"dividendTokenBalanceOf","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"dividendTracker","outputs":[{"internalType":"contract SafeBankDividendTracker","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"account","type":"address"},{"internalType":"bool","name":"excluded","type":"bool"}],"name":"excludeFromFees","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address[]","name":"accounts","type":"address[]"},{"internalType":"bool","name":"excluded","type":"bool"}],"name":"excludeMultipleAccountsFromFees","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"gasForProcessing","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"account","type":"address"}],"name":"getAccountDividendsInfo","outputs":[{"internalType":"address","name":"","type":"address"},{"internalType":"int256","name":"","type":"int256"},{"internalType":"int256","name":"","type":"int256"},{"internalType":"uint256","name":"","type":"uint256"},{"internalType":"uint256","name":"","type":"uint256"},{"internalType":"uint256","name":"","type":"uint256"},{"internalType":"uint256","name":"","type":"uint256"},{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"index","type":"uint256"}],"name":"getAccountDividendsInfoAtIndex","outputs":[{"internalType":"address","name":"","type":"address"},{"internalType":"int256","name":"","type":"int256"},{"internalType":"int256","name":"","type":"int256"},{"internalType":"uint256","name":"","type":"uint256"},{"internalType":"uint256","name":"","type":"uint256"},{"internalType":"uint256","name":"","type":"uint256"},{"internalType":"uint256","name":"","type":"uint256"},{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"getClaimWait","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"getLastProcessedIndex","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"getNumberOfDividendTokenHolders","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"getTotalDividendsDistributed","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"getTradingIsEnabled","outputs":[{"internalType":"bool","name":"","type":"bool"}],"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":"isExcludedFromFees","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"liquidityFee","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"maxBuyTranscationAmount","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"maxSellTransactionAmount","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"name","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"owner","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"presaleAddress","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"gas","type":"uint256"}],"name":"processDividendTracker","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"renounceOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"sellFeeIncreaseFactor","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"pair","type":"address"},{"internalType":"bool","name":"value","type":"bool"}],"name":"setAutomatedMarketMakerPair","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"tokens","type":"uint256"}],"name":"swapAndLiquify","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"swapTokensAtAmount","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"symbol","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"tokenRewardsFee","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"totalFees","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"totalSupply","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"tradingEnabledTimestamp","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":[],"name":"uniswapV2Pair","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"uniswapV2Router","outputs":[{"internalType":"contract IUniswapV2Router02","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"claimWait","type":"uint256"}],"name":"updateClaimWait","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"newAddress","type":"address"}],"name":"updateDividendTracker","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_tokenRewardsFee","type":"uint256"},{"internalType":"uint256","name":"_liquidityFee","type":"uint256"},{"internalType":"uint256","name":"_wallet1Fee","type":"uint256"},{"internalType":"uint256","name":"_wallet2Fee","type":"uint256"},{"internalType":"uint256","name":"_wallet3Fee","type":"uint256"}],"name":"updateFees","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"newValue","type":"uint256"}],"name":"updateGasForProcessing","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"newAmountNoDecimials","type":"uint256"}],"name":"updateMaxWalletAmount","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"newAmountNoDecimials","type":"uint256"}],"name":"updateMinimumBalanceForDividends","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"newFeeWholeNumber","type":"uint256"}],"name":"updateSellIncreaseFee","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"newAmountNoDecimials","type":"uint256"}],"name":"updateSwapAtAmount","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"newAddress","type":"address"}],"name":"updateTokenForDividend","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"newTimeInEpoch","type":"uint256"}],"name":"updateTradingEnabledTime","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"newAddress","type":"address"}],"name":"updateUniswapV2Router","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address payable","name":"newAddress","type":"address"}],"name":"updateWallet1Address","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"newAddress","type":"address"}],"name":"updateWallet1TokenFeeAddress","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address payable","name":"newAddress","type":"address"}],"name":"updateWallet2Address","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"newAddress","type":"address"}],"name":"updateWallet2TokenFeeAddress","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address payable","name":"newAddress","type":"address"}],"name":"updateWallet3Address","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"newAddress","type":"address"}],"name":"updateWallet3TokenFeeAddress","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"wallet1Address","outputs":[{"internalType":"address payable","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"wallet1Fee","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"wallet1TokenAddressForFee","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"wallet2Address","outputs":[{"internalType":"address payable","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"wallet2Fee","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"wallet2TokenAddressForFee","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"wallet3Address","outputs":[{"internalType":"address payable","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"wallet3Fee","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"wallet3TokenAddressForFee","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_presaleAddress","type":"address"},{"internalType":"address","name":"_routerAddress","type":"address"}],"name":"whitelistDxSale","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"account","type":"address"}],"name":"withdrawableDividendOf","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"stateMutability":"payable","type":"receive"}]

60a0604052600780546001600160a01b031990811690915569152d02c7e14af6800000600a819055600b8190556611c37937e08000600c55600d556104b0601a5562061a80601b55601c805490911690556360f45000601d553480156200006557600080fd5b5060408051808201825260088152675361666542616e6b60c01b6020808301918252835180850190945260058452647342414e4b60d81b908401528151919291620000b39160039162000989565b508051620000c990600490602084019062000989565b50506005805460ff19166009179055506000620000e5620005ef565b60058054610100600160a81b0319166101006001600160a01b03841690810291909117909155604051919250906000907f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e0908290a35060036017819055600260188190556001601481905560158290556016829055818062000191816200017d818187818b84620005f3602090811b6200380e17901c565b620005f360201b6200380e1790919060201c565b601955604051620001a29062000a1e565b604051809103906000f080158015620001bf573d6000803e3d6000fd5b50600880546001600160a01b03929092166001600160a01b03199283161790556009805490911661dead1790556040805163c45a015560e01b81529051737a250d5630b4cf539739df2c5dacb4c659f2488d91600091839163c45a0155916004808301926020929190829003018186803b1580156200023d57600080fd5b505afa15801562000252573d6000803e3d6000fd5b505050506040513d60208110156200026957600080fd5b5051604080516315ab88c960e31b815290516001600160a01b039283169263c9c653969230929187169163ad5c464891600480820192602092909190829003018186803b158015620002ba57600080fd5b505afa158015620002cf573d6000803e3d6000fd5b505050506040513d6020811015620002e657600080fd5b5051604080516001600160e01b031960e086901b1681526001600160a01b0393841660048201529290911660248301525160448083019260209291908290030181600087803b1580156200033957600080fd5b505af11580156200034e573d6000803e3d6000fd5b505050506040513d60208110156200036557600080fd5b5051600680546001600160a01b0319166001600160a01b038516179055606081901b6001600160601b0319166080529050620003a381600162000655565b6008546040805163031e79db60e41b81526001600160a01b0390921660048301819052905190916331e79db091602480830192600092919082900301818387803b158015620003f157600080fd5b505af115801562000406573d6000803e3d6000fd5b50506008546040805163031e79db60e41b815230600482015290516001600160a01b0390921693506331e79db0925060248082019260009290919082900301818387803b1580156200045757600080fd5b505af11580156200046c573d6000803e3d6000fd5b50506008546040805163031e79db60e41b81526001600160a01b03878116600483015291519190921693506331e79db09250602480830192600092919082900301818387803b158015620004bf57600080fd5b505af1158015620004d4573d6000803e3d6000fd5b50506008546001600160a01b031691506331e79db09050620004f562000788565b6040518263ffffffff1660e01b815260040180826001600160a01b03168152602001915050600060405180830381600087803b1580156200053557600080fd5b505af11580156200054a573d6000803e3d6000fd5b50506009546200056892506001600160a01b0316905060016200079c565b620005753060016200079c565b6200058b6200058362000788565b60016200079c565b6001602060006200059b62000788565b6001600160a01b031681526020810191909152604001600020805460ff1916911515919091179055620005e2620005d162000788565b69152d02c7e14af680000062000875565b5050505050505062000a43565b3390565b6000828201838110156200064e576040805162461bcd60e51b815260206004820152601b60248201527f536166654d6174683a206164646974696f6e206f766572666c6f770000000000604482015290519081900360640190fd5b9392505050565b6001600160a01b03821660009081526021602052604090205460ff1615158115151415620006b55760405162461bcd60e51b8152600401808060200182810382526042815260200180620085606042913960600191505060405180910390fd5b6001600160a01b0382166000908152602160205260409020805460ff191682158015919091179091556200074c576008546040805163031e79db60e41b81526001600160a01b038581166004830152915191909216916331e79db091602480830192600092919082900301818387803b1580156200073257600080fd5b505af115801562000747573d6000803e3d6000fd5b505050505b604051811515906001600160a01b038416907fffa9187bf1f18bf477bd0ea1bcbb64e93b6a98132473929edfce215cd9b16fab90600090a35050565b60055461010090046001600160a01b031690565b620007a6620005ef565b6001600160a01b0316620007b962000788565b6001600160a01b03161462000815576040805162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572604482015290519081900360640190fd5b6001600160a01b0382166000818152601e6020908152604091829020805460ff1916851515908117909155825190815291517f9d8f7706ea1113d1a167b526eca956215946dd36cc7df39eb16180222d8b5df79281900390910190a25050565b6001600160a01b038216620008d1576040805162461bcd60e51b815260206004820152601f60248201527f45524332303a206d696e7420746f20746865207a65726f206164647265737300604482015290519081900360640190fd5b620008df6000838362000984565b620008fb81600254620005f360201b6200380e1790919060201c565b6002556001600160a01b038216600090815260208181526040909120546200092e9183906200380e620005f3821b17901c565b6001600160a01b0383166000818152602081815260408083209490945583518581529351929391927fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef9281900390910190a35050565b505050565b828054600181600116156101000203166002900490600052602060002090601f016020900481019282620009c1576000855562000a0c565b82601f10620009dc57805160ff191683800117855562000a0c565b8280016001018555821562000a0c579182015b8281111562000a0c578251825591602001919060010190620009ef565b5062000a1a92915062000a2c565b5090565b61295d8062005c0383390190565b5b8082111562000a1a576000815560010162000a2d565b60805160601c61519862000a6b60003980611c3b5280612c915280613f3352506151986000f3fe60806040526004361061050b5760003560e01c806370a0823111610294578063b13b0e251161015e578063e11dc473116100d6578063e7841ec01161008a578063e98030c71161006f578063e98030c7146110be578063f27fd254146110e8578063f2fde38b1461111257610512565b8063e7841ec014611094578063e7adeccc146110a957610512565b8063e2f45605116100bb578063e2f4560514611055578063e708a0f91461106a578063e76ed0e31461107f57610512565b8063e11dc4731461102b578063e2bac0d81461104057610512565b8063c18bc1951161012d578063c55d228711610112578063c55d228714610fb1578063d651886914610fdb578063dd62ed3e14610ff057610512565b8063c18bc19514610f08578063c492f04614610f3257610512565b8063b13b0e2514610e52578063b62496f514610e67578063b787f82714610e9a578063c024666814610ecd57610512565b806398118cb41161020c578063a457c2d7116101c0578063a8b9d240116101a5578063a8b9d24014610d68578063a9059cbb14610d9b578063ad56c13c14610dd457610512565b8063a457c2d714610d1a578063a6b6388814610d5357610512565b80639c1b8af5116101f15780639c1b8af514610cc05780639cd441da14610cd5578063a26579ad14610d0557610512565b806398118cb414610c705780639a7a23d614610c8557610512565b806378109e541161026357806388bdd9be1161024857806388bdd9be14610c135780638da5cb5b14610c4657806395d89b4114610c5b57610512565b806378109e5414610bd4578063871c128d14610be957610512565b806370a0823114610b4457806370d5ae0514610b77578063715018a614610b8c5780637493c35e14610ba157610512565b806339509351116103d55780634fbee1931161034d57806365b8dbc0116103015780636843cd84116102e65780636843cd8414610ab45780636b83830114610ae7578063700bb19114610b1a57610512565b806365b8dbc014610a6c57806366da2e7814610a9f57610512565b806357967b511161033257806357967b5114610a18578063616c594614610a2d57806364b0f65314610a5757610512565b80634fbee193146109b257806350f41ef4146109e557610512565b8063457ffca2116103a45780634d33b1ef116103895780634d33b1ef1461094d5780634e71d92d146109625780634e83ff6f1461097757610512565b8063457ffca2146108f657806349bd5a5e1461093857610512565b8063395093511461082d5780633a78a994146108665780633b364da8146108995780633d0c8963146108c357610512565b806318160ddd1161048357806330bb4cff116104375780633182f9671161041c5780633182f967146107d057806337b66346146107e5578063387c41671461081857610512565b806330bb4cff14610790578063313ce567146107a557610512565b806323b872dd1161046857806323b872dd1461072357806325b61703146107665780632c1f52161461077b57610512565b806318160ddd146106f957806322bd3f7f1461070e57610512565b8063122fe685116104da578063147de658116104bf578063147de6581461068e5780631694505e146106ba578063173865ad146106cf57610512565b8063122fe6851461064857806313114a9d1461067957610512565b806302259e9e1461051757806306fdde031461053e5780630855f25d146105c8578063095ea7b31461060f57610512565b3661051257005b600080fd5b34801561052357600080fd5b5061052c611145565b60408051918252519081900360200190f35b34801561054a57600080fd5b5061055361114b565b6040805160208082528351818301528351919283929083019185019080838360005b8381101561058d578181015183820152602001610575565b50505050905090810190601f1680156105ba5780820380516001836020036101000a031916815260200191505b509250505060405180910390f35b3480156105d457600080fd5b506105fb600480360360208110156105eb57600080fd5b50356001600160a01b03166111ff565b604080519115158252519081900360200190f35b34801561061b57600080fd5b506105fb6004803603604081101561063257600080fd5b506001600160a01b038135169060200135611214565b34801561065457600080fd5b5061065d611232565b604080516001600160a01b039092168252519081900360200190f35b34801561068557600080fd5b5061052c611241565b34801561069a57600080fd5b506106b8600480360360208110156106b157600080fd5b5035611247565b005b3480156106c657600080fd5b5061065d6112c0565b3480156106db57600080fd5b506106b8600480360360208110156106f257600080fd5b50356112cf565b34801561070557600080fd5b5061052c61163b565b34801561071a57600080fd5b5061052c611641565b34801561072f57600080fd5b506105fb6004803603606081101561074657600080fd5b506001600160a01b03813581169160208101359091169060400135611647565b34801561077257600080fd5b5061052c6116ce565b34801561078757600080fd5b5061065d6116d4565b34801561079c57600080fd5b5061052c6116e3565b3480156107b157600080fd5b506107ba611772565b6040805160ff9092168252519081900360200190f35b3480156107dc57600080fd5b5061052c61177b565b3480156107f157600080fd5b506106b86004803603602081101561080857600080fd5b50356001600160a01b0316611781565b34801561082457600080fd5b5061065d61182f565b34801561083957600080fd5b506105fb6004803603604081101561085057600080fd5b506001600160a01b03813516906020013561183e565b34801561087257600080fd5b506106b86004803603602081101561088957600080fd5b50356001600160a01b031661188c565b3480156108a557600080fd5b506106b8600480360360208110156108bc57600080fd5b503561193a565b3480156108cf57600080fd5b506106b8600480360360208110156108e657600080fd5b50356001600160a01b0316611a2f565b34801561090257600080fd5b506106b8600480360360a081101561091957600080fd5b5080359060208101359060408101359060608101359060800135611b8a565b34801561094457600080fd5b5061065d611c39565b34801561095957600080fd5b506105fb611c5d565b34801561096e57600080fd5b506106b8611c66565b34801561098357600080fd5b506106b86004803603604081101561099a57600080fd5b506001600160a01b0381358116916020013516611d04565b3480156109be57600080fd5b506105fb600480360360208110156109d557600080fd5b50356001600160a01b0316611f22565b3480156109f157600080fd5b506106b860048036036020811015610a0857600080fd5b50356001600160a01b0316611f40565b348015610a2457600080fd5b5061052c612063565b348015610a3957600080fd5b506106b860048036036020811015610a5057600080fd5b5035612069565b348015610a6357600080fd5b5061052c6120e2565b348015610a7857600080fd5b506106b860048036036020811015610a8f57600080fd5b50356001600160a01b0316612140565b348015610aab57600080fd5b5061065d612276565b348015610ac057600080fd5b5061052c60048036036020811015610ad757600080fd5b50356001600160a01b0316612285565b348015610af357600080fd5b506106b860048036036020811015610b0a57600080fd5b50356001600160a01b0316612321565b348015610b2657600080fd5b506106b860048036036020811015610b3d57600080fd5b5035612444565b348015610b5057600080fd5b5061052c60048036036020811015610b6757600080fd5b50356001600160a01b0316612541565b348015610b8357600080fd5b5061065d61255c565b348015610b9857600080fd5b506106b861256b565b348015610bad57600080fd5b506106b860048036036020811015610bc457600080fd5b50356001600160a01b0316612646565b348015610be057600080fd5b5061052c6126f4565b348015610bf557600080fd5b506106b860048036036020811015610c0c57600080fd5b50356126fa565b348015610c1f57600080fd5b506106b860048036036020811015610c3657600080fd5b50356001600160a01b0316612833565b348015610c5257600080fd5b5061065d612b82565b348015610c6757600080fd5b50610553612b96565b348015610c7c57600080fd5b5061052c612c15565b348015610c9157600080fd5b506106b860048036036040811015610ca857600080fd5b506001600160a01b0381351690602001351515612c1b565b348015610ccc57600080fd5b5061052c612d0a565b348015610ce157600080fd5b506106b860048036036040811015610cf857600080fd5b5080359060200135612d10565b348015610d1157600080fd5b5061052c612de7565b348015610d2657600080fd5b506105fb60048036036040811015610d3d57600080fd5b506001600160a01b038135169060200135612e45565b348015610d5f57600080fd5b5061052c612ead565b348015610d7457600080fd5b5061052c60048036036020811015610d8b57600080fd5b50356001600160a01b0316612eb3565b348015610da757600080fd5b506105fb60048036036040811015610dbe57600080fd5b506001600160a01b038135169060200135612f1d565b348015610de057600080fd5b50610e0760048036036020811015610df757600080fd5b50356001600160a01b0316612f31565b604080516001600160a01b0390991689526020890197909752878701959095526060870193909352608086019190915260a085015260c084015260e083015251908190036101000190f35b348015610e5e57600080fd5b5061065d613011565b348015610e7357600080fd5b506105fb60048036036020811015610e8a57600080fd5b50356001600160a01b0316613020565b348015610ea657600080fd5b506106b860048036036020811015610ebd57600080fd5b50356001600160a01b0316613035565b348015610ed957600080fd5b506106b860048036036040811015610ef057600080fd5b506001600160a01b0381351690602001351515613158565b348015610f1457600080fd5b506106b860048036036020811015610f2b57600080fd5b503561324a565b348015610f3e57600080fd5b506106b860048036036040811015610f5557600080fd5b810190602081018135640100000000811115610f7057600080fd5b820183602082011115610f8257600080fd5b80359060200191846020830284011164010000000083111715610fa457600080fd5b91935091503515156132c9565b348015610fbd57600080fd5b506106b860048036036020811015610fd457600080fd5b5035613441565b348015610fe757600080fd5b5061052c6134c0565b348015610ffc57600080fd5b5061052c6004803603604081101561101357600080fd5b506001600160a01b03813581169160200135166134c6565b34801561103757600080fd5b5061065d6134f1565b34801561104c57600080fd5b5061065d613500565b34801561106157600080fd5b5061052c61350f565b34801561107657600080fd5b5061052c613515565b34801561108b57600080fd5b5061065d61351b565b3480156110a057600080fd5b5061052c61352a565b3480156110b557600080fd5b5061065d613588565b3480156110ca57600080fd5b506106b8600480360360208110156110e157600080fd5b5035613597565b3480156110f457600080fd5b50610e076004803603602081101561110b57600080fd5b5035613671565b34801561111e57600080fd5b506106b86004803603602081101561113557600080fd5b50356001600160a01b03166136d7565b600b5481565b60038054604080516020601f60027fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff6101006001881615020190951694909404938401819004810282018101909252828152606093909290918301828280156111f55780601f106111ca576101008083540402835291602001916111f5565b820191906000526020600020905b8154815290600101906020018083116111d857829003601f168201915b5050505050905090565b601f6020526000908152604090205460ff1681565b600061122861122161386f565b8484613873565b5060015b92915050565b601c546001600160a01b031681565b60195481565b61124f61386f565b6001600160a01b0316611260612b82565b6001600160a01b0316146112bb576040805162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572604482015290519081900360640190fd5b601a55565b6006546001600160a01b031681565b60006112dc82600261395f565b905060006112ea83836139c6565b9050476112f683613a23565b600061130247836139c6565b905060006113276014546113216019548561395f90919063ffffffff16565b90613c04565b905060006113466014546113216019548861395f90919063ffffffff16565b905060006113656015546113216019548761395f90919063ffffffff16565b905060006113846015546113216019548a61395f90919063ffffffff16565b905060006113a36016546113216019548961395f90919063ffffffff16565b905060006113c26016546113216019548c61395f90919063ffffffff16565b90508584018201858401820160006113da8a846139c6565b905060006113e88d846139c6565b6011549091506001600160a01b03161561141f57601154600e5461141a918c916001600160a01b039182169116613c5d565b611492565b600e5460405160009182916001600160a01b03909116908d908381818185875af1925050503d8060008114611470576040519150601f19603f3d011682016040523d82523d6000602084013e611475565b606091505b509150915081156114855761148f565b61148f8b8d612d10565b50505b6012546001600160a01b0316156114c657601254600f546114c1918a916001600160a01b039182169116613c5d565b611539565b600f5460405160009182916001600160a01b03909116908b908381818185875af1925050503d8060008114611517576040519150601f19603f3d011682016040523d82523d6000602084013e61151c565b606091505b5091509150811561152c57611536565b611536898b612d10565b50505b6013546001600160a01b03161561156d576013546010546115689188916001600160a01b039182169116613c5d565b6115e0565b60105460405160009182916001600160a01b039091169089908381818185875af1925050503d80600081146115be576040519150601f19603f3d011682016040523d82523d6000602084013e6115c3565b606091505b509150915081156115d3576115dd565b6115dd8789612d10565b50505b6115ea8183612d10565b604080518f8152602081018d90528082018f905290517f17bbfb9a6069321b6ded73bd96327c9e6b7212a5cd51ff219cd61370acafb5619181900360600190a1505050505050505050505050505050565b60025490565b601a5481565b6000611654848484613e33565b6116c48461166061386f565b6116bf85604051806060016040528060288152602001614fcf602891396001600160a01b038a1660009081526001602052604081209061169e61386f565b6001600160a01b0316815260208101919091526040016000205491906145a2565b613873565b5060019392505050565b601d5481565b6008546001600160a01b031681565b600854604080517f85a6b3ae00000000000000000000000000000000000000000000000000000000815290516000926001600160a01b0316916385a6b3ae916004808301926020929190829003018186803b15801561174157600080fd5b505afa158015611755573d6000803e3d6000fd5b505050506040513d602081101561176b57600080fd5b5051905090565b60055460ff1690565b60155481565b61178961386f565b6001600160a01b031661179a612b82565b6001600160a01b0316146117f5576040805162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572604482015290519081900360640190fd5b601180547fffffffffffffffffffffffff0000000000000000000000000000000000000000166001600160a01b0392909216919091179055565b600e546001600160a01b031681565b600061122861184b61386f565b846116bf856001600061185c61386f565b6001600160a01b03908116825260208083019390935260409182016000908120918c16815292529020549061380e565b61189461386f565b6001600160a01b03166118a5612b82565b6001600160a01b031614611900576040805162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572604482015290519081900360640190fd5b601280547fffffffffffffffffffffffff0000000000000000000000000000000000000000166001600160a01b0392909216919091179055565b61194261386f565b6001600160a01b0316611953612b82565b6001600160a01b0316146119ae576040805162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572604482015290519081900360640190fd5b600854604080517f3b364da80000000000000000000000000000000000000000000000000000000081526004810184905290516001600160a01b0390921691633b364da89160248082019260009290919082900301818387803b158015611a1457600080fd5b505af1158015611a28573d6000803e3d6000fd5b5050505050565b611a3761386f565b6001600160a01b0316611a48612b82565b6001600160a01b031614611aa3576040805162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572604482015290519081900360640190fd5b600854604080517f3d0c89630000000000000000000000000000000000000000000000000000000081526001600160a01b03848116600483015291519190921691633d0c896391602480830192600092919082900301818387803b158015611b0a57600080fd5b505af1158015611b1e573d6000803e3d6000fd5b5050600780547fffffffffffffffffffffffff0000000000000000000000000000000000000000166001600160a01b03858116918217928390556040519216935091507fc50953975bfb7dc05da466bc18236bf737c11097c9cc7ef9d079bc7ed96f03ae90600090a350565b611b9261386f565b6001600160a01b0316611ba3612b82565b6001600160a01b031614611bfe576040805162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572604482015290519081900360640190fd5b60178590556018849055601483905560158290556016819055611c2f81611c29848187818b8b61380e565b9061380e565b6019555050505050565b7f000000000000000000000000000000000000000000000000000000000000000081565b601d5442101590565b600854604080517fbc4c4b3700000000000000000000000000000000000000000000000000000000815233600482015260006024820181905291516001600160a01b039093169263bc4c4b3792604480840193602093929083900390910190829087803b158015611cd657600080fd5b505af1158015611cea573d6000803e3d6000fd5b505050506040513d6020811015611d0057600080fd5b5050565b611d0c61386f565b6001600160a01b0316611d1d612b82565b6001600160a01b031614611d78576040805162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572604482015290519081900360640190fd5b601c80547fffffffffffffffffffffffff0000000000000000000000000000000000000000166001600160a01b0384811691821792839055918216600090815260208052604080822080547fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff0016600117905560085481517f31e79db0000000000000000000000000000000000000000000000000000000008152600481019490945290519316926331e79db092602480820193929182900301818387803b158015611e4257600080fd5b505af1158015611e56573d6000803e3d6000fd5b50505050611e65826001613158565b6001600160a01b03808216600081815260208052604080822080547fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff0016600117905560085481517f31e79db0000000000000000000000000000000000000000000000000000000008152600481019490945290519316926331e79db092602480820193929182900301818387803b158015611eff57600080fd5b505af1158015611f13573d6000803e3d6000fd5b50505050611d00816001613158565b6001600160a01b03166000908152601e602052604090205460ff1690565b611f4861386f565b6001600160a01b0316611f59612b82565b6001600160a01b031614611fb4576040805162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572604482015290519081900360640190fd5b600f80547fffffffffffffffffffffffff0000000000000000000000000000000000000000166001600160a01b038381169190911791829055611ff991166001613158565b600854600f54604080517f31e79db00000000000000000000000000000000000000000000000000000000081526001600160a01b039283166004820152905191909216916331e79db091602480830192600092919082900301818387803b158015611a1457600080fd5b60145481565b61207161386f565b6001600160a01b0316612082612b82565b6001600160a01b0316146120dd576040805162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572604482015290519081900360640190fd5b601d55565b600854604080517f09bbedde00000000000000000000000000000000000000000000000000000000815290516000926001600160a01b0316916309bbedde916004808301926020929190829003018186803b15801561174157600080fd5b61214861386f565b6001600160a01b0316612159612b82565b6001600160a01b0316146121b4576040805162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572604482015290519081900360640190fd5b6006546001600160a01b03828116911614156122015760405162461bcd60e51b815260040180806020018281038252602d815260200180614ee5602d913960400191505060405180910390fd5b6006546040516001600160a01b03918216918316907f8fc842bbd331dfa973645f4ed48b11683d501ebf1352708d77a5da2ab49a576e90600090a3600680547fffffffffffffffffffffffff0000000000000000000000000000000000000000166001600160a01b0392909216919091179055565b6012546001600160a01b031681565b600854604080517f70a082310000000000000000000000000000000000000000000000000000000081526001600160a01b038481166004830152915160009392909216916370a0823191602480820192602092909190829003018186803b1580156122ef57600080fd5b505afa158015612303573d6000803e3d6000fd5b505050506040513d602081101561231957600080fd5b505192915050565b61232961386f565b6001600160a01b031661233a612b82565b6001600160a01b031614612395576040805162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572604482015290519081900360640190fd5b600e80547fffffffffffffffffffffffff0000000000000000000000000000000000000000166001600160a01b0383811691909117918290556123da91166001613158565b600854600e54604080517f31e79db00000000000000000000000000000000000000000000000000000000081526001600160a01b039283166004820152905191909216916331e79db091602480830192600092919082900301818387803b158015611a1457600080fd5b600854604080517fffb2c479000000000000000000000000000000000000000000000000000000008152600481018490529051600092839283926001600160a01b039092169163ffb2c4799160248082019260609290919082900301818787803b1580156124b157600080fd5b505af11580156124c5573d6000803e3d6000fd5b505050506040513d60608110156124db57600080fd5b5080516020808301516040938401518451848152928301829052828501819052606083018990529351929650945091925032916000917fc864333d6121033635ab41b29ae52f10a22cf4438c3e4f1c4c68518feb2f8a989181900360800190a350505050565b6001600160a01b031660009081526020819052604090205490565b6009546001600160a01b031681565b61257361386f565b6001600160a01b0316612584612b82565b6001600160a01b0316146125df576040805162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572604482015290519081900360640190fd5b60055460405160009161010090046001600160a01b0316907f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e0908390a3600580547fffffffffffffffffffffff0000000000000000000000000000000000000000ff169055565b61264e61386f565b6001600160a01b031661265f612b82565b6001600160a01b0316146126ba576040805162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572604482015290519081900360640190fd5b601380547fffffffffffffffffffffffff0000000000000000000000000000000000000000166001600160a01b0392909216919091179055565b600d5481565b61270261386f565b6001600160a01b0316612713612b82565b6001600160a01b03161461276e576040805162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572604482015290519081900360640190fd5b62030d40811015801561278457506207a1208111155b6127bf5760405162461bcd60e51b815260040180806020018281038252603e81526020018061508d603e913960400191505060405180910390fd5b601b548114156128005760405162461bcd60e51b8152600401808060200182810382526036815260200180614f786036913960400191505060405180910390fd5b601b5460405182907f40d7e40e79af4e8e5a9b3c57030d8ea93f13d669c06d448c4d631d4ae7d23db790600090a3601b55565b61283b61386f565b6001600160a01b031661284c612b82565b6001600160a01b0316146128a7576040805162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572604482015290519081900360640190fd5b6008546001600160a01b03828116911614156128f45760405162461bcd60e51b8152600401808060200182810382526037815260200180614ff76037913960400191505060405180910390fd5b6000819050306001600160a01b0316816001600160a01b0316638da5cb5b6040518163ffffffff1660e01b815260040160206040518083038186803b15801561293c57600080fd5b505afa158015612950573d6000803e3d6000fd5b505050506040513d602081101561296657600080fd5b50516001600160a01b0316146129ad5760405162461bcd60e51b815260040180806020018281038252604f8152602001806150ef604f913960600191505060405180910390fd5b806001600160a01b03166331e79db0826040518263ffffffff1660e01b815260040180826001600160a01b03168152602001915050600060405180830381600087803b1580156129fc57600080fd5b505af1158015612a10573d6000803e3d6000fd5b5050604080517f31e79db000000000000000000000000000000000000000000000000000000000815230600482015290516001600160a01b03851693506331e79db09250602480830192600092919082900301818387803b158015612a7457600080fd5b505af1158015612a88573d6000803e3d6000fd5b5050600654604080517f31e79db00000000000000000000000000000000000000000000000000000000081526001600160a01b039283166004820152905191851693506331e79db0925060248082019260009290919082900301818387803b158015612af357600080fd5b505af1158015612b07573d6000803e3d6000fd5b50506008546040516001600160a01b03918216935090851691507f90c7d74461c613da5efa97d90740869367d74ab3aa5837aa4ae9a975f954b7a890600090a3600880547fffffffffffffffffffffffff0000000000000000000000000000000000000000166001600160a01b039290921691909117905550565b60055461010090046001600160a01b031690565b60048054604080516020601f60027fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff6101006001881615020190951694909404938401819004810282018101909252828152606093909290918301828280156111f55780601f106111ca576101008083540402835291602001916111f5565b60185481565b612c2361386f565b6001600160a01b0316612c34612b82565b6001600160a01b031614612c8f576040805162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572604482015290519081900360640190fd5b7f00000000000000000000000000000000000000000000000000000000000000006001600160a01b0316826001600160a01b03161415612d005760405162461bcd60e51b815260040180806020018281038252604f815260200180614e06604f913960600191505060405180910390fd5b611d008282614639565b601b5481565b600654612d289030906001600160a01b031684613873565b600654600954604080517ff305d7190000000000000000000000000000000000000000000000000000000081523060048201526024810186905260006044820181905260648201526001600160a01b0392831660848201524260a48201529051919092169163f305d71991849160c48082019260609290919082900301818588803b158015612db657600080fd5b505af1158015612dca573d6000803e3d6000fd5b50505050506040513d6060811015612de157600080fd5b50505050565b600854604080517f6f2789ec00000000000000000000000000000000000000000000000000000000815290516000926001600160a01b031691636f2789ec916004808301926020929190829003018186803b15801561174157600080fd5b6000611228612e5261386f565b846116bf8560405180606001604052806025815260200161513e6025913960016000612e7c61386f565b6001600160a01b03908116825260208083019390935260409182016000908120918d168152925290205491906145a2565b600a5481565b600854604080517fa8b9d2400000000000000000000000000000000000000000000000000000000081526001600160a01b0384811660048301529151600093929092169163a8b9d24091602480820192602092909190829003018186803b1580156122ef57600080fd5b6000611228612f2a61386f565b8484613e33565b600080600080600080600080600860009054906101000a90046001600160a01b03166001600160a01b031663fbcbc0f18a6040518263ffffffff1660e01b815260040180826001600160a01b031681526020019150506101006040518083038186803b158015612fa057600080fd5b505afa158015612fb4573d6000803e3d6000fd5b505050506040513d610100811015612fcb57600080fd5b508051602082015160408301516060840151608085015160a086015160c087015160e090970151959e50939c50919a509850965094509092509050919395975091939597565b6013546001600160a01b031681565b60216020526000908152604090205460ff1681565b61303d61386f565b6001600160a01b031661304e612b82565b6001600160a01b0316146130a9576040805162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572604482015290519081900360640190fd5b601080547fffffffffffffffffffffffff0000000000000000000000000000000000000000166001600160a01b0383811691909117918290556130ee91166001613158565b600854601054604080517f31e79db00000000000000000000000000000000000000000000000000000000081526001600160a01b039283166004820152905191909216916331e79db091602480830192600092919082900301818387803b158015611a1457600080fd5b61316061386f565b6001600160a01b0316613171612b82565b6001600160a01b0316146131cc576040805162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572604482015290519081900360640190fd5b6001600160a01b0382166000818152601e602090815260409182902080547fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff0016851515908117909155825190815291517f9d8f7706ea1113d1a167b526eca956215946dd36cc7df39eb16180222d8b5df79281900390910190a25050565b61325261386f565b6001600160a01b0316613263612b82565b6001600160a01b0316146132be576040805162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572604482015290519081900360640190fd5b633b9aca0002600d55565b6132d161386f565b6001600160a01b03166132e2612b82565b6001600160a01b03161461333d576040805162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572604482015290519081900360640190fd5b60005b828110156133b05781601e600086868581811061335957fe5b602090810292909201356001600160a01b031683525081019190915260400160002080547fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff0016911515919091179055600101613340565b507f7fdaf542373fa84f4ee8d662c642f44e4c2276a217d7d29e548b6eb29a233b35838383604051808060200183151581526020018281038252858582818152602001925060200280828437600083820152604051601f9091017fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe016909201829003965090945050505050a1505050565b61344961386f565b6001600160a01b031661345a612b82565b6001600160a01b0316146134b5576040805162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572604482015290519081900360640190fd5b633b9aca0002600c55565b60165481565b6001600160a01b03918216600090815260016020908152604080832093909416825291909152205490565b600f546001600160a01b031681565b6011546001600160a01b031681565b600c5481565b60175481565b6007546001600160a01b031681565b600854604080517fe7841ec000000000000000000000000000000000000000000000000000000000815290516000926001600160a01b03169163e7841ec0916004808301926020929190829003018186803b15801561174157600080fd5b6010546001600160a01b031681565b61359f61386f565b6001600160a01b03166135b0612b82565b6001600160a01b03161461360b576040805162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572604482015290519081900360640190fd5b600854604080517fe98030c70000000000000000000000000000000000000000000000000000000081526004810184905290516001600160a01b039092169163e98030c79160248082019260009290919082900301818387803b158015611a1457600080fd5b600080600080600080600080600860009054906101000a90046001600160a01b03166001600160a01b0316635183d6fd8a6040518263ffffffff1660e01b8152600401808281526020019150506101006040518083038186803b158015612fa057600080fd5b6136df61386f565b6001600160a01b03166136f0612b82565b6001600160a01b03161461374b576040805162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572604482015290519081900360640190fd5b6001600160a01b0381166137905760405162461bcd60e51b8152600401808060200182810382526026815260200180614dbe6026913960400191505060405180910390fd5b6005546040516001600160a01b0380841692610100900416907f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e090600090a3600580546001600160a01b03909216610100027fffffffffffffffffffffff0000000000000000000000000000000000000000ff909216919091179055565b600082820183811015613868576040805162461bcd60e51b815260206004820152601b60248201527f536166654d6174683a206164646974696f6e206f766572666c6f770000000000604482015290519081900360640190fd5b9392505050565b3390565b6001600160a01b0383166138b85760405162461bcd60e51b81526004018080602001828103825260248152602001806150cb6024913960400191505060405180910390fd5b6001600160a01b0382166138fd5760405162461bcd60e51b8152600401808060200182810382526022815260200180614de46022913960400191505060405180910390fd5b6001600160a01b03808416600081815260016020908152604080832094871680845294825291829020859055815185815291517f8c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b9259281900390910190a3505050565b60008082116139b5576040805162461bcd60e51b815260206004820152601a60248201527f536166654d6174683a206469766973696f6e206279207a65726f000000000000604482015290519081900360640190fd5b8183816139be57fe5b049392505050565b600082821115613a1d576040805162461bcd60e51b815260206004820152601e60248201527f536166654d6174683a207375627472616374696f6e206f766572666c6f770000604482015290519081900360640190fd5b50900390565b6040805160028082526060820183526000926020830190803683370190505090503081600081518110613a5257fe5b6001600160a01b03928316602091820292909201810191909152600654604080517fad5c46480000000000000000000000000000000000000000000000000000000081529051919093169263ad5c4648926004808301939192829003018186803b158015613abf57600080fd5b505afa158015613ad3573d6000803e3d6000fd5b505050506040513d6020811015613ae957600080fd5b5051815182906001908110613afa57fe5b6001600160a01b039283166020918202929092010152600654613b209130911684613873565b6006546040517f791ac947000000000000000000000000000000000000000000000000000000008152600481018481526000602483018190523060648401819052426084850181905260a060448601908152875160a487015287516001600160a01b039097169663791ac947968a968a9594939092909160c40190602080880191028083838b5b83811015613bbf578181015183820152602001613ba7565b505050509050019650505050505050600060405180830381600087803b158015613be857600080fd5b505af1158015613bfc573d6000803e3d6000fd5b505050505050565b600082613c135750600061122c565b82820282848281613c2057fe5b04146138685760405162461bcd60e51b8152600401808060200182810382526021815260200180614fae6021913960400191505060405180910390fd5b6040805160028082526060820183526000926020830190803683375050600654604080517fad5c464800000000000000000000000000000000000000000000000000000000815290519394506001600160a01b039091169263ad5c464892506004808301926020929190829003018186803b158015613cdb57600080fd5b505afa158015613cef573d6000803e3d6000fd5b505050506040513d6020811015613d0557600080fd5b505181518290600090613d1457fe5b60200260200101906001600160a01b031690816001600160a01b0316815250508281600181518110613d4257fe5b6001600160a01b039283166020918202929092018101919091526006546040517fb6f9de9500000000000000000000000000000000000000000000000000000000815260006004820181815287861660448401524260648401819052608060248501908152885160848601528851959097169663b6f9de95968c9694958a958c95939260a49092019187810191028083838b5b83811015613ded578181015183820152602001613dd5565b50505050905001955050505050506000604051808303818588803b158015613e1457600080fd5b505af1158015613e28573d6000803e3d6000fd5b505050505050505050565b6001600160a01b038316613e785760405162461bcd60e51b815260040180806020018281038252602581526020018061502e6025913960400191505060405180910390fd5b6001600160a01b038216613ebd5760405162461bcd60e51b8152600401808060200182810382526023815260200180614d9b6023913960400191505060405180910390fd5b613ec5612b82565b6001600160a01b0316836001600160a01b031614158015613eff5750613ee9612b82565b6001600160a01b0316826001600160a01b031614155b8015613f1357506001600160a01b03821615155b8015613f2a57506001600160a01b03821661dead14155b8015613f6857507f00000000000000000000000000000000000000000000000000000000000000006001600160a01b0316826001600160a01b031614155b1561400057600a54811115613fae5760405162461bcd60e51b8152600401808060200182810382526028815260200180614ebd6028913960400191505060405180910390fd5b6000613fb983612541565b9050600d548282011115613ffe5760405162461bcd60e51b8152600401808060200182810382526024815260200180614f126024913960400191505060405180910390fd5b505b600061400a611c5d565b905080614067576001600160a01b038416600090815260208052604090205460ff166140675760405162461bcd60e51b8152600401808060200182810382526042815260200180614e7b6042913960600191505060405180910390fd5b8161407e576140788484600061479e565b5061459d565b60075474010000000000000000000000000000000000000000900460ff161580156140a65750805b80156140ca57506001600160a01b03831660009081526021602052604090205460ff165b80156140e457506006546001600160a01b03858116911614155b801561410957506001600160a01b0383166000908152601e602052604090205460ff16155b1561414f57600b5482111561414f5760405162461bcd60e51b815260040180806020018281038252603a815260200180615053603a913960400191505060405180910390fd5b600061415a30612541565b600c5490915081101582801561416d5750805b8015614194575060075474010000000000000000000000000000000000000000900460ff16155b80156141b957506001600160a01b03861660009081526021602052604090205460ff16155b80156141d357506009546001600160a01b03878116911614155b80156141ed57506009546001600160a01b03868116911614155b1561429b57600780547fffffffffffffffffffffff00ffffffffffffffffffffffffffffffffffffffff167401000000000000000000000000000000000000000017905560195460185460009161424f91614249908690613c04565b9061395f565b905061425a816112cf565b600061426530612541565b9050614270816148f9565b5050600780547fffffffffffffffffffffff00ffffffffffffffffffffffffffffffffffffffff1690555b60008380156142c5575060075474010000000000000000000000000000000000000000900460ff16155b6001600160a01b0388166000908152601e602052604090205490915060ff168061430757506001600160a01b0386166000908152601e602052604090205460ff165b15614310575060005b801561438c576000614332606461424960195489613c0490919063ffffffff16565b6001600160a01b03881660009081526021602052604090205490915060ff1615614373576143706064614249601a5484613c0490919063ffffffff16565b90505b61437d86826139c6565b955061438a88308361479e565b505b61439787878761479e565b6008546001600160a01b031663e30443bc886143b281612541565b6040518363ffffffff1660e01b815260040180836001600160a01b0316815260200182815260200192505050600060405180830381600087803b1580156143f857600080fd5b505af1925050508015614409575060015b506008546001600160a01b031663e30443bc8761442581612541565b6040518363ffffffff1660e01b815260040180836001600160a01b0316815260200182815260200192505050600060405180830381600087803b15801561446b57600080fd5b505af192505050801561447c575060015b5060075474010000000000000000000000000000000000000000900460ff1661459857601b54600854604080517fffb2c4790000000000000000000000000000000000000000000000000000000081526004810184905290516001600160a01b039092169163ffb2c479916024808201926060929091908290030181600087803b15801561450957600080fd5b505af192505050801561453d57506040513d606081101561452957600080fd5b508051602082015160409092015190919060015b61454657614596565b604080518481526020810184905280820183905260608101869052905132916001917fc864333d6121033635ab41b29ae52f10a22cf4438c3e4f1c4c68518feb2f8a989181900360800190a35050505b505b505050505b505050565b600081848411156146315760405162461bcd60e51b81526004018080602001828103825283818151815260200191508051906020019080838360005b838110156145f65781810151838201526020016145de565b50505050905090810190601f1680156146235780820380516001836020036101000a031916815260200191505b509250505060405180910390fd5b505050900390565b6001600160a01b03821660009081526021602052604090205460ff16151581151514156146975760405162461bcd60e51b8152600401808060200182810382526042815260200180614f366042913960600191505060405180910390fd5b6001600160a01b038216600090815260216020526040902080547fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff0016821580159190911790915561476257600854604080517f31e79db00000000000000000000000000000000000000000000000000000000081526001600160a01b038581166004830152915191909216916331e79db091602480830192600092919082900301818387803b15801561474957600080fd5b505af115801561475d573d6000803e3d6000fd5b505050505b604051811515906001600160a01b038416907fffa9187bf1f18bf477bd0ea1bcbb64e93b6a98132473929edfce215cd9b16fab90600090a35050565b6001600160a01b0383166147e35760405162461bcd60e51b815260040180806020018281038252602581526020018061502e6025913960400191505060405180910390fd5b6001600160a01b0382166148285760405162461bcd60e51b8152600401808060200182810382526023815260200180614d9b6023913960400191505060405180910390fd5b61483383838361459d565b61487081604051806060016040528060268152602001614e55602691396001600160a01b03861660009081526020819052604090205491906145a2565b6001600160a01b03808516600090815260208190526040808220939093559084168152205461489f908261380e565b6001600160a01b038084166000818152602081815260409182902094909455805185815290519193928716927fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef92918290030190a3505050565b6008546007546001600160a01b039182169160009182911615614a57576149208430614b94565b600754604080517f70a0823100000000000000000000000000000000000000000000000000000000815230600482015290516001600160a01b03909216916370a0823191602480820192602092909190829003018186803b15801561498457600080fd5b505afa158015614998573d6000803e3d6000fd5b505050506040513d60208110156149ae57600080fd5b5051600754600854604080517fa9059cbb0000000000000000000000000000000000000000000000000000000081526001600160a01b03928316600482015260248101859052905193945091169163a9059cbb916044808201926020929091908290030181600087803b158015614a2457600080fd5b505af1158015614a38573d6000803e3d6000fd5b505050506040513d6020811015614a4e57600080fd5b50519150614acf565b47614a6185613a23565b6000614a6d47836139c6565b60405190935083915060009081906001600160a01b0388169084908381818185875af1925050503d8060008114614ac0576040519150601f19603f3d011682016040523d82523d6000602084013e614ac5565b606091505b5090965050505050505b8115612de157600854604080517fb0c7ce370000000000000000000000000000000000000000000000000000000081526004810184905290516001600160a01b039092169163b0c7ce379160248082019260009290919082900301818387803b158015614b3b57600080fd5b505af1158015614b4f573d6000803e3d6000fd5b5050604080518781526020810185905281517f80195cc573b02cc48460cbca6e6e4cc85ddb91959d946e1c3025ea3d87942dc39450908190039091019150a150505050565b60408051600380825260808201909252600091602082016060803683370190505090503081600081518110614bc557fe5b6001600160a01b03928316602091820292909201810191909152600654604080517fad5c46480000000000000000000000000000000000000000000000000000000081529051919093169263ad5c4648926004808301939192829003018186803b158015614c3257600080fd5b505afa158015614c46573d6000803e3d6000fd5b505050506040513d6020811015614c5c57600080fd5b5051815182906001908110614c6d57fe5b6001600160a01b039283166020918202929092010152600754825191169082906002908110614c9857fe5b6001600160a01b039283166020918202929092010152600654614cbe9130911685613873565b6006546040517f5c11d795000000000000000000000000000000000000000000000000000000008152600481018581526000602483018190526001600160a01b038681166064850152426084850181905260a060448601908152875160a487015287519290961695635c11d795958a9589948b9493919260c40190602080880191028083838b5b83811015614d5d578181015183820152602001614d45565b505050509050019650505050505050600060405180830381600087803b158015614d8657600080fd5b505af1158015614598573d6000803e3d6000fdfe45524332303a207472616e7366657220746f20746865207a65726f20616464726573734f776e61626c653a206e6577206f776e657220697320746865207a65726f206164647265737345524332303a20617070726f766520746f20746865207a65726f20616464726573735361666542616e6b3a205468652050616e63616b655377617020706169722063616e6e6f742062652072656d6f7665642066726f6d206175746f6d617465644d61726b65744d616b6572506169727345524332303a207472616e7366657220616d6f756e7420657863656564732062616c616e63655361666542616e6b3a2054686973206163636f756e742063616e6e6f742073656e6420746f6b656e7320756e74696c2074726164696e6720697320656e61626c65645472616e7366657220616d6f756e74206578636565647320746865206d61785478416d6f756e742e5361666542616e6b3a2054686520726f7574657220616c7265616479206861732074686174206164647265737345786365656473206d6178696d756d2077616c6c657420746f6b656e20616d6f756e742e5361666542616e6b3a204175746f6d61746564206d61726b6574206d616b6572207061697220697320616c72656164792073657420746f20746861742076616c75655361666542616e6b3a2043616e6e6f742075706461746520676173466f7250726f63657373696e6720746f2073616d652076616c7565536166654d6174683a206d756c7469706c69636174696f6e206f766572666c6f7745524332303a207472616e7366657220616d6f756e74206578636565647320616c6c6f77616e63655361666542616e6b3a20546865206469766964656e6420747261636b657220616c7265616479206861732074686174206164647265737345524332303a207472616e736665722066726f6d20746865207a65726f206164647265737353656c6c207472616e7366657220616d6f756e74206578636565647320746865206d617853656c6c5472616e73616374696f6e416d6f756e742e5361666542616e6b3a20676173466f7250726f63657373696e67206d757374206265206265747765656e203230302c30303020616e64203530302c30303045524332303a20617070726f76652066726f6d20746865207a65726f20616464726573735361666542616e6b3a20546865206e6577206469766964656e6420747261636b6572206d757374206265206f776e656420627920746865205361666542616e6b20746f6b656e20636f6e747261637445524332303a2064656372656173656420616c6c6f77616e63652062656c6f77207a65726fa26469706673582212203f8ef2f8e553101242b331abbed1e7f272164dcd78b04de9121c5e919e618ed964736f6c634300070600336080604052600880546001600160a01b03191690553480156200002157600080fd5b5060408051808201825260198082527f5361666542616e6b5f4469766964656e645f547261636b657200000000000000602080840182815285518087019096529285528401528151919291839183916200007e916003916200011e565b508051620000949060049060208401906200011e565b50506005805460ff191660091790555060009150620000b490506200011a565b600c80546001600160a01b0319166001600160a01b038316908117909155604051919250906000907f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e0908290a350610e10601455670de0b6b3a7640000601555620001ca565b3390565b828054600181600116156101000203166002900490600052602060002090601f016020900481019282620001565760008555620001a1565b82601f106200017157805160ff1916838001178555620001a1565b82800160010185558215620001a1579182015b82811115620001a157825182559160200191906001019062000184565b50620001af929150620001b3565b5090565b5b80821115620001af5760008155600101620001b4565b61278380620001da6000396000f3fe6080604052600436106102a45760003560e01c8063715018a61161016e578063bc4c4b37116100cb578063e7841ec01161007f578063f2fde38b11610064578063f2fde38b14610941578063fbcbc0f114610974578063ffb2c479146109a7576102ab565b8063e7841ec014610902578063e98030c714610917576102ab565b8063dd62ed3e116100b0578063dd62ed3e14610879578063e30443bc146108b4578063e76ed0e3146108ed576102ab565b8063bc4c4b3714610829578063be10b61414610864576102ab565b8063a457c2d711610122578063a9059cbb11610107578063a9059cbb14610793578063aafd847a146107cc578063b0c7ce37146107ff576102ab565b8063a457c2d714610727578063a8b9d24014610760576102ab565b80638da5cb5b116101535780638da5cb5b146106ae57806391b89fba146106df57806395d89b4114610712576102ab565b8063715018a61461068457806385a6b3ae14610699576102ab565b8063313ce5671161021c5780634e7b827f116101d05780636a474002116101b55780636a474002146106275780636f2789ec1461063c57806370a0823114610651576102ab565b80634e7b827f1461057f5780635183d6fd146105b2576102ab565b8063395093511161020157806339509351146104e95780633b364da8146105225780633d0c89631461054c576102ab565b8063313ce5671461048b57806331e79db0146104b6576102ab565b806318160ddd1161027357806323b872dd1161025857806323b872dd1461040057806327ce0147146104435780633009a60914610476576102ab565b806318160ddd146103b8578063226cfa3d146103cd576102ab565b806303c83302146102b057806306fdde03146102ba578063095ea7b31461034457806309bbedde14610391576102ab565b366102ab57005b600080fd5b6102b86109ef565b005b3480156102c657600080fd5b506102cf610a8d565b6040805160208082528351818301528351919283929083019185019080838360005b838110156103095781810151838201526020016102f1565b50505050905090810190601f1680156103365780820380516001836020036101000a031916815260200191505b509250505060405180910390f35b34801561035057600080fd5b5061037d6004803603604081101561036757600080fd5b506001600160a01b038135169060200135610b41565b604080519115158252519081900360200190f35b34801561039d57600080fd5b506103a6610b5f565b60408051918252519081900360200190f35b3480156103c457600080fd5b506103a6610b65565b3480156103d957600080fd5b506103a6600480360360208110156103f057600080fd5b50356001600160a01b0316610b6b565b34801561040c57600080fd5b5061037d6004803603606081101561042357600080fd5b506001600160a01b03813581169160208101359091169060400135610b7d565b34801561044f57600080fd5b506103a66004803603602081101561046657600080fd5b50356001600160a01b0316610c04565b34801561048257600080fd5b506103a6610c70565b34801561049757600080fd5b506104a0610c76565b6040805160ff9092168252519081900360200190f35b3480156104c257600080fd5b506102b8600480360360208110156104d957600080fd5b50356001600160a01b0316610c7f565b3480156104f557600080fd5b5061037d6004803603604081101561050c57600080fd5b506001600160a01b038135169060200135610dfa565b34801561052e57600080fd5b506102b86004803603602081101561054557600080fd5b5035610e48565b34801561055857600080fd5b506102b86004803603602081101561056f57600080fd5b50356001600160a01b0316610ec7565b34801561058b57600080fd5b5061037d600480360360208110156105a257600080fd5b50356001600160a01b0316610f75565b3480156105be57600080fd5b506105dc600480360360208110156105d557600080fd5b5035610f8a565b604080516001600160a01b0390991689526020890197909752878701959095526060870193909352608086019190915260a085015260c084015260e083015251908190036101000190f35b34801561063357600080fd5b506102b8611107565b34801561064857600080fd5b506103a661113e565b34801561065d57600080fd5b506103a66004803603602081101561067457600080fd5b50356001600160a01b0316611144565b34801561069057600080fd5b506102b861115f565b3480156106a557600080fd5b506103a6611235565b3480156106ba57600080fd5b506106c361123b565b604080516001600160a01b039092168252519081900360200190f35b3480156106eb57600080fd5b506103a66004803603602081101561070257600080fd5b50356001600160a01b031661124a565b34801561071e57600080fd5b506102cf611255565b34801561073357600080fd5b5061037d6004803603604081101561074a57600080fd5b506001600160a01b0381351690602001356112d4565b34801561076c57600080fd5b506103a66004803603602081101561078357600080fd5b50356001600160a01b031661133c565b34801561079f57600080fd5b5061037d600480360360408110156107b657600080fd5b506001600160a01b038135169060200135611368565b3480156107d857600080fd5b506103a6600480360360208110156107ef57600080fd5b50356001600160a01b031661137c565b34801561080b57600080fd5b506102b86004803603602081101561082257600080fd5b5035611397565b34801561083557600080fd5b5061037d6004803603604081101561084c57600080fd5b506001600160a01b0381351690602001351515611424565b34801561087057600080fd5b506103a6611515565b34801561088557600080fd5b506103a66004803603604081101561089c57600080fd5b506001600160a01b038135811691602001351661151b565b3480156108c057600080fd5b506102b8600480360360408110156108d757600080fd5b506001600160a01b038135169060200135611546565b3480156108f957600080fd5b506106c3611740565b34801561090e57600080fd5b506103a661174f565b34801561092357600080fd5b506102b86004803603602081101561093a57600080fd5b5035611755565b34801561094d57600080fd5b506102b86004803603602081101561096457600080fd5b50356001600160a01b031661188d565b34801561098057600080fd5b506105dc6004803603602081101561099757600080fd5b50356001600160a01b03166119ba565b3480156109b357600080fd5b506109d1600480360360208110156109ca57600080fd5b5035611b4b565b60408051938452602084019290925282820152519081900360600190f35b60006109f9610b65565b11610a0357600080fd5b3415610a8b57610a41610a14610b65565b610a2f34700100000000000000000000000000000000611c49565b81610a3657fe5b600654919004611ca9565b60065560408051348152905133917fa493a9229478c3fcd73f66d2cdeb7f94fd0f341da924d1054236d78454116511919081900360200190a2600b54610a879034611ca9565b600b555b565b60038054604080516020601f60027fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff610100600188161502019095169490940493840181900481028201810190925282815260609390929091830182828015610b375780601f10610b0c57610100808354040283529160200191610b37565b820191906000526020600020905b815481529060010190602001808311610b1a57829003601f168201915b5050505050905090565b6000610b55610b4e611d03565b8484611d07565b5060015b92915050565b600d5490565b60025490565b60136020526000908152604090205481565b6000610b8a848484611df3565b610bfa84610b96611d03565b610bf585604051806060016040528060288152602001612620602891396001600160a01b038a16600090815260016020526040812090610bd4611d03565b6001600160a01b031681526020810191909152604001600020549190611e2a565b611d07565b5060019392505050565b6001600160a01b03811660009081526009602052604081205470010000000000000000000000000000000090610c6090610c5b90610c55610c50610c4788611144565b60065490611c49565b611ec1565b90611ed1565b611f04565b81610c6757fe5b0490505b919050565b60115481565b60055460ff1690565b610c87611d03565b6001600160a01b0316610c9861123b565b6001600160a01b031614610cf3576040805162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572604482015290519081900360640190fd5b6001600160a01b03811660009081526012602052604090205460ff1615610d1957600080fd5b6001600160a01b038116600090815260126020526040812080547fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff00166001179055610d65908290611f17565b604080517f4c60db9c000000000000000000000000000000000000000000000000000000008152600d60048201526001600160a01b0383166024820152905173ca45381e7f3b6b9f314d5ff9bd09530155ab261491634c60db9c916044808301926000929190829003018186803b158015610ddf57600080fd5b505af4158015610df3573d6000803e3d6000fd5b5050505050565b6000610b55610e07611d03565b84610bf58560016000610e18611d03565b6001600160a01b03908116825260208083019390935260409182016000908120918c168152925290205490611ca9565b610e50611d03565b6001600160a01b0316610e6161123b565b6001600160a01b031614610ebc576040805162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572604482015290519081900360640190fd5b633b9aca0002601555565b610ecf611d03565b6001600160a01b0316610ee061123b565b6001600160a01b031614610f3b576040805162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572604482015290519081900360640190fd5b600880547fffffffffffffffffffffffff0000000000000000000000000000000000000000166001600160a01b0392909216919091179055565b60126020526000908152604090205460ff1681565b600080600080600080600080600d73ca45381e7f3b6b9f314d5ff9bd09530155ab261463deb3d89690916040518263ffffffff1660e01b81526004018082815260200191505060206040518083038186803b158015610fe857600080fd5b505af4158015610ffc573d6000803e3d6000fd5b505050506040513d602081101561101257600080fd5b505189106110575750600096507fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff9550859450869350839250829150819050806110fc565b6000600d73ca45381e7f3b6b9f314d5ff9bd09530155ab261463d1aa9e7e90918c6040518363ffffffff1660e01b8152600401808381526020018281526020019250505060206040518083038186803b1580156110b357600080fd5b505af41580156110c7573d6000803e3d6000fd5b505050506040513d60208110156110dd57600080fd5b505190506110ea816119ba565b98509850985098509850985098509850505b919395975091939597565b60405162461bcd60e51b815260040180806020018281038252606d815260200180612698606d913960800191505060405180910390fd5b60145481565b6001600160a01b031660009081526020819052604090205490565b611167611d03565b6001600160a01b031661117861123b565b6001600160a01b0316146111d3576040805162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572604482015290519081900360640190fd5b600c546040516000916001600160a01b0316907f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e0908390a3600c80547fffffffffffffffffffffffff0000000000000000000000000000000000000000169055565b600b5481565b600c546001600160a01b031690565b6000610b598261133c565b60048054604080516020601f60027fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff610100600188161502019095169490940493840181900481028201810190925282815260609390929091830182828015610b375780601f10610b0c57610100808354040283529160200191610b37565b6000610b556112e1611d03565b84610bf585604051806060016040528060258152602001612729602591396001600061130b611d03565b6001600160a01b03908116825260208083019390935260409182016000908120918d16815292529020549190611e2a565b6001600160a01b0381166000908152600a6020526040812054610b599061136284610c04565b90611f70565b6000610b55611375611d03565b8484611df3565b6001600160a01b03166000908152600a602052604090205490565b60006113a1610b65565b116113ab57600080fd5b8015611421576113d76113bc610b65565b610a2f83700100000000000000000000000000000000611c49565b60065560408051828152905133917fa493a9229478c3fcd73f66d2cdeb7f94fd0f341da924d1054236d78454116511919081900360200190a2600b5461141d9082611ca9565b600b555b50565b600061142e611d03565b6001600160a01b031661143f61123b565b6001600160a01b03161461149a576040805162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572604482015290519081900360640190fd5b60006114a584611fcd565b9050801561150b576001600160a01b0384166000818152601360209081526040918290204290558151848152915186151593927fa2c38e2d2fb7e3e1912d937fd1ca11ed6d51864dee4cfa7a7bf02becd7acf09292908290030190a36001915050610b59565b5060009392505050565b60155481565b6001600160a01b03918216600090815260016020908152604080832093909416825291909152205490565b61154e611d03565b6001600160a01b031661155f61123b565b6001600160a01b0316146115ba576040805162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572604482015290519081900360640190fd5b6001600160a01b03821660009081526012602052604090205460ff16156115e05761173c565b6015548110611691576115f38282611f17565b604080517fbc2b405c000000000000000000000000000000000000000000000000000000008152600d60048201526001600160a01b038416602482015260448101839052905173ca45381e7f3b6b9f314d5ff9bd09530155ab26149163bc2b405c916064808301926000929190829003018186803b15801561167457600080fd5b505af4158015611688573d6000803e3d6000fd5b5050505061172f565b61169c826000611f17565b604080517f4c60db9c000000000000000000000000000000000000000000000000000000008152600d60048201526001600160a01b0384166024820152905173ca45381e7f3b6b9f314d5ff9bd09530155ab261491634c60db9c916044808301926000929190829003018186803b15801561171657600080fd5b505af415801561172a573d6000803e3d6000fd5b505050505b61173a826001611424565b505b5050565b6008546001600160a01b031681565b60115490565b61175d611d03565b6001600160a01b031661176e61123b565b6001600160a01b0316146117c9576040805162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572604482015290519081900360640190fd5b610e1081101580156117de5750620151808111155b6118195760405162461bcd60e51b815260040180806020018281038252604e8152602001806125b1604e913960600191505060405180910390fd5b60145481141561185a5760405162461bcd60e51b81526004018080602001828103825260408152602001806125716040913960400191505060405180910390fd5b60145460405182907f474ea64804364a1e29a4487ddb63c3342a2dd826ccd8acf48825e680a0e6f20f90600090a3601455565b611895611d03565b6001600160a01b03166118a661123b565b6001600160a01b031614611901576040805162461bcd60e51b815260206004820181905260248201527f4f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e6572604482015290519081900360640190fd5b6001600160a01b0381166119465760405162461bcd60e51b81526004018080602001828103825260268152602001806125296026913960400191505060405180910390fd5b600c546040516001600160a01b038084169216907f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e090600090a3600c80547fffffffffffffffffffffffff0000000000000000000000000000000000000000166001600160a01b0392909216919091179055565b600080600080600080600080889750600d73ca45381e7f3b6b9f314d5ff9bd09530155ab26146317e142d190918a6040518363ffffffff1660e01b815260040180838152602001826001600160a01b031681526020019250505060206040518083038186803b158015611a2c57600080fd5b505af4158015611a40573d6000803e3d6000fd5b505050506040513d6020811015611a5657600080fd5b505196507fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff955060008712611ad857601154871115611aa457601154611a9d908890612217565b9550611ad8565b601154600d5460009110611ab9576000611ac8565b601154600d54611ac891611f70565b9050611ad48882611ed1565b9650505b611ae18861133c565b9450611aec88610c04565b6001600160a01b038916600090815260136020526040902054909450925082611b16576000611b24565b601454611b24908490611ca9565b9150428211611b34576000611b3e565b611b3e8242611f70565b9050919395975091939597565b600d546000908190819080611b6b57505060115460009250829150611c42565b6011546000805a90506000805b8984108015611b8657508582105b15611c3157600d546001909501948510611b9f57600094505b6000600d6000018681548110611bb157fe5b60009182526020808320909101546001600160a01b03168083526013909152604090912054909150611be29061224b565b15611bfe57611bf2816001611424565b15611bfe576001909101905b60019092019160005a905080851115611c2857611c25611c1e8683611f70565b8790611ca9565b95505b9350611b789050565b601185905590975095509193505050505b9193909250565b600082611c5857506000610b59565b82820282848281611c6557fe5b0414611ca25760405162461bcd60e51b81526004018080602001828103825260218152602001806125ff6021913960400191505060405180910390fd5b9392505050565b600082820183811015611ca2576040805162461bcd60e51b815260206004820152601b60248201527f536166654d6174683a206164646974696f6e206f766572666c6f770000000000604482015290519081900360640190fd5b3390565b6001600160a01b038316611d4c5760405162461bcd60e51b81526004018080602001828103825260248152602001806127056024913960400191505060405180910390fd5b6001600160a01b038216611d915760405162461bcd60e51b815260040180806020018281038252602281526020018061254f6022913960400191505060405180910390fd5b6001600160a01b03808416600081815260016020908152604080832094871680845294825291829020859055815185815291517f8c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b9259281900390910190a3505050565b60405162461bcd60e51b815260040180806020018281038252602f815260200180612648602f913960400191505060405180910390fd5b60008184841115611eb95760405162461bcd60e51b81526004018080602001828103825283818151815260200191508051906020019080838360005b83811015611e7e578181015183820152602001611e66565b50505050905090810190601f168015611eab5780820380516001836020036101000a031916815260200191505b509250505060405180910390fd5b505050900390565b60008181811215610b5957600080fd5b6000828201818312801590611ee65750838112155b80611efb5750600083128015611efb57508381125b611ca257600080fd5b600080821215611f1357600080fd5b5090565b6000611f2283611144565b905080821115611f4a576000611f388383611f70565b9050611f448482612272565b5061173a565b8082101561173a576000611f5e8284611f70565b9050611f6a84826122d6565b50505050565b600082821115611fc7576040805162461bcd60e51b815260206004820152601e60248201527f536166654d6174683a207375627472616374696f6e206f766572666c6f770000604482015290519081900360640190fd5b50900390565b600080611fd98361133c565b9050801561220e576001600160a01b0383166000908152600a60205260409020546120049082611ca9565b6001600160a01b038085166000908152600a6020526040812092909255600854166120d35760405160009081906001600160a01b0387169085908381818185875af1925050503d8060008114612076576040519150601f19603f3d011682016040523d82523d6000602084013e61207b565b606091505b506008546040805188815290519396508695509193506001600160a01b0390811692908916917feb063efb53b3790d2bc15284b59af7544466c8787c2883321ee27095647911b6919081900360200190a350506121b8565b600854604080517fa9059cbb0000000000000000000000000000000000000000000000000000000081526001600160a01b038781166004830152602482018690529151919092169163a9059cbb9160448083019260209291908290030181600087803b15801561214257600080fd5b505af1158015612156573d6000803e3d6000fd5b505050506040513d602081101561216c57600080fd5b50516008546040805185815290519293506001600160a01b0391821692918716917feb063efb53b3790d2bc15284b59af7544466c8787c2883321ee27095647911b69181900360200190a35b80612206576001600160a01b0384166000908152600a60205260409020546121e09083611f70565b6001600160a01b0385166000908152600a60205260408120919091559250610c6b915050565b509050610c6b565b50600092915050565b600080821215801561222b57508282840313155b806122425750600082128015612242575082828403135b611fc757600080fd5b60004282111561225d57506000610c6b565b60145461226a4284611f70565b101592915050565b61227c828261231a565b6122b6612297610c5083600654611c4990919063ffffffff16565b6001600160a01b03841660009081526009602052604090205490612217565b6001600160a01b0390921660009081526009602052604090209190915550565b6122e0828261240a565b6122b66122fb610c5083600654611c4990919063ffffffff16565b6001600160a01b03841660009081526009602052604090205490611ed1565b6001600160a01b038216612375576040805162461bcd60e51b815260206004820152601f60248201527f45524332303a206d696e7420746f20746865207a65726f206164647265737300604482015290519081900360640190fd5b6123816000838361173a565b60025461238e9082611ca9565b6002556001600160a01b0382166000908152602081905260409020546123b49082611ca9565b6001600160a01b0383166000818152602081815260408083209490945583518581529351929391927fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef9281900390910190a35050565b6001600160a01b03821661244f5760405162461bcd60e51b81526004018080602001828103825260218152602001806126776021913960400191505060405180910390fd5b61245b8260008361173a565b61249881604051806060016040528060228152602001612507602291396001600160a01b0385166000908152602081905260409020549190611e2a565b6001600160a01b0383166000908152602081905260409020556002546124be9082611f70565b6002556040805182815290516000916001600160a01b038516917fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef9181900360200190a3505056fe45524332303a206275726e20616d6f756e7420657863656564732062616c616e63654f776e61626c653a206e6577206f776e657220697320746865207a65726f206164647265737345524332303a20617070726f766520746f20746865207a65726f20616464726573735361666542616e6b5f4469766964656e645f547261636b65723a2043616e6e6f742075706461746520636c61696d5761697420746f2073616d652076616c75655361666542616e6b5f4469766964656e645f547261636b65723a20636c61696d57616974206d757374206265207570646174656420746f206265747765656e203120616e6420323420686f757273536166654d6174683a206d756c7469706c69636174696f6e206f766572666c6f7745524332303a207472616e7366657220616d6f756e74206578636565647320616c6c6f77616e63655361666542616e6b5f4469766964656e645f547261636b65723a204e6f207472616e736665727320616c6c6f77656445524332303a206275726e2066726f6d20746865207a65726f20616464726573735361666542616e6b5f4469766964656e645f547261636b65723a2077697468647261774469766964656e642064697361626c65642e20557365207468652027636c61696d272066756e6374696f6e206f6e20746865206d61696e205361666542616e6b20636f6e74726163742e45524332303a20617070726f76652066726f6d20746865207a65726f206164647265737345524332303a2064656372656173656420616c6c6f77616e63652062656c6f77207a65726fa264697066735822122043bdd2b2c90e37a5c23bb70551b59ca84fad3c035a12762ff29c15c0bcfe9c8d64736f6c634300070600335361666542616e6b3a204175746f6d61746564206d61726b6574206d616b6572207061697220697320616c72656164792073657420746f20746861742076616c7565

Libraries Used

IterableMapping : 0xca45381e7f3b6b9f314d5ff9bd09530155ab2614Unverified

Deployed ByteCode Sourcemap

49483:25520:0:-:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;49903:67;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;23771:91;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;51068:68;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;51068:68:0;-1:-1:-1;;;;;51068:68:0;;:::i;:::-;;;;;;;;;;;;;;;;;;25917:169;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;;;;;;25917:169:0;;;;;;;;:::i;50771:42::-;;;;;;;;;;;;;:::i;:::-;;;;-1:-1:-1;;;;;50771:42:0;;;;;;;;;;;;;;50558:24;;;;;;;;;;;;;:::i;55308:138::-;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;55308:138:0;;:::i;:::-;;49561:41;;;;;;;;;;;;;:::i;67211:4316::-;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;67211:4316:0;;:::i;24870:108::-;;;;;;;;;;;;;:::i;50636:43::-;;;;;;;;;;;;;:::i;26568:321::-;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;;;;;;26568:321:0;;;;;;;;;;;;;;;;;:::i;50895:51::-;;;;;;;;;;;;;:::i;49736:46::-;;;;;;;;;;;;;:::i;61305:141::-;;;;;;;;;;;;;:::i;24714:91::-;;;;;;;;;;;;;:::i;:::-;;;;;;;;;;;;;;;;;;;50423:25;;;;;;;;;;;;;:::i;56780:134::-;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;56780:134:0;-1:-1:-1;;;;;56780:134:0;;:::i;50118:37::-;;;;;;;;;;;;;:::i;27298:218::-;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;;;;;;27298:218:0;;;;;;;;:::i;56926:134::-;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;56926:134:0;-1:-1:-1;;;;;56926:134:0;;:::i;55115:181::-;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;55115:181:0;;:::i;55774:248::-;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;55774:248:0;-1:-1:-1;;;;;55774:248:0;;:::i;57214:452::-;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;57214:452:0;;;;;;;;;;;;;;;;;;;;;;:::i;49609:38::-;;;;;;;;;;;;;:::i;63222:126::-;;;;;;;;;;;;;:::i;62834:94::-;;;;;;;;;;;;;:::i;57680:504::-;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;;;;;;57680:504:0;;;;;;;;;;:::i;61454:125::-;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;61454:125:0;-1:-1:-1;;;;;61454:125:0;;:::i;56285:234::-;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;56285:234:0;-1:-1:-1;;;;;56285:234:0;;:::i;50391:25::-;;;;;;;;;;;;;:::i;54965:137::-;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;54965:137:0;;:::i;63073:141::-;;;;;;;;;;;;;:::i;58993:319::-;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;58993:319:0;-1:-1:-1;;;;;58993:319:0;;:::i;50297:40::-;;;;;;;;;;;;;:::i;61746:139::-;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;61746:139:0;-1:-1:-1;;;;;61746:139:0;;:::i;56034:239::-;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;56034:239:0;-1:-1:-1;;;;;56034:239:0;;:::i;62555:271::-;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;62555:271:0;;:::i;25041:127::-;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;25041:127:0;-1:-1:-1;;;;;25041:127:0;;:::i;49791:26::-;;;;;;;;;;;;;:::i;16118:148::-;;;;;;;;;;;;;:::i;57072:134::-;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;57072:134:0;-1:-1:-1;;;;;57072:134:0;;:::i;50037:58::-;;;;;;;;;;;;;:::i;60637:412::-;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;60637:412:0;;:::i;58192:793::-;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;58192:793:0;-1:-1:-1;;;;;58192:793:0;;:::i;15467:87::-;;;;;;;;;;;;;:::i;23981:95::-;;;;;;;;;;;;;:::i;50524:27::-;;;;;;;;;;;;;:::i;59944:268::-;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;;;;;;59944:268:0;;;;;;;;;;:::i;50718:40::-;;;;;;;;;;;;;:::i;73450:535::-;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;73450:535:0;;;;;;;:::i;61189:108::-;;;;;;;;;;;;;:::i;28019:269::-;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;;;;;;28019:269:0;;;;;;;;:::i;49830:66::-;;;;;;;;;;;;;:::i;61587:151::-;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;61587:151:0;-1:-1:-1;;;;;61587:151:0;;:::i;25381:175::-;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;;;;;;25381:175:0;;;;;;;;:::i;61893:318::-;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;61893:318:0;-1:-1:-1;;;;;61893:318:0;;:::i;:::-;;;;-1:-1:-1;;;;;61893:318:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;50344:40;;;;;;;;;;;;;:::i;51435:58::-;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;51435:58:0;-1:-1:-1;;;;;51435:58:0;;:::i;56534:234::-;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;56534:234:0;-1:-1:-1;;;;;56534:234:0;;:::i;59320:302::-;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;;;;;;59320:302:0;;;;;;;;;;:::i;55458:147::-;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;55458:147:0;;:::i;59630:306::-;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;59630:306:0;-1:-1:-1;59630:306:0;;;;:::i;55618:147::-;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;55618:147:0;;:::i;50455:25::-;;;;;;;;;;;;;:::i;25619:151::-;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;;;;;;25619:151:0;;;;;;;;;;:::i;50162:37::-;;;;;;;;;;;;;:::i;50250:40::-;;;;;;;;;;;;;:::i;49977:53::-;;;;;;;;;;;;;:::i;50487:30::-;;;;;;;;;;;;;:::i;49656:41::-;;;;;;;;;;;;;:::i;62936:129::-;;;;;;;;;;;;;:::i;50206:37::-;;;;;;;;;;;;;:::i;61057:124::-;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;61057:124:0;;:::i;62219:328::-;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;62219:328:0;;:::i;16421:244::-;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;16421:244:0;-1:-1:-1;;;;;16421:244:0;;:::i;49903:67::-;;;;:::o;23771:91::-;23849:5;23842:12;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;23816:13;;23842:12;;23849:5;;23842:12;;23849:5;23842:12;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;23771:91;:::o;51068:68::-;;;;;;;;;;;;;;;:::o;25917:169::-;26000:4;26017:39;26026:12;:10;:12::i;:::-;26040:7;26049:6;26017:8;:39::i;:::-;-1:-1:-1;26074:4:0;25917:169;;;;;:::o;50771:42::-;;;-1:-1:-1;;;;;50771:42:0;;:::o;50558:24::-;;;;:::o;55308:138::-;15698:12;:10;:12::i;:::-;-1:-1:-1;;;;;15687:23:0;:7;:5;:7::i;:::-;-1:-1:-1;;;;;15687:23:0;;15679:68;;;;;-1:-1:-1;;;15679:68:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;55397:21:::1;:41:::0;55308:138::o;49561:41::-;;;-1:-1:-1;;;;;49561:41:0;;:::o;67211:4316::-;67321:12;67336:13;:6;67347:1;67336:10;:13::i;:::-;67321:28;-1:-1:-1;67360:17:0;67380:16;:6;67321:28;67380:10;:16::i;:::-;67360:36;-1:-1:-1;67699:21:0;67765:22;67782:4;67765:16;:22::i;:::-;67918:18;67939:41;:21;67965:14;67939:25;:41::i;:::-;67918:62;;68074:25;68102:41;68132:10;;68102:25;68117:9;;68102:10;:14;;:25;;;;:::i;:::-;:29;;:41::i;:::-;68074:69;;68154:25;68182:40;68211:10;;68182:24;68196:9;;68182;:13;;:24;;;;:::i;:40::-;68154:68;;68314:25;68342:41;68372:10;;68342:25;68357:9;;68342:10;:14;;:25;;;;:::i;:41::-;68314:69;;68394:25;68422:40;68451:10;;68422:24;68436:9;;68422;:13;;:24;;;;:::i;:40::-;68394:68;;68546:25;68574:41;68604:10;;68574:25;68589:9;;68574:10;:14;;:25;;;;:::i;:41::-;68546:69;;68626:25;68654:40;68683:10;;68654:24;68668:9;;68654;:13;;:24;;;;:::i;:40::-;68626:68;-1:-1:-1;68734:37:0;;;:57;;68831:37;;;:57;;68705:26;68922:34;:10;68734:57;68922:14;:34::i;:::-;68899:57;-1:-1:-1;68967:17:0;68987:33;:9;69001:18;68987:13;:33::i;:::-;69111:25;;68967:53;;-1:-1:-1;;;;;;69111:25:0;:39;69108:698;;69202:25;;69229:14;;69166:78;;69183:17;;-1:-1:-1;;;;;69202:25:0;;;;69229:14;69166:16;:78::i;:::-;69108:698;;;69458:14;;:49;;69426:9;;;;-1:-1:-1;;;;;69458:14:0;;;;69485:17;;69426:9;69458:49;69426:9;69458:49;69485:17;69458:14;:49;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;69425:82;;;;69525:4;69522:273;;;;;;69729:50;69742:17;69761;69729:12;:50::i;:::-;69108:698;;;69886:25;;-1:-1:-1;;;;;69886:25:0;:39;69883:698;;69977:25;;70004:14;;69941:78;;69958:17;;-1:-1:-1;;;;;69977:25:0;;;;70004:14;69941:16;:78::i;:::-;69883:698;;;70233:14;;:49;;70201:9;;;;-1:-1:-1;;;;;70233:14:0;;;;70260:17;;70201:9;70233:49;70201:9;70233:49;70260:17;70233:14;:49;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;70200:82;;;;70300:4;70297:273;;;;;;70504:50;70517:17;70536;70504:12;:50::i;:::-;69883:698;;;70661:25;;-1:-1:-1;;;;;70661:25:0;:39;70658:698;;70752:25;;70779:14;;70716:78;;70733:17;;-1:-1:-1;;;;;70752:25:0;;;;70779:14;70716:16;:78::i;:::-;70658:698;;;71008:14;;:49;;70976:9;;;;-1:-1:-1;;;;;71008:14:0;;;;71035:17;;70976:9;71008:49;70976:9;71008:49;71035:17;71008:14;:49;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;70975:82;;;;71075:4;71072:273;;;;;;71279:50;71292:17;71311;71279:12;:50::i;:::-;70658:698;;;71413:37;71426:9;71437:12;71413;:37::i;:::-;71476:43;;;;;;;;;;;;;;;;;;;;;;;;;;;;;67211:4316;;;;;;;;;;;;;;;:::o;24870:108::-;24958:12;;24870:108;:::o;50636:43::-;;;;:::o;26568:321::-;26674:4;26691:36;26701:6;26709:9;26720:6;26691:9;:36::i;:::-;26738:121;26747:6;26755:12;:10;:12::i;:::-;26769:89;26807:6;26769:89;;;;;;;;;;;;;;;;;-1:-1:-1;;;;;26769:19:0;;;;;;:11;:19;;;;;;26789:12;:10;:12::i;:::-;-1:-1:-1;;;;;26769:33:0;;;;;;;;;;;;-1:-1:-1;26769:33:0;;;:89;:37;:89::i;:::-;26738:8;:121::i;:::-;-1:-1:-1;26877:4:0;26568:321;;;;;:::o;50895:51::-;;;;:::o;49736:46::-;;;-1:-1:-1;;;;;49736:46:0;;:::o;61305:141::-;61395:15;;:43;;;;;;;;61368:7;;-1:-1:-1;;;;;61395:15:0;;:41;;:43;;;;;;;;;;;;;;:15;:43;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;61395:43:0;;-1:-1:-1;61305:141:0;:::o;24714:91::-;24788:9;;;;24714:91;:::o;50423:25::-;;;;:::o;56780:134::-;15698:12;:10;:12::i;:::-;-1:-1:-1;;;;;15687:23:0;:7;:5;:7::i;:::-;-1:-1:-1;;;;;15687:23:0;;15679:68;;;;;-1:-1:-1;;;15679:68:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;56868:25:::1;:38:::0;;;::::1;-1:-1:-1::0;;;;;56868:38:0;;;::::1;::::0;;;::::1;::::0;;56780:134::o;50118:37::-;;;-1:-1:-1;;;;;50118:37:0;;:::o;27298:218::-;27386:4;27403:83;27412:12;:10;:12::i;:::-;27426:7;27435:50;27474:10;27435:11;:25;27447:12;:10;:12::i;:::-;-1:-1:-1;;;;;27435:25:0;;;;;;;;;;;;;;;;;-1:-1:-1;27435:25:0;;;:34;;;;;;;;;;;:38;:50::i;56926:134::-;15698:12;:10;:12::i;:::-;-1:-1:-1;;;;;15687:23:0;:7;:5;:7::i;:::-;-1:-1:-1;;;;;15687:23:0;;15679:68;;;;;-1:-1:-1;;;15679:68:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;57014:25:::1;:38:::0;;;::::1;-1:-1:-1::0;;;;;57014:38:0;;;::::1;::::0;;;::::1;::::0;;56926:134::o;55115:181::-;15698:12;:10;:12::i;:::-;-1:-1:-1;;;;;15687:23:0;:7;:5;:7::i;:::-;-1:-1:-1;;;;;15687:23:0;;15679:68;;;;;-1:-1:-1;;;15679:68:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;55218:15:::1;::::0;:70:::1;::::0;;;;;::::1;::::0;::::1;::::0;;;;;-1:-1:-1;;;;;55218:15:0;;::::1;::::0;:48:::1;::::0;:70;;;;;:15:::1;::::0;:70;;;;;;;;:15;;:70;::::1;;::::0;::::1;;;;::::0;::::1;;;;;;;;;;;;::::0;::::1;;;;;;;;;55115:181:::0;:::o;55774:248::-;15698:12;:10;:12::i;:::-;-1:-1:-1;;;;;15687:23:0;:7;:5;:7::i;:::-;-1:-1:-1;;;;;15687:23:0;;15679:68;;;;;-1:-1:-1;;;15679:68:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;55856:15:::1;::::0;:50:::1;::::0;;;;;-1:-1:-1;;;;;55856:50:0;;::::1;;::::0;::::1;::::0;;;:15;;;::::1;::::0;:38:::1;::::0;:50;;;;;:15:::1;::::0;:50;;;;;;;:15;;:50;::::1;;::::0;::::1;;;;::::0;::::1;;;;;;;;;;;;::::0;::::1;;;;;-1:-1:-1::0;;55917:13:0::1;:26:::0;;;::::1;-1:-1:-1::0;;;;;55917:26:0;;::::1;::::0;;::::1;::::0;;;;55959:55:::1;::::0;55999:13;::::1;::::0;-1:-1:-1;55917:26:0;-1:-1:-1;55959:55:0::1;::::0;-1:-1:-1;;55959:55:0::1;55774:248:::0;:::o;57214:452::-;15698:12;:10;:12::i;:::-;-1:-1:-1;;;;;15687:23:0;:7;:5;:7::i;:::-;-1:-1:-1;;;;;15687:23:0;;15679:68;;;;;-1:-1:-1;;;15679:68:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;57376:15:::1;:34:::0;;;57421:12:::1;:28:::0;;;57460:10:::1;:24:::0;;;57495:10:::1;:24:::0;;;57530:10:::1;:24:::0;;;57577:81:::1;57543:11:::0;57577:65:::1;57508:11:::0;57577:65;57473:11;57577:65;57394:16;57436:13;57577:19:::1;:33::i;:::-;:37:::0;::::1;:49::i;:81::-;57565:9;:93:::0;-1:-1:-1;;;;;57214:452:0:o;49609:38::-;;;:::o;63222:126::-;63317:23;;63298:15;:42;;63222:126;:::o;62834:94::-;62871:15;;:49;;;;;;62902:10;62871:49;;;;:15;:49;;;;;;;;-1:-1:-1;;;;;62871:15:0;;;;:30;;:49;;;;;;;;;;;;;;;;;;:15;:49;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;;62834:94:0:o;57680:504::-;15698:12;:10;:12::i;:::-;-1:-1:-1;;;;;15687:23:0;:7;:5;:7::i;:::-;-1:-1:-1;;;;;15687:23:0;;15679:68;;;;;-1:-1:-1;;;15679:68:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;57784:14:::1;:32:::0;;;::::1;-1:-1:-1::0;;;;;57784:32:0;;::::1;::::0;;::::1;::::0;;;;57861:14;;::::1;-1:-1:-1::0;57827:49:0;;;:33:::1;:49:::0;;;;;;:56;;;::::1;-1:-1:-1::0;57827:56:0::1;::::0;;57894:15:::1;::::0;:53;;;;;::::1;::::0;::::1;::::0;;;;;;:15;::::1;::::0;:36:::1;::::0;:53;;;;;-1:-1:-1;57894:53:0;;;;;;-1:-1:-1;57894:15:0;:53;::::1;;::::0;::::1;;;;::::0;::::1;;;;;;;;;;;;::::0;::::1;;;;;;;;;57958:38;57974:15;57991:4;57958:15;:38::i;:::-;-1:-1:-1::0;;;;;58009:49:0;;::::1;;::::0;;;:33:::1;:49:::0;;;;;;:56;;;::::1;58061:4;58009:56;::::0;;58076:15:::1;::::0;:52;;;;;::::1;::::0;::::1;::::0;;;;;;:15;::::1;::::0;:36:::1;::::0;:52;;;;;58009:49;58076:52;;;;;;58009:49;58076:15;:52;::::1;;::::0;::::1;;;;::::0;::::1;;;;;;;;;;;;::::0;::::1;;;;;;;;;58139:37;58155:14;58171:4;58139:15;:37::i;61454:125::-:0;-1:-1:-1;;;;;61543:28:0;61519:4;61543:28;;;:19;:28;;;;;;;;;61454:125::o;56285:234::-;15698:12;:10;:12::i;:::-;-1:-1:-1;;;;;15687:23:0;:7;:5;:7::i;:::-;-1:-1:-1;;;;;15687:23:0;;15679:68;;;;;-1:-1:-1;;;15679:68:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;56373:14:::1;:27:::0;;;::::1;-1:-1:-1::0;;;;;56373:27:0;;::::1;::::0;;;::::1;::::0;;;;56411:37:::1;::::0;56427:14:::1;-1:-1:-1::0;56411:15:0::1;:37::i;:::-;56459:15;::::0;56496:14:::1;::::0;56459:52:::1;::::0;;;;;-1:-1:-1;;;;;56496:14:0;;::::1;56459:52;::::0;::::1;::::0;;;:15;;;::::1;::::0;:36:::1;::::0;:52;;;;;:15:::1;::::0;:52;;;;;;;:15;;:52;::::1;;::::0;::::1;;;;::::0;::::1;50391:25:::0;;;;:::o;54965:137::-;15698:12;:10;:12::i;:::-;-1:-1:-1;;;;;15687:23:0;:7;:5;:7::i;:::-;-1:-1:-1;;;;;15687:23:0;;15679:68;;;;;-1:-1:-1;;;15679:68:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;55054:23:::1;:40:::0;54965:137::o;63073:141::-;63165:15;;:41;;;;;;;;63138:7;;-1:-1:-1;;;;;63165:15:0;;:39;;:41;;;;;;;;;;;;;;:15;:41;;;;;;;;;;58993:319;15698:12;:10;:12::i;:::-;-1:-1:-1;;;;;15687:23:0;:7;:5;:7::i;:::-;-1:-1:-1;;;;;15687:23:0;;15679:68;;;;;-1:-1:-1;;;15679:68:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;59104:15:::1;::::0;-1:-1:-1;;;;;59082:38:0;;::::1;59104:15:::0;::::1;59082:38;;59074:96;;;;-1:-1:-1::0;;;59074:96:0::1;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;59228:15;::::0;59186:59:::1;::::0;-1:-1:-1;;;;;59228:15:0;;::::1;::::0;59186:59;::::1;::::0;::::1;::::0;59228:15:::1;::::0;59186:59:::1;59256:15;:48:::0;;;::::1;-1:-1:-1::0;;;;;59256:48:0;;;::::1;::::0;;;::::1;::::0;;58993:319::o;50297:40::-;;;-1:-1:-1;;;;;50297:40:0;;:::o;61746:139::-;61843:15;;:34;;;;;;-1:-1:-1;;;;;61843:34:0;;;;;;;;;61816:7;;61843:15;;;;;:25;;:34;;;;;;;;;;;;;;;:15;:34;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;61843:34:0;;61746:139;-1:-1:-1;;61746:139:0:o;56034:239::-;15698:12;:10;:12::i;:::-;-1:-1:-1;;;;;15687:23:0;:7;:5;:7::i;:::-;-1:-1:-1;;;;;15687:23:0;;15679:68;;;;;-1:-1:-1;;;15679:68:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;56122:14:::1;:27:::0;;;::::1;-1:-1:-1::0;;;;;56122:27:0;;::::1;::::0;;;::::1;::::0;;;;56160:37:::1;::::0;56176:14:::1;-1:-1:-1::0;56160:15:0::1;:37::i;:::-;56208:15;::::0;56245:14:::1;::::0;56208:52:::1;::::0;;;;;-1:-1:-1;;;;;56245:14:0;;::::1;56208:52;::::0;::::1;::::0;;;:15;;;::::1;::::0;:36:::1;::::0;:52;;;;;:15:::1;::::0;:52;;;;;;;:15;;:52;::::1;;::::0;::::1;;;;::::0;::::1;62555:271:::0;62687:15;;:28;;;;;;;;;;;;;;62621:18;;;;;;-1:-1:-1;;;;;62687:15:0;;;;:23;;:28;;;;;;;;;;;;;;;62621:18;62687:15;:28;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;62687:28:0;;;;;;;;;;;;62731:87;;;;;;;;;;;;;;;;;62687:28;62731:87;;;;;;;62687:28;;-1:-1:-1;62687:28:0;-1:-1:-1;62687:28:0;;-1:-1:-1;62808:9:0;;62796:5;;62731:87;;;;;;;;;62555:271;;;;:::o;25041:127::-;-1:-1:-1;;;;;25142:18:0;25115:7;25142:18;;;;;;;;;;;;25041:127::o;49791:26::-;;;-1:-1:-1;;;;;49791:26:0;;:::o;16118:148::-;15698:12;:10;:12::i;:::-;-1:-1:-1;;;;;15687:23:0;:7;:5;:7::i;:::-;-1:-1:-1;;;;;15687:23:0;;15679:68;;;;;-1:-1:-1;;;15679:68:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;16209:6:::1;::::0;16188:40:::1;::::0;16225:1:::1;::::0;16209:6:::1;::::0;::::1;-1:-1:-1::0;;;;;16209:6:0::1;::::0;16188:40:::1;::::0;16225:1;;16188:40:::1;16239:6;:19:::0;;;::::1;::::0;;16118:148::o;57072:134::-;15698:12;:10;:12::i;:::-;-1:-1:-1;;;;;15687:23:0;:7;:5;:7::i;:::-;-1:-1:-1;;;;;15687:23:0;;15679:68;;;;;-1:-1:-1;;;15679:68:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;57160:25:::1;:38:::0;;;::::1;-1:-1:-1::0;;;;;57160:38:0;;;::::1;::::0;;;::::1;::::0;;57072:134::o;50037:58::-;;;;:::o;60637:412::-;15698:12;:10;:12::i;:::-;-1:-1:-1;;;;;15687:23:0;:7;:5;:7::i;:::-;-1:-1:-1;;;;;15687:23:0;;15679:68;;;;;-1:-1:-1;;;15679:68:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;60735:6:::1;60723:8;:18;;:40;;;;;60757:6;60745:8;:18;;60723:40;60715:115;;;;-1:-1:-1::0;;;60715:115:0::1;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;60861:16;;60849:8;:28;;60841:95;;;;-1:-1:-1::0;;;60841:95:0::1;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;60986:16;::::0;60952:51:::1;::::0;60976:8;;60952:51:::1;::::0;;;::::1;61014:16;:27:::0;60637:412::o;58192:793::-;15698:12;:10;:12::i;:::-;-1:-1:-1;;;;;15687:23:0;:7;:5;:7::i;:::-;-1:-1:-1;;;;;15687:23:0;;15679:68;;;;;-1:-1:-1;;;15679:68:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;58303:15:::1;::::0;-1:-1:-1;;;;;58281:38:0;;::::1;58303:15:::0;::::1;58281:38;;58273:106;;;;-1:-1:-1::0;;;58273:106:0::1;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;58392:42;58469:10;58392:89;;58540:4;-1:-1:-1::0;;;;;58502:43:0::1;:18;-1:-1:-1::0;;;;;58502:24:0::1;;:26;;;;;;;;;;;;;;;;;;;;;;;;;;;::::0;::::1;;;;;;;;;;;;::::0;::::1;;;;;;;;;;;;;;;;;;;::::0;::::1;;-1:-1:-1::0;58502:26:0;-1:-1:-1;;;;;58502:43:0::1;;58494:135;;;;-1:-1:-1::0;;;58494:135:0::1;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;58642:18;-1:-1:-1::0;;;;;58642:39:0::1;;58690:18;58642:68;;;;;;;;;;;;;-1:-1:-1::0;;;;;58642:68:0::1;;;;;;;;;;;;;;;;;;;;;;;;;::::0;::::1;;;;;;;;;;;;::::0;::::1;;;;;-1:-1:-1::0;;58721:54:0::1;::::0;;;;;58769:4:::1;58721:54;::::0;::::1;::::0;;;-1:-1:-1;;;;;58721:39:0;::::1;::::0;-1:-1:-1;58721:39:0::1;::::0;-1:-1:-1;58721:54:0;;;;;-1:-1:-1;;58721:54:0;;;;;;;-1:-1:-1;58721:39:0;:54;::::1;;::::0;::::1;;;;::::0;::::1;;;;;;;;;;;;::::0;::::1;;;;;-1:-1:-1::0;;58834:15:0::1;::::0;58786:65:::1;::::0;;;;;-1:-1:-1;;;;;58834:15:0;;::::1;58786:65;::::0;::::1;::::0;;;:39;;::::1;::::0;-1:-1:-1;58786:39:0::1;::::0;-1:-1:-1;58786:65:0;;;;;58834:15:::1;::::0;58786:65;;;;;;;;58834:15;58786:39;:65;::::1;;::::0;::::1;;;;::::0;::::1;;;;;;;;;;;;::::0;::::1;;;;;-1:-1:-1::0;;58911:15:0::1;::::0;58869:59:::1;::::0;-1:-1:-1;;;;;58911:15:0;;::::1;::::0;-1:-1:-1;58869:59:0;;::::1;::::0;-1:-1:-1;58869:59:0::1;::::0;58911:15:::1;::::0;58869:59:::1;58941:15;:36:::0;;;::::1;-1:-1:-1::0;;;;;58941:36:0;;;::::1;::::0;;;::::1;::::0;;-1:-1:-1;58192:793:0:o;15467:87::-;15540:6;;;;;-1:-1:-1;;;;;15540:6:0;;15467:87::o;23981:95::-;24061:7;24054:14;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;24028:13;;24054:14;;24061:7;;24054:14;;24061:7;24054:14;;;;;;;;;;;;;;;;;;;;;;;;50524:27;;;;:::o;59944:268::-;15698:12;:10;:12::i;:::-;-1:-1:-1;;;;;15687:23:0;:7;:5;:7::i;:::-;-1:-1:-1;;;;;15687:23:0;;15679:68;;;;;-1:-1:-1;;;15679:68:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;60053:13:::1;-1:-1:-1::0;;;;;60045:21:0::1;:4;-1:-1:-1::0;;;;;60045:21:0::1;;;60037:113;;;;-1:-1:-1::0;;;60037:113:0::1;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;60163:41;60192:4;60198:5;60163:28;:41::i;50718:40::-:0;;;;:::o;73450:535::-;73639:15;;73607:62;;73624:4;;-1:-1:-1;;;;;73639:15:0;73657:11;73607:8;:62::i;:::-;73711:15;;73915:11;;73711:256;;;;;;73783:4;73711:256;;;;;;;;;;:15;:256;;;;;;;;;;-1:-1:-1;;;;;73915:11:0;;;73711:256;;;;73941:15;73711:256;;;;;;:15;;;;;:31;;73750:9;;73711:256;;;;;;;;;;;;;;;73750:9;73711:15;:256;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;;;;73450:535:0:o;61189:108::-;61262:15;;:27;;;;;;;;61235:7;;-1:-1:-1;;;;;61262:15:0;;:25;;:27;;;;;;;;;;;;;;:15;:27;;;;;;;;;;28019:269;28112:4;28129:129;28138:12;:10;:12::i;:::-;28152:7;28161:96;28200:15;28161:96;;;;;;;;;;;;;;;;;:11;:25;28173:12;:10;:12::i;:::-;-1:-1:-1;;;;;28161:25:0;;;;;;;;;;;;;;;;;-1:-1:-1;28161:25:0;;;:34;;;;;;;;;;;:96;:38;:96::i;49830:66::-;;;;:::o;61587:151::-;61683:15;;:47;;;;;;-1:-1:-1;;;;;61683:47:0;;;;;;;;;61656:7;;61683:15;;;;;:38;;:47;;;;;;;;;;;;;;;:15;:47;;;;;;;;;;25381:175;25467:4;25484:42;25494:12;:10;:12::i;:::-;25508:9;25519:6;25484:9;:42::i;61893:318::-;61989:7;62011:6;62032;62053:7;62075;62097;62119;62141;62168:15;;;;;;;;;-1:-1:-1;;;;;62168:15:0;-1:-1:-1;;;;;62168:26:0;;62195:7;62168:35;;;;;;;;;;;;;-1:-1:-1;;;;;62168:35:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;62168:35:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;62168:35:0;;-1:-1:-1;62168:35:0;;-1:-1:-1;62168:35:0;-1:-1:-1;62168:35:0;-1:-1:-1;62168:35:0;-1:-1:-1;62168:35:0;;-1:-1:-1;62168:35:0;-1:-1:-1;61893:318:0;;;;;;;;;:::o;50344:40::-;;;-1:-1:-1;;;;;50344:40:0;;:::o;51435:58::-;;;;;;;;;;;;;;;:::o;56534:234::-;15698:12;:10;:12::i;:::-;-1:-1:-1;;;;;15687:23:0;:7;:5;:7::i;:::-;-1:-1:-1;;;;;15687:23:0;;15679:68;;;;;-1:-1:-1;;;15679:68:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;56622:14:::1;:27:::0;;;::::1;-1:-1:-1::0;;;;;56622:27:0;;::::1;::::0;;;::::1;::::0;;;;56660:37:::1;::::0;56676:14:::1;-1:-1:-1::0;56660:15:0::1;:37::i;:::-;56708:15;::::0;56745:14:::1;::::0;56708:52:::1;::::0;;;;;-1:-1:-1;;;;;56745:14:0;;::::1;56708:52;::::0;::::1;::::0;;;:15;;;::::1;::::0;:36:::1;::::0;:52;;;;;:15:::1;::::0;:52;;;;;;;:15;;:52;::::1;;::::0;::::1;;;;::::0;::::1;59320:302:::0;15698:12;:10;:12::i;:::-;-1:-1:-1;;;;;15687:23:0;:7;:5;:7::i;:::-;-1:-1:-1;;;;;15687:23:0;;15679:68;;;;;-1:-1:-1;;;15679:68:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;;;;;59523:28:0;::::1;;::::0;;;:19:::1;:28;::::0;;;;;;;;:39;;;::::1;::::0;::::1;;::::0;;::::1;::::0;;;59580:34;;;;;;;::::1;::::0;;;;;;;;::::1;59320:302:::0;;:::o;55458:147::-;15698:12;:10;:12::i;:::-;-1:-1:-1;;;;;15687:23:0;:7;:5;:7::i;:::-;-1:-1:-1;;;;;15687:23:0;;15679:68;;;;;-1:-1:-1;;;15679:68:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;55591:5:::1;55567:30;55549:15;:48:::0;55458:147::o;59630:306::-;15698:12;:10;:12::i;:::-;-1:-1:-1;;;;;15687:23:0;:7;:5;:7::i;:::-;-1:-1:-1;;;;;15687:23:0;;15679:68;;;;;-1:-1:-1;;;15679:68:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;59749:9:::1;59745:115;59764:19:::0;;::::1;59745:115;;;59840:8;59805:19;:32;59825:8;;59834:1;59825:11;;;;;;;;::::0;;::::1;::::0;;;::::1;;-1:-1:-1::0;;;;;59825:11:0::1;59805:32:::0;;-1:-1:-1;59805:32:0;::::1;::::0;;;;;;-1:-1:-1;59805:32:0;:43;;;::::1;::::0;::::1;;::::0;;;::::1;::::0;;-1:-1:-1;59785:3:0::1;59745:115;;;;59877:51;59909:8;;59919;59877:51;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;::::0;;::::1;::::0;::::1;::::0;::::1;::::0;;::::1;::::0;::::1;::::0;;::::1;::::0;;::::1;::::0;-1:-1:-1;59877:51:0;;-1:-1:-1;;;;;59877:51:0::1;59630:306:::0;;;:::o;55618:147::-;15698:12;:10;:12::i;:::-;-1:-1:-1;;;;;15687:23:0;:7;:5;:7::i;:::-;-1:-1:-1;;;;;15687:23:0;;15679:68;;;;;-1:-1:-1;;;15679:68:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;55751:5:::1;55727:30;55706:18;:51:::0;55618:147::o;50455:25::-;;;;:::o;25619:151::-;-1:-1:-1;;;;;25735:18:0;;;25708:7;25735:18;;;:11;:18;;;;;;;;:27;;;;;;;;;;;;;25619:151::o;50162:37::-;;;-1:-1:-1;;;;;50162:37:0;;:::o;50250:40::-;;;-1:-1:-1;;;;;50250:40:0;;:::o;49977:53::-;;;;:::o;50487:30::-;;;;:::o;49656:41::-;;;-1:-1:-1;;;;;49656:41:0;;:::o;62936:129::-;63018:15;;:39;;;;;;;;62991:7;;-1:-1:-1;;;;;63018:15:0;;:37;;:39;;;;;;;;;;;;;;:15;:39;;;;;;;;;;50206:37;;;-1:-1:-1;;;;;50206:37:0;;:::o;61057:124::-;15698:12;:10;:12::i;:::-;-1:-1:-1;;;;;15687:23:0;:7;:5;:7::i;:::-;-1:-1:-1;;;;;15687:23:0;;15679:68;;;;;-1:-1:-1;;;15679:68:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;61131:15:::1;::::0;:42:::1;::::0;;;;;::::1;::::0;::::1;::::0;;;;;-1:-1:-1;;;;;61131:15:0;;::::1;::::0;:31:::1;::::0;:42;;;;;:15:::1;::::0;:42;;;;;;;;:15;;:42;::::1;;::::0;::::1;;;;::::0;::::1;62219:328:::0;62320:7;62342:6;62363;62384:7;62406;62428;62450;62472;62499:15;;;;;;;;;-1:-1:-1;;;;;62499:15:0;-1:-1:-1;;;;;62499:33:0;;62533:5;62499:40;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;16421:244;15698:12;:10;:12::i;:::-;-1:-1:-1;;;;;15687:23:0;:7;:5;:7::i;:::-;-1:-1:-1;;;;;15687:23:0;;15679:68;;;;;-1:-1:-1;;;15679:68:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;;;;;16510:22:0;::::1;16502:73;;;;-1:-1:-1::0;;;16502:73:0::1;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;16612:6;::::0;16591:38:::1;::::0;-1:-1:-1;;;;;16591:38:0;;::::1;::::0;16612:6:::1;::::0;::::1;;::::0;16591:38:::1;::::0;;;::::1;16640:6;:17:::0;;-1:-1:-1;;;;;16640:17:0;;::::1;;;::::0;;;::::1;::::0;;;::::1;::::0;;16421:244::o;35415:179::-;35473:7;35505:5;;;35529:6;;;;35521:46;;;;;-1:-1:-1;;;35521:46:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;35585:1;35415:179;-1:-1:-1;;;35415:179:0:o;4086:106::-;4174:10;4086:106;:::o;31166:346::-;-1:-1:-1;;;;;31268:19:0;;31260:68;;;;-1:-1:-1;;;31260:68:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;;;;;31347:21:0;;31339:68;;;;-1:-1:-1;;;31339:68:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;;;;;31420:18:0;;;;;;;:11;:18;;;;;;;;:27;;;;;;;;;;;;;:36;;;31472:32;;;;;;;;;;;;;;;;;31166:346;;;:::o;36992:153::-;37050:7;37082:1;37078;:5;37070:44;;;;;-1:-1:-1;;;37070:44:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;37136:1;37132;:5;;;;;;;36992:153;-1:-1:-1;;;36992:153:0:o;35877:158::-;35935:7;35968:1;35963;:6;;35955:49;;;;;-1:-1:-1;;;35955:49:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;36022:5:0;;;35877:158::o;72158:611::-;72320:16;;;72334:1;72320:16;;;;;;;;72296:21;;72320:16;;;;;;;;;;-1:-1:-1;72320:16:0;72296:40;;72365:4;72347;72352:1;72347:7;;;;;;;;-1:-1:-1;;;;;72347:23:0;;;:7;;;;;;;;;;:23;;;;72391:15;;:22;;;;;;;;:15;;;;;:20;;:22;;;;;72347:7;;72391:22;;;;;:15;:22;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;72391:22:0;72381:7;;:4;;72386:1;;72381:7;;;;;;-1:-1:-1;;;;;72381:32:0;;;:7;;;;;;;;;:32;72458:15;;72426:62;;72443:4;;72458:15;72476:11;72426:8;:62::i;:::-;72527:15;;:224;;;;;;;;;;;:15;:224;;;;;;72705:4;72527:224;;;;;;72725:15;72527:224;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;;;;;72527:15:0;;;;:66;;72608:11;;72678:4;;72705;72725:15;72527:224;;;;;;;;;;;;;;;;:15;:224;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;72158:611;;:::o;36294:220::-;36352:7;36376:6;36372:20;;-1:-1:-1;36391:1:0;36384:8;;36372:20;36415:5;;;36419:1;36415;:5;:1;36439:5;;;;;:10;36431:56;;;;-1:-1:-1;;;36431:56:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;71612:538;71800:16;;;71814:1;71800:16;;;;;;;;71776:21;;71800:16;;;;;;;;-1:-1:-1;;71837:15:0;;:22;;;;;;;;71776:40;;-1:-1:-1;;;;;;71837:15:0;;;;:20;;-1:-1:-1;71837:22:0;;;;;;;;;;;;;;:15;:22;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;71837:22:0;71827:7;;:4;;71832:1;;71827:7;;;;;;;;;:32;-1:-1:-1;;;;;71827:32:0;;;-1:-1:-1;;;;;71827:32:0;;;;;71880:12;71870:4;71875:1;71870:7;;;;;;;;-1:-1:-1;;;;;71870:22:0;;;:7;;;;;;;;;;:22;;;;71931:15;;:211;;;;;:15;:211;;;;;;;;;;;;;72116:15;71931:211;;;;;;;;;;;;;;;;;;;;;:15;;;;;:66;;72005:9;;71931:15;;72074:4;;72093:8;;71931:211;;;;;;;;;;;;;;;:15;:211;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;71612:538;;;;:::o;63356:3811::-;-1:-1:-1;;;;;63488:18:0;;63480:68;;;;-1:-1:-1;;;63480:68:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;;;;;63567:16:0;;63559:64;;;;-1:-1:-1;;;63559:64:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;63674:7;:5;:7::i;:::-;-1:-1:-1;;;;;63666:15:0;:4;-1:-1:-1;;;;;63666:15:0;;;:45;;;;;63704:7;:5;:7::i;:::-;-1:-1:-1;;;;;63698:13:0;:2;-1:-1:-1;;;;;63698:13:0;;;63666:45;:78;;;;-1:-1:-1;;;;;;63728:16:0;;;;63666:78;:116;;;;-1:-1:-1;;;;;;63761:21:0;;63775:6;63761:21;;63666:116;:152;;;;;63805:13;-1:-1:-1;;;;;63799:19:0;:2;-1:-1:-1;;;;;63799:19:0;;;63666:152;63648:586;;;63881:23;;63871:6;:33;;63845:135;;;;-1:-1:-1;;;63845:135:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;64009:32;64044:13;64054:2;64044:9;:13::i;:::-;64009:48;;64135:15;;64125:6;64098:24;:33;:52;;64072:150;;;;-1:-1:-1;;;64072:150:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;63648:586;;64274:21;64298;:19;:21::i;:::-;64274:45;;64477:16;64473:167;;-1:-1:-1;;;;;64518:39:0;;;;;;:33;:39;;;;;;;;64510:118;;;;-1:-1:-1;;;64510:118:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;64655:11;64652:92;;64683:28;64699:4;64705:2;64709:1;64683:15;:28::i;:::-;64726:7;;;64652:92;64775:8;;;;;;;64774:9;:42;;;;;64800:16;64774:42;:88;;;;-1:-1:-1;;;;;;64833:29:0;;;;;;:25;:29;;;;;;;;64774:88;:204;;;;-1:-1:-1;64962:15:0;;-1:-1:-1;;;;;64946:32:0;;;64962:15;;64946:32;;64774:204;:309;;;;-1:-1:-1;;;;;;65060:23:0;;;;;;:19;:23;;;;;;;;65059:24;64774:309;64756:509;;;65166:24;;65156:6;:34;;65148:105;;;;-1:-1:-1;;;65148:105:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;65277:28;65308:24;65326:4;65308:9;:24::i;:::-;65392:18;;65277:55;;-1:-1:-1;65368:42:0;;;65440:16;:41;;;;;65474:7;65440:41;:67;;;;-1:-1:-1;65499:8:0;;;;;;;65498:9;65440:67;:116;;;;-1:-1:-1;;;;;;65525:31:0;;;;;;:25;:31;;;;;;;;65524:32;65440:116;:152;;;;-1:-1:-1;65581:11:0;;-1:-1:-1;;;;;65573:19:0;;;65581:11;;65573:19;;65440:152;:186;;;;-1:-1:-1;65615:11:0;;-1:-1:-1;;;;;65609:17:0;;;65615:11;;65609:17;;65440:186;65423:531;;;65653:8;:15;;;;;;;;65749:9;;65731:12;;65653:15;;65706:53;;:38;;:20;;:24;:38::i;:::-;:42;;:53::i;:::-;65685:74;;65774:26;65789:10;65774:14;:26::i;:::-;65817:18;65838:24;65856:4;65838:9;:24::i;:::-;65817:45;;65877:32;65898:10;65877:20;:32::i;:::-;-1:-1:-1;;65926:8:0;:16;;;;;;65423:531;65968:12;65983:16;:29;;;;-1:-1:-1;66004:8:0;;;;;;;66003:9;65983:29;-1:-1:-1;;;;;66113:25:0;;;;;;:19;:25;;;;;;65968:44;;-1:-1:-1;66113:25:0;;;:52;;-1:-1:-1;;;;;;66142:23:0;;;;;;:19;:23;;;;;;;;66113:52;66110:99;;;-1:-1:-1;66192:5:0;66110:99;66224:7;66221:358;;;66248:12;66263:30;66289:3;66263:21;66274:9;;66263:6;:10;;:21;;;;:::i;:30::-;-1:-1:-1;;;;;66354:29:0;;;;;;:25;:29;;;;;;66248:45;;-1:-1:-1;66354:29:0;;66351:116;;;66411:40;66447:3;66411:31;66420:21;;66411:4;:8;;:31;;;;:::i;:40::-;66404:47;;66351:116;66492:16;:6;66503:4;66492:10;:16::i;:::-;66483:25;;66525:42;66541:4;66555;66562;66525:15;:42::i;:::-;66221:358;;66591:33;66607:4;66613:2;66617:6;66591:15;:33::i;:::-;66641:15;;-1:-1:-1;;;;;66641:15:0;:26;66676:4;66683:15;66676:4;66683:9;:15::i;:::-;66641:58;;;;;;;;;;;;;-1:-1:-1;;;;;66641:58:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;66637:74;66725:15;;-1:-1:-1;;;;;66725:15:0;:26;66760:2;66765:13;66760:2;66765:9;:13::i;:::-;66725:54;;;;;;;;;;;;;-1:-1:-1;;;;;66725:54:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;66721:70;66807:8;;;;;;;66803:357;;66846:16;;66883:15;;:28;;;;;;;;;;;;;;-1:-1:-1;;;;;66883:15:0;;;;:23;;:28;;;;;;;;;;;;;;;66832:11;66883:15;:28;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;66883:28:0;;;;;;;;;;;;;;;;66879:270;;;;;67009:86;;;;;;;;;;;;;;;;;;;;;;;;;;67085:9;;67074:4;;67009:86;;;;;;;;;66912:199;;;66879:270;66803:357;;63356:3811;;;;;;;;:::o;38242:166::-;38328:7;38364:12;38356:6;;;;38348:29;;;;-1:-1:-1;;;38348:29:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;;;38395:5:0;;;38242:166::o;60220:409::-;-1:-1:-1;;;;;60311:31:0;;;;;;:25;:31;;;;;;;;:40;;;;;;;60303:119;;;;-1:-1:-1;;;60303:119:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;;;;;60433:31:0;;;;;;:25;:31;;;;;:39;;;;;;;;;;;;;;;60485:79;;60510:15;;:42;;;;;;-1:-1:-1;;;;;60510:42:0;;;;;;;;;:15;;;;;:36;;:42;;;;;:15;;:42;;;;;;;:15;;:42;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;60485:79;60581:40;;;;;;-1:-1:-1;;;;;60581:40:0;;;;;;;;60220:409;;:::o;28778:539::-;-1:-1:-1;;;;;28884:20:0;;28876:70;;;;-1:-1:-1;;;28876:70:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;;;;;28965:23:0;;28957:71;;;;-1:-1:-1;;;28957:71:0;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;29041:47;29062:6;29070:9;29081:6;29041:20;:47::i;:::-;29121:71;29143:6;29121:71;;;;;;;;;;;;;;;;;-1:-1:-1;;;;;29121:17:0;;:9;:17;;;;;;;;;;;;:71;:21;:71::i;:::-;-1:-1:-1;;;;;29101:17:0;;;:9;:17;;;;;;;;;;;:91;;;;29226:20;;;;;;;:32;;29251:6;29226:24;:32::i;:::-;-1:-1:-1;;;;;29203:20:0;;;:9;:20;;;;;;;;;;;;:55;;;;29274:35;;;;;;;29203:20;;29274:35;;;;;;;;;;;;;28778:539;;;:::o;74034:960::-;74137:15;;74226:13;;-1:-1:-1;;;;;74137:15:0;;;;74099:27;;;;74226:13;:27;74223:611;;74269:42;74289:6;74305:4;74269:19;:42::i;:::-;74345:13;;74338:46;;;;;;74378:4;74338:46;;;;;;-1:-1:-1;;;;;74345:13:0;;;;74338:31;;:46;;;;;;;;;;;;;;;74345:13;74338:46;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;74338:46:0;74416:13;;74448:15;;74409:67;;;;;;-1:-1:-1;;;;;74448:15:0;;;74409:67;;;;;;;;;;;;74338:46;;-1:-1:-1;74416:13:0;;;74409:30;;:67;;;;;74338:46;;74409:67;;;;;;;;74416:13;;74409:67;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;74409:67:0;;-1:-1:-1;74223:611:0;;;74532:21;74568:24;74585:6;74568:16;:24::i;:::-;74607:18;74628:41;:21;74654:14;74628:25;:41::i;:::-;74754:39;;74607:62;;-1:-1:-1;74607:62:0;;-1:-1:-1;74722:9:0;;;;-1:-1:-1;;;;;74754:16:0;;;74607:62;;74722:9;74754:39;74722:9;74754:39;74607:62;74754:16;:39;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;74721:72:0;;-1:-1:-1;;;;;;74223:611:0;74848:7;74844:143;;;74872:15;;:51;;;;;;;;;;;;;;-1:-1:-1;;;;;74872:15:0;;;;:40;;:51;;;;;:15;;:51;;;;;;;;:15;;:51;;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;;74943:32:0;;;;;;;;;;;;;;;;-1:-1:-1;74943:32:0;;;;;;;;-1:-1:-1;74943:32:0;74034:960;;;;:::o;72779:663::-;72959:16;;;72973:1;72959:16;;;;;;;;;72935:21;;72959:16;;;;;;;;;;-1:-1:-1;72959:16:0;72935:40;;73004:4;72986;72991:1;72986:7;;;;;;;;-1:-1:-1;;;;;72986:23:0;;;:7;;;;;;;;;;:23;;;;73030:15;;:22;;;;;;;;:15;;;;;:20;;:22;;;;;72986:7;;73030:22;;;;;:15;:22;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;-1:-1:-1;73030:22:0;73020:7;;:4;;73025:1;;73020:7;;;;;;-1:-1:-1;;;;;73020:32:0;;;:7;;;;;;;;;:32;73073:13;;73063:7;;73073:13;;;73063:4;;73068:1;;73063:7;;;;;;-1:-1:-1;;;;;73063:23:0;;;:7;;;;;;;;;:23;73131:15;;73099:62;;73116:4;;73131:15;73149:11;73099:8;:62::i;:::-;73200:15;;:224;;;;;;;;;;;:15;:224;;;;;;-1:-1:-1;;;;;73200:224:0;;;;;;;73398:15;73200:224;;;;;;;;;;;;;;;;;;;;;:15;;;;;:69;;73284:11;;73355:4;;73374:9;;73398:15;73200:224;;;;;;;;;;;;;;:15;:224;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;

Swarm Source

ipfs://43bdd2b2c90e37a5c23bb70551b59ca84fad3c035a12762ff29c15c0bcfe9c8d

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.