ETH Price: $3,523.89 (+5.05%)

Token

AffeMitWaffe (AMW)
 

Overview

Max Total Supply

250 AMW

Holders

101

Market

Volume (24H)

N/A

Min Price (24H)

N/A

Max Price (24H)

N/A

Other Info

Balance
1 AMW
0x696fd103734c34cb150166e139da1cf5f4193525
Loading...
Loading
Loading...
Loading
Loading...
Loading

Click here to update the token information / general information
# Exchange Pair Price  24H Volume % Volume

Contract Source Code Verified (Exact Match)

Contract Name:
AffeMitWaffe

Compiler Version
v0.8.13+commit.abaa5c0e

Optimization Enabled:
Yes with 250 runs

Other Settings:
default evmVersion
File 1 of 21 : AffeMitWaffe.sol
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.13;

import "@openzeppelin/contracts/token/ERC721/ERC721.sol";
import "@openzeppelin/contracts/token/ERC721/extensions/ERC721Enumerable.sol";
import "@openzeppelin/contracts/security/Pausable.sol";
import "@openzeppelin/contracts/access/AccessControl.sol";
import "@openzeppelin/contracts/token/ERC721/extensions/ERC721Burnable.sol";
import "./URIManager.sol";
import "./ERC2981GlobalRoyalties.sol";
import "./ERC721Lending.sol";

//                                       .,*.                                   
//                                      *(//*                                   
//                              .      .//&/                                    
//                  .*.       (*/// , *(/,*  ,,,                                
//               ./&#*,(*.,   ,/((%./#(//#.//((#(###/**///#%#%(%(**             
//               //*/%&*#&/*/,.*/,*%##/*,(/*(/*#,,,*(#&(//                      
//                        (*#/(#/(*,        */%(/*/.#,,                         
//                      /#(.&(* //(/,/        ,,./**                            
//                    ./(%//,         .,*(/(((*/%%*.,                           
//                    */,.(/.   ,#%%,.      */.(*/*.                            
//                     *&%(%&%#&%(##%&(#   %*,(#&@/                             
//                      *#(/%*%(/*(%,*#%%(%%/##(%*.                             
//                        *.*/%,/,/*(,//%(#/#%,#*,                              
//                         . %/(%(#*##%#%(*,/(**                                
//                         ,(@##*&**##**/#/(/,                                  
//                          (,%#((/%((//(%/%.                                   
//                          , **,(/**/*/**((,                                   
//                           / (,..  ../*.*#,                                   
//                            ,(*##((%((#(/@,                                   
//                             (.%*( * .*(.&,                                   
//                              (/#%#/%#%(*%,                                   
//                             */,###(%%(*%#/                                   
//                           ,//#%**(//,,(((*.                                  
//                          ((*..**((#%/*(*//*,                                 
//                         ..%#,*((#.  ,.(,(/**                                 
//                          /...**,     .,/((/,                                 
//                          /. %*/(,     /.((%.*                                
//                            *#/,%./     ( #/,                                 
//                             */**       /.//#,                                
//                         (*#**,*#.      (**#&,                                
//                                     *,,/##,       

/**
 * @title Affe mit Waffe NFT smart contract.
 * @notice Implementation of ERC-721 standard for the genesis NFT of the Monkeyverse DAO.
 *   "Affe mit Waffe" is a symbiosis of artificial intelligence and the human mind. The
 *   artwork is a stencil inspired by graffiti culture and lays the foundation for creative
 *   development. The colors were decided by our AI – the Real Vision Bot. It determines
 *   the colors based on emotions obtained via natural language processing from the Real
 *   Vision interviews. Human creativity completes the piece for the finishing touch. Each
 *   Affe wants to connect, contrast, and stand out. Like the different colors and emotions
 *   of the day, the Affen are born to connect people, minds and ideas, countries, racesand
 *   genders through comparison and contrast.Despite their bossy appearance they are a
 *   happy hungry bunch at heart. They may look tough on the outside but are soft on the
 *   inside – and are easy to win over with a few bananas. The raised gun symbolizes our
 *   own strength and talents; it shall motivate us to use them wisely to overcome our
 *   differences for tolerance and resolve our conflicts peacefully.
 */

contract AffeMitWaffe is ERC721, ERC721Enumerable, Pausable, AccessControl,
                   ERC721Burnable, ERC2981GlobalRoyalties, URIManager, ERC721Lending {
    // Create the hashes that identify various roles. Note that the naming below diverges
    // from the naming of the DEFAULT_ADMIN_ROLE, whereby OpenZeppelin chose to put
    // the 'ROLE' part of the variable name at the end. Here, instead, all other roles are  named
    // with 'ROLE' at the beginning of the name, because this makes them much easier to
    // find and identify (they naturally get grouped together) in graphical tools like Remix
    // or Etherscan.
    bytes32 public constant ROLE_PAUSER = keccak256("ROLE_PAUSER");
    bytes32 public constant ROLE_MINTER = keccak256("ROLE_MINTER");
    bytes32 public constant ROLE_ROYALTY_SETTING = keccak256("ROLE_ROYALTY_SETTING");
    bytes32 public constant ROLE_METADATA_UPDATER = keccak256("ROLE_METADATA_UPDATER");
    bytes32 public constant ROLE_METADATA_FREEZER = keccak256("ROLE_METADATA_FREEZER");

    /**
     * @notice The owner variable below is 'honorary' in the sense that it serves no purpose
     *   as far as the smart contract itself is concerned. The only reason for implementing
     *   this variable, is that OpenSea queries owner() (according to an article in their Help
     *   Center) in order to decide who can login to the OpenSea interface and change
     *   collection-wide settings, such as the collection banner, or more importantly, royalty
     *   amount and destination (as of this writing, OpenSea implements their own royalty
     *   settings, rather than EIP-2981.)
     *   Semantically, for our purposes (because this contract uses AccessControl rather than
     *   Ownable) it would be more accurate to call this variable something like
     *   'openSeaCollectionAdmin' (but sadly OpenSea is looking for 'owner' specifically.)
     */
    address public owner;

    uint8 constant MAX_SUPPLY = 250;
    /**
     * @dev The variable below keeps track of the number of Affen that have been minted.
     *   HOWEVER, note that the variable is never decreased. Therefore, if an Affe is burned
     *   this does not allow for a new Affe to be minted. There will ever only be 250 MINTED.
     */
    uint8 public numTokensMinted;
    
    /**
     * @dev From our testing, it seems OpenSea will only honor a new collection-level administrator
     *   (the person who can login to the interface and, for example, change royalty
     *   amount/destination), if an event is emmitted (as coded in the OpenZeppelin Ownable contract)
     *   announcing the ownership transfer. Therefore, in order to ensure the OpenSea collection
     *   admin can be updated if ever needed, the following event has been included in this smart
     *   contract.
     */
    event OwnershipTransferred(address indexed previousOwner, address indexed newOwner);

    /**
     * @notice Constructor of the Affe mit Waffe ERC-721 NFT smart contract.
     * @param name is the name of the ERC-721 smart contract and NFT collection.
     * @param symbol is the symbol for the collection.
     * @param initialBaseURI is the base URI string that will concatenated with the tokenId to create
     *   the URI where each token's metadata can be found.
     * @param initialContractURI is the location where metadata about the collection as a whole
     *   can be found. For the most part it is an OpenSea-specific requirement (they will try
     *   to find metadata about the collection at this URI when the collecitons is initially
     *   imported into OpenSea.)
     */
    constructor(string memory name, string memory symbol, string memory initialBaseURI, string memory initialContractURI)
    ERC721(name, symbol)
    URIManager(initialBaseURI, initialContractURI) {
        // To start with we will only grant the DEFAULT_ADMIN_ROLE role to the msg.sender
        // The DEFAULT_ADMIN_ROLE is not granted any rights initially. The only privileges
        // the DEFAULT_ADMIN_ROLE has at contract deployment time are: the ability to grant other
        // roles, and the ability to set the 'honorary' contract owner (see comments above.)
        // For any functionality to be enabled, the DEFAULT_ADMIN_ROLE must explicitly grant those roles to
        // other accounts or to itself.
        _grantRole(DEFAULT_ADMIN_ROLE, msg.sender);
        setHonoraryOwner(msg.sender);
    }

    /**
     * @notice The 'honorary' portion of this function's name refers to the fact that the 'owner' variable
     *   serves no purpose in this smart contract itself. 'Ownership' is mostly meaningless in the context
     *   of a smart contract that implements security with RBAC (Role Based Access Control); so 'owndership'
     *   is only implemented here to allow for certain collection-wide admin functionality within the
     *   OpenSea web interface.
     * @param honoraryOwner is the address that one would like to designate as the 'owner' of this contract
     *   (most likely with the sole purpose of being able to login to OpenSea as an administrator of the
     *   collection.)
     */
    function setHonoraryOwner(address honoraryOwner) public onlyRole(DEFAULT_ADMIN_ROLE) {
        require(honoraryOwner != address(0), "New owner cannot be the zero address.");
        address priorOwner = owner;
        owner = honoraryOwner;
        emit OwnershipTransferred(priorOwner, honoraryOwner);
    }


    // Capabilities of ROLE_PAUSER

    /**
     * @notice A function which can be called externally by an acount with the
     *   ROLE_PAUSER, with the purpose of (in the case of an emergency) pausing all transfers
     *   of tokens in the contract (which includes minting/burning/transferring.)
     * @dev This function calls the internal _pause() function from
     *   OpenZeppelin's Pausable contract.
     */
    function pause() external onlyRole(ROLE_PAUSER) {
        _pause();
    }

    /**
     * @notice A function which can be called externally by an acount with the
     *   ROLE_PAUSER, with the purpose of UNpausing all transfers
     *   of tokens in the contract (which includes minting/burning/transferring.)
     * @dev This function calls the internal _unpause() function from
     *   OpenZeppelin's Pausable contract.
     */
    function unpause() external onlyRole(ROLE_PAUSER) {
        _unpause();
    }

    /**
     * @notice A function which can be called externally by an acount with the
     *   ROLE_PAUSER, with the purpose of pausing all token lending. When loans
     *   are paused, new loans cannot be made, but existing loans can be recalled.
     * @dev This function calls the internal _pauseLending() function of the
     *   ERC721Lending contract.
     */
    function pauseLending() external onlyRole(ROLE_PAUSER) {
        _pauseLending();
    }

    /**
     * @notice A function which can be called externally by an acount with the
     *   ROLE_PAUSER, with the purpose of UNpausing all token lending.
     * @dev This function calls the internal _unpauseLending() function of the
     *   ERC721Lending contract.
     */
    function unpauseLending() external onlyRole(ROLE_PAUSER) {
        _unpauseLending();
    }


    // Capabilities of ROLE_MINTER

    // the main minting function
    function safeMint(address to, uint256 tokenId) external onlyRole(ROLE_MINTER) {
        require(numTokensMinted < MAX_SUPPLY, "The maximum number of tokens that can ever be minted has been reached.");
        numTokensMinted += 1;
        _safeMint(to, tokenId);
    }


    // Capabilities of ROLE_ROYALTY_SETTING
    
    function setRoyaltyAmountInBips(uint16 newRoyaltyInBips) external onlyRole(ROLE_ROYALTY_SETTING) {
        _setRoyaltyAmountInBips(newRoyaltyInBips);
    }

    function setRoyaltyDestination(address newRoyaltyDestination) external onlyRole(ROLE_ROYALTY_SETTING) {
        _setRoyaltyDestination(newRoyaltyDestination);
    }


    // Capabilities of ROLE_METADATA_UPDATER

    function setBaseURI(string calldata newURI) external onlyRole(ROLE_METADATA_UPDATER) allowIfNotFrozen {
        _setBaseURI(newURI);
    }

    function setContractURI(string calldata newContractURI) external onlyRole(ROLE_METADATA_UPDATER) allowIfNotFrozen {
        _setContractURI(newContractURI);
    }

    
    // Capabilities of ROLE_METADATA_FREEZER

    function freezeURIsForever() external onlyRole(ROLE_METADATA_FREEZER) allowIfNotFrozen {
        _freezeURIsForever();
    }


    // Information fetching - external/public

    /**
     * @dev See {IERC721Metadata-tokenURI}.
     */
    function tokenURI(uint256 tokenId) public view override returns (string memory) {
        require(_exists(tokenId), "URI query for nonexistent token");
        return _buildTokenURI(tokenId);
    }

    function royaltyInfo(uint256 tokenId, uint256 salePrice)
        public
        view
        override
        returns (address, uint256)
    {
        require(_exists(tokenId), "Royalty requested for non-existing token");
        return _globalRoyaltyInfo(salePrice);
    }

    /**
     * @notice Returns all the token IDs owned by a given address. NOTE that 'owner',
     *   in this context, is the meaning as stipulated in EIP-721, which is the address
     *   returned by the ownerOf function. Therefore this function will enumerate the
     *   borrower as the current owner of a token on loan, rather than the original owner.
     * @param tokenOwner is the address to request ownership information about.
     * @return an array that has all the tokenIds owned by an address.
     */
    function ownedTokensByAddress(address tokenOwner) external view returns (uint256[] memory) {
        uint256 totalTokensOwned = balanceOf(tokenOwner);
        uint256[] memory allTokenIdsOfOwner = new uint256[](totalTokensOwned);
        for (uint256 i = 0; i < totalTokensOwned; i++) {
            allTokenIdsOfOwner[i] = (tokenOfOwnerByIndex(tokenOwner, i));
        }
        return allTokenIdsOfOwner;
    }

    /**
     * @notice Function retrieves the specific token ids on loan by a given address.
     * @param rightfulOwner is the original/rightful owner for whom one wishes to find the
     *   tokenIds on loan.
     * @return an array with the tokenIds currently on loan by the origina/rightful owner.
     */
    function loanedTokensByAddress(address rightfulOwner) external view returns (uint256[] memory) {
        require(rightfulOwner != address(0), "ERC721Lending: Balance query for the zero address");
        uint256 numTokensLoanedByRightfulOwner = loanedBalanceOf(rightfulOwner);
        uint256 numGlobalTotalTokens = totalSupply();
        uint256 nextTokenIdToQuery;

        uint256[] memory theTokenIDsOfRightfulOwner = new uint256[](numTokensLoanedByRightfulOwner);
        // If the address in question hasn't lent any tokens, there is no reason to enter the loop.
        if (numTokensLoanedByRightfulOwner > 0) {
            uint256 numMatchingTokensFound = 0;
            // Continue searching in the loop until either all tokens in the collection have been examined
            // or the number of tokens being searched for (the number owned originally by the rightful
            // owner) have been found.
            for (uint256 i = 0; numMatchingTokensFound < numTokensLoanedByRightfulOwner && i < numGlobalTotalTokens; i++) {
                // TokenIds may not be sequential or even within a specific range, so we get the next tokenId (to 
                // lookup in the mapping) from the global array holding all tokens.
                nextTokenIdToQuery = tokenByIndex(i);
                if (mapFromTokenIdToRightfulOwner[nextTokenIdToQuery] == rightfulOwner) {
                    theTokenIDsOfRightfulOwner[numMatchingTokensFound] = nextTokenIdToQuery;
                    numMatchingTokensFound++;
                }
            }
        }
        return theTokenIDsOfRightfulOwner;
    }


    // Hook overrides

    function _beforeTokenTransfer(address from, address to, uint256 tokenId)
        internal
        whenNotPaused
        override(ERC721, ERC721Enumerable, ERC721Lending)
    {
        super._beforeTokenTransfer(from, to, tokenId);
    }


    // The following functions are overrides required by Solidity.

    function supportsInterface(bytes4 interfaceId)
        public
        view
        override(ERC721, ERC721Enumerable, AccessControl, ERC2981GlobalRoyalties)
        returns (bool)
    {
        return super.supportsInterface(interfaceId);
    }

}

File 2 of 21 : Pausable.sol
// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts (last updated v4.7.0) (security/Pausable.sol)

pragma solidity ^0.8.0;

import "../utils/Context.sol";

/**
 * @dev Contract module which allows children to implement an emergency stop
 * mechanism that can be triggered by an authorized account.
 *
 * This module is used through inheritance. It will make available the
 * modifiers `whenNotPaused` and `whenPaused`, which can be applied to
 * the functions of your contract. Note that they will not be pausable by
 * simply including this module, only once the modifiers are put in place.
 */
abstract contract Pausable is Context {
    /**
     * @dev Emitted when the pause is triggered by `account`.
     */
    event Paused(address account);

    /**
     * @dev Emitted when the pause is lifted by `account`.
     */
    event Unpaused(address account);

    bool private _paused;

    /**
     * @dev Initializes the contract in unpaused state.
     */
    constructor() {
        _paused = false;
    }

    /**
     * @dev Modifier to make a function callable only when the contract is not paused.
     *
     * Requirements:
     *
     * - The contract must not be paused.
     */
    modifier whenNotPaused() {
        _requireNotPaused();
        _;
    }

    /**
     * @dev Modifier to make a function callable only when the contract is paused.
     *
     * Requirements:
     *
     * - The contract must be paused.
     */
    modifier whenPaused() {
        _requirePaused();
        _;
    }

    /**
     * @dev Returns true if the contract is paused, and false otherwise.
     */
    function paused() public view virtual returns (bool) {
        return _paused;
    }

    /**
     * @dev Throws if the contract is paused.
     */
    function _requireNotPaused() internal view virtual {
        require(!paused(), "Pausable: paused");
    }

    /**
     * @dev Throws if the contract is not paused.
     */
    function _requirePaused() internal view virtual {
        require(paused(), "Pausable: not paused");
    }

    /**
     * @dev Triggers stopped state.
     *
     * Requirements:
     *
     * - The contract must not be paused.
     */
    function _pause() internal virtual whenNotPaused {
        _paused = true;
        emit Paused(_msgSender());
    }

    /**
     * @dev Returns to normal state.
     *
     * Requirements:
     *
     * - The contract must be paused.
     */
    function _unpause() internal virtual whenPaused {
        _paused = false;
        emit Unpaused(_msgSender());
    }
}

File 3 of 21 : AccessControl.sol
// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts (last updated v4.7.0) (access/AccessControl.sol)

pragma solidity ^0.8.0;

import "./IAccessControl.sol";
import "../utils/Context.sol";
import "../utils/Strings.sol";
import "../utils/introspection/ERC165.sol";

/**
 * @dev Contract module that allows children to implement role-based access
 * control mechanisms. This is a lightweight version that doesn't allow enumerating role
 * members except through off-chain means by accessing the contract event logs. Some
 * applications may benefit from on-chain enumerability, for those cases see
 * {AccessControlEnumerable}.
 *
 * Roles are referred to by their `bytes32` identifier. These should be exposed
 * in the external API and be unique. The best way to achieve this is by
 * using `public constant` hash digests:
 *
 * ```
 * bytes32 public constant MY_ROLE = keccak256("MY_ROLE");
 * ```
 *
 * Roles can be used to represent a set of permissions. To restrict access to a
 * function call, use {hasRole}:
 *
 * ```
 * function foo() public {
 *     require(hasRole(MY_ROLE, msg.sender));
 *     ...
 * }
 * ```
 *
 * Roles can be granted and revoked dynamically via the {grantRole} and
 * {revokeRole} functions. Each role has an associated admin role, and only
 * accounts that have a role's admin role can call {grantRole} and {revokeRole}.
 *
 * By default, the admin role for all roles is `DEFAULT_ADMIN_ROLE`, which means
 * that only accounts with this role will be able to grant or revoke other
 * roles. More complex role relationships can be created by using
 * {_setRoleAdmin}.
 *
 * WARNING: The `DEFAULT_ADMIN_ROLE` is also its own admin: it has permission to
 * grant and revoke this role. Extra precautions should be taken to secure
 * accounts that have been granted it.
 */
abstract contract AccessControl is Context, IAccessControl, ERC165 {
    struct RoleData {
        mapping(address => bool) members;
        bytes32 adminRole;
    }

    mapping(bytes32 => RoleData) private _roles;

    bytes32 public constant DEFAULT_ADMIN_ROLE = 0x00;

    /**
     * @dev Modifier that checks that an account has a specific role. Reverts
     * with a standardized message including the required role.
     *
     * The format of the revert reason is given by the following regular expression:
     *
     *  /^AccessControl: account (0x[0-9a-f]{40}) is missing role (0x[0-9a-f]{64})$/
     *
     * _Available since v4.1._
     */
    modifier onlyRole(bytes32 role) {
        _checkRole(role);
        _;
    }

    /**
     * @dev See {IERC165-supportsInterface}.
     */
    function supportsInterface(bytes4 interfaceId) public view virtual override returns (bool) {
        return interfaceId == type(IAccessControl).interfaceId || super.supportsInterface(interfaceId);
    }

    /**
     * @dev Returns `true` if `account` has been granted `role`.
     */
    function hasRole(bytes32 role, address account) public view virtual override returns (bool) {
        return _roles[role].members[account];
    }

    /**
     * @dev Revert with a standard message if `_msgSender()` is missing `role`.
     * Overriding this function changes the behavior of the {onlyRole} modifier.
     *
     * Format of the revert message is described in {_checkRole}.
     *
     * _Available since v4.6._
     */
    function _checkRole(bytes32 role) internal view virtual {
        _checkRole(role, _msgSender());
    }

    /**
     * @dev Revert with a standard message if `account` is missing `role`.
     *
     * The format of the revert reason is given by the following regular expression:
     *
     *  /^AccessControl: account (0x[0-9a-f]{40}) is missing role (0x[0-9a-f]{64})$/
     */
    function _checkRole(bytes32 role, address account) internal view virtual {
        if (!hasRole(role, account)) {
            revert(
                string(
                    abi.encodePacked(
                        "AccessControl: account ",
                        Strings.toHexString(uint160(account), 20),
                        " is missing role ",
                        Strings.toHexString(uint256(role), 32)
                    )
                )
            );
        }
    }

    /**
     * @dev Returns the admin role that controls `role`. See {grantRole} and
     * {revokeRole}.
     *
     * To change a role's admin, use {_setRoleAdmin}.
     */
    function getRoleAdmin(bytes32 role) public view virtual override returns (bytes32) {
        return _roles[role].adminRole;
    }

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

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

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

        _revokeRole(role, account);
    }

    /**
     * @dev Grants `role` to `account`.
     *
     * If `account` had not been already granted `role`, emits a {RoleGranted}
     * event. Note that unlike {grantRole}, this function doesn't perform any
     * checks on the calling account.
     *
     * May emit a {RoleGranted} event.
     *
     * [WARNING]
     * ====
     * This function should only be called from the constructor when setting
     * up the initial roles for the system.
     *
     * Using this function in any other way is effectively circumventing the admin
     * system imposed by {AccessControl}.
     * ====
     *
     * NOTE: This function is deprecated in favor of {_grantRole}.
     */
    function _setupRole(bytes32 role, address account) internal virtual {
        _grantRole(role, account);
    }

    /**
     * @dev Sets `adminRole` as ``role``'s admin role.
     *
     * Emits a {RoleAdminChanged} event.
     */
    function _setRoleAdmin(bytes32 role, bytes32 adminRole) internal virtual {
        bytes32 previousAdminRole = getRoleAdmin(role);
        _roles[role].adminRole = adminRole;
        emit RoleAdminChanged(role, previousAdminRole, adminRole);
    }

    /**
     * @dev Grants `role` to `account`.
     *
     * Internal function without access restriction.
     *
     * May emit a {RoleGranted} event.
     */
    function _grantRole(bytes32 role, address account) internal virtual {
        if (!hasRole(role, account)) {
            _roles[role].members[account] = true;
            emit RoleGranted(role, account, _msgSender());
        }
    }

    /**
     * @dev Revokes `role` from `account`.
     *
     * Internal function without access restriction.
     *
     * May emit a {RoleRevoked} event.
     */
    function _revokeRole(bytes32 role, address account) internal virtual {
        if (hasRole(role, account)) {
            _roles[role].members[account] = false;
            emit RoleRevoked(role, account, _msgSender());
        }
    }
}

File 4 of 21 : ERC721.sol
// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts (last updated v4.7.0) (token/ERC721/ERC721.sol)

pragma solidity ^0.8.0;

import "./IERC721.sol";
import "./IERC721Receiver.sol";
import "./extensions/IERC721Metadata.sol";
import "../../utils/Address.sol";
import "../../utils/Context.sol";
import "../../utils/Strings.sol";
import "../../utils/introspection/ERC165.sol";

/**
 * @dev Implementation of https://eips.ethereum.org/EIPS/eip-721[ERC721] Non-Fungible Token Standard, including
 * the Metadata extension, but not including the Enumerable extension, which is available separately as
 * {ERC721Enumerable}.
 */
contract ERC721 is Context, ERC165, IERC721, IERC721Metadata {
    using Address for address;
    using Strings for uint256;

    // Token name
    string private _name;

    // Token symbol
    string private _symbol;

    // Mapping from token ID to owner address
    mapping(uint256 => address) private _owners;

    // Mapping owner address to token count
    mapping(address => uint256) private _balances;

    // Mapping from token ID to approved address
    mapping(uint256 => address) private _tokenApprovals;

    // Mapping from owner to operator approvals
    mapping(address => mapping(address => bool)) private _operatorApprovals;

    /**
     * @dev Initializes the contract by setting a `name` and a `symbol` to the token collection.
     */
    constructor(string memory name_, string memory symbol_) {
        _name = name_;
        _symbol = symbol_;
    }

    /**
     * @dev See {IERC165-supportsInterface}.
     */
    function supportsInterface(bytes4 interfaceId) public view virtual override(ERC165, IERC165) returns (bool) {
        return
            interfaceId == type(IERC721).interfaceId ||
            interfaceId == type(IERC721Metadata).interfaceId ||
            super.supportsInterface(interfaceId);
    }

    /**
     * @dev See {IERC721-balanceOf}.
     */
    function balanceOf(address owner) public view virtual override returns (uint256) {
        require(owner != address(0), "ERC721: address zero is not a valid owner");
        return _balances[owner];
    }

    /**
     * @dev See {IERC721-ownerOf}.
     */
    function ownerOf(uint256 tokenId) public view virtual override returns (address) {
        address owner = _owners[tokenId];
        require(owner != address(0), "ERC721: invalid token ID");
        return owner;
    }

    /**
     * @dev See {IERC721Metadata-name}.
     */
    function name() public view virtual override returns (string memory) {
        return _name;
    }

    /**
     * @dev See {IERC721Metadata-symbol}.
     */
    function symbol() public view virtual override returns (string memory) {
        return _symbol;
    }

    /**
     * @dev See {IERC721Metadata-tokenURI}.
     */
    function tokenURI(uint256 tokenId) public view virtual override returns (string memory) {
        _requireMinted(tokenId);

        string memory baseURI = _baseURI();
        return bytes(baseURI).length > 0 ? string(abi.encodePacked(baseURI, tokenId.toString())) : "";
    }

    /**
     * @dev Base URI for computing {tokenURI}. If set, the resulting URI for each
     * token will be the concatenation of the `baseURI` and the `tokenId`. Empty
     * by default, can be overridden in child contracts.
     */
    function _baseURI() internal view virtual returns (string memory) {
        return "";
    }

    /**
     * @dev See {IERC721-approve}.
     */
    function approve(address to, uint256 tokenId) public virtual override {
        address owner = ERC721.ownerOf(tokenId);
        require(to != owner, "ERC721: approval to current owner");

        require(
            _msgSender() == owner || isApprovedForAll(owner, _msgSender()),
            "ERC721: approve caller is not token owner nor approved for all"
        );

        _approve(to, tokenId);
    }

    /**
     * @dev See {IERC721-getApproved}.
     */
    function getApproved(uint256 tokenId) public view virtual override returns (address) {
        _requireMinted(tokenId);

        return _tokenApprovals[tokenId];
    }

    /**
     * @dev See {IERC721-setApprovalForAll}.
     */
    function setApprovalForAll(address operator, bool approved) public virtual override {
        _setApprovalForAll(_msgSender(), operator, approved);
    }

    /**
     * @dev See {IERC721-isApprovedForAll}.
     */
    function isApprovedForAll(address owner, address operator) public view virtual override returns (bool) {
        return _operatorApprovals[owner][operator];
    }

    /**
     * @dev See {IERC721-transferFrom}.
     */
    function transferFrom(
        address from,
        address to,
        uint256 tokenId
    ) public virtual override {
        //solhint-disable-next-line max-line-length
        require(_isApprovedOrOwner(_msgSender(), tokenId), "ERC721: caller is not token owner nor approved");

        _transfer(from, to, tokenId);
    }

    /**
     * @dev See {IERC721-safeTransferFrom}.
     */
    function safeTransferFrom(
        address from,
        address to,
        uint256 tokenId
    ) public virtual override {
        safeTransferFrom(from, to, tokenId, "");
    }

    /**
     * @dev See {IERC721-safeTransferFrom}.
     */
    function safeTransferFrom(
        address from,
        address to,
        uint256 tokenId,
        bytes memory data
    ) public virtual override {
        require(_isApprovedOrOwner(_msgSender(), tokenId), "ERC721: caller is not token owner nor approved");
        _safeTransfer(from, to, tokenId, data);
    }

    /**
     * @dev Safely transfers `tokenId` token from `from` to `to`, checking first that contract recipients
     * are aware of the ERC721 protocol to prevent tokens from being forever locked.
     *
     * `data` is additional data, it has no specified format and it is sent in call to `to`.
     *
     * This internal function is equivalent to {safeTransferFrom}, and can be used to e.g.
     * implement alternative mechanisms to perform token transfer, such as signature-based.
     *
     * Requirements:
     *
     * - `from` cannot be the zero address.
     * - `to` cannot be the zero address.
     * - `tokenId` token must exist and be owned by `from`.
     * - If `to` refers to a smart contract, it must implement {IERC721Receiver-onERC721Received}, which is called upon a safe transfer.
     *
     * Emits a {Transfer} event.
     */
    function _safeTransfer(
        address from,
        address to,
        uint256 tokenId,
        bytes memory data
    ) internal virtual {
        _transfer(from, to, tokenId);
        require(_checkOnERC721Received(from, to, tokenId, data), "ERC721: transfer to non ERC721Receiver implementer");
    }

    /**
     * @dev Returns whether `tokenId` exists.
     *
     * Tokens can be managed by their owner or approved accounts via {approve} or {setApprovalForAll}.
     *
     * Tokens start existing when they are minted (`_mint`),
     * and stop existing when they are burned (`_burn`).
     */
    function _exists(uint256 tokenId) internal view virtual returns (bool) {
        return _owners[tokenId] != address(0);
    }

    /**
     * @dev Returns whether `spender` is allowed to manage `tokenId`.
     *
     * Requirements:
     *
     * - `tokenId` must exist.
     */
    function _isApprovedOrOwner(address spender, uint256 tokenId) internal view virtual returns (bool) {
        address owner = ERC721.ownerOf(tokenId);
        return (spender == owner || isApprovedForAll(owner, spender) || getApproved(tokenId) == spender);
    }

    /**
     * @dev Safely mints `tokenId` and transfers it to `to`.
     *
     * Requirements:
     *
     * - `tokenId` must not exist.
     * - If `to` refers to a smart contract, it must implement {IERC721Receiver-onERC721Received}, which is called upon a safe transfer.
     *
     * Emits a {Transfer} event.
     */
    function _safeMint(address to, uint256 tokenId) internal virtual {
        _safeMint(to, tokenId, "");
    }

    /**
     * @dev Same as {xref-ERC721-_safeMint-address-uint256-}[`_safeMint`], with an additional `data` parameter which is
     * forwarded in {IERC721Receiver-onERC721Received} to contract recipients.
     */
    function _safeMint(
        address to,
        uint256 tokenId,
        bytes memory data
    ) internal virtual {
        _mint(to, tokenId);
        require(
            _checkOnERC721Received(address(0), to, tokenId, data),
            "ERC721: transfer to non ERC721Receiver implementer"
        );
    }

    /**
     * @dev Mints `tokenId` and transfers it to `to`.
     *
     * WARNING: Usage of this method is discouraged, use {_safeMint} whenever possible
     *
     * Requirements:
     *
     * - `tokenId` must not exist.
     * - `to` cannot be the zero address.
     *
     * Emits a {Transfer} event.
     */
    function _mint(address to, uint256 tokenId) internal virtual {
        require(to != address(0), "ERC721: mint to the zero address");
        require(!_exists(tokenId), "ERC721: token already minted");

        _beforeTokenTransfer(address(0), to, tokenId);

        _balances[to] += 1;
        _owners[tokenId] = to;

        emit Transfer(address(0), to, tokenId);

        _afterTokenTransfer(address(0), to, tokenId);
    }

    /**
     * @dev Destroys `tokenId`.
     * The approval is cleared when the token is burned.
     *
     * Requirements:
     *
     * - `tokenId` must exist.
     *
     * Emits a {Transfer} event.
     */
    function _burn(uint256 tokenId) internal virtual {
        address owner = ERC721.ownerOf(tokenId);

        _beforeTokenTransfer(owner, address(0), tokenId);

        // Clear approvals
        _approve(address(0), tokenId);

        _balances[owner] -= 1;
        delete _owners[tokenId];

        emit Transfer(owner, address(0), tokenId);

        _afterTokenTransfer(owner, address(0), tokenId);
    }

    /**
     * @dev Transfers `tokenId` from `from` to `to`.
     *  As opposed to {transferFrom}, this imposes no restrictions on msg.sender.
     *
     * Requirements:
     *
     * - `to` cannot be the zero address.
     * - `tokenId` token must be owned by `from`.
     *
     * Emits a {Transfer} event.
     */
    function _transfer(
        address from,
        address to,
        uint256 tokenId
    ) internal virtual {
        require(ERC721.ownerOf(tokenId) == from, "ERC721: transfer from incorrect owner");
        require(to != address(0), "ERC721: transfer to the zero address");

        _beforeTokenTransfer(from, to, tokenId);

        // Clear approvals from the previous owner
        _approve(address(0), tokenId);

        _balances[from] -= 1;
        _balances[to] += 1;
        _owners[tokenId] = to;

        emit Transfer(from, to, tokenId);

        _afterTokenTransfer(from, to, tokenId);
    }

    /**
     * @dev Approve `to` to operate on `tokenId`
     *
     * Emits an {Approval} event.
     */
    function _approve(address to, uint256 tokenId) internal virtual {
        _tokenApprovals[tokenId] = to;
        emit Approval(ERC721.ownerOf(tokenId), to, tokenId);
    }

    /**
     * @dev Approve `operator` to operate on all of `owner` tokens
     *
     * Emits an {ApprovalForAll} event.
     */
    function _setApprovalForAll(
        address owner,
        address operator,
        bool approved
    ) internal virtual {
        require(owner != operator, "ERC721: approve to caller");
        _operatorApprovals[owner][operator] = approved;
        emit ApprovalForAll(owner, operator, approved);
    }

    /**
     * @dev Reverts if the `tokenId` has not been minted yet.
     */
    function _requireMinted(uint256 tokenId) internal view virtual {
        require(_exists(tokenId), "ERC721: invalid token ID");
    }

    /**
     * @dev Internal function to invoke {IERC721Receiver-onERC721Received} on a target address.
     * The call is not executed if the target address is not a contract.
     *
     * @param from address representing the previous owner of the given token ID
     * @param to target address that will receive the tokens
     * @param tokenId uint256 ID of the token to be transferred
     * @param data bytes optional data to send along with the call
     * @return bool whether the call correctly returned the expected magic value
     */
    function _checkOnERC721Received(
        address from,
        address to,
        uint256 tokenId,
        bytes memory data
    ) private returns (bool) {
        if (to.isContract()) {
            try IERC721Receiver(to).onERC721Received(_msgSender(), from, tokenId, data) returns (bytes4 retval) {
                return retval == IERC721Receiver.onERC721Received.selector;
            } catch (bytes memory reason) {
                if (reason.length == 0) {
                    revert("ERC721: transfer to non ERC721Receiver implementer");
                } else {
                    /// @solidity memory-safe-assembly
                    assembly {
                        revert(add(32, reason), mload(reason))
                    }
                }
            }
        } else {
            return true;
        }
    }

    /**
     * @dev Hook that is called before any token transfer. This includes minting
     * and burning.
     *
     * Calling conditions:
     *
     * - When `from` and `to` are both non-zero, ``from``'s `tokenId` will be
     * transferred to `to`.
     * - When `from` is zero, `tokenId` will be minted for `to`.
     * - When `to` is zero, ``from``'s `tokenId` 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 tokenId
    ) internal virtual {}

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

File 5 of 21 : ERC721Enumerable.sol
// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts v4.4.1 (token/ERC721/extensions/ERC721Enumerable.sol)

pragma solidity ^0.8.0;

import "../ERC721.sol";
import "./IERC721Enumerable.sol";

/**
 * @dev This implements an optional extension of {ERC721} defined in the EIP that adds
 * enumerability of all the token ids in the contract as well as all token ids owned by each
 * account.
 */
abstract contract ERC721Enumerable is ERC721, IERC721Enumerable {
    // Mapping from owner to list of owned token IDs
    mapping(address => mapping(uint256 => uint256)) private _ownedTokens;

    // Mapping from token ID to index of the owner tokens list
    mapping(uint256 => uint256) private _ownedTokensIndex;

    // Array with all token ids, used for enumeration
    uint256[] private _allTokens;

    // Mapping from token id to position in the allTokens array
    mapping(uint256 => uint256) private _allTokensIndex;

    /**
     * @dev See {IERC165-supportsInterface}.
     */
    function supportsInterface(bytes4 interfaceId) public view virtual override(IERC165, ERC721) returns (bool) {
        return interfaceId == type(IERC721Enumerable).interfaceId || super.supportsInterface(interfaceId);
    }

    /**
     * @dev See {IERC721Enumerable-tokenOfOwnerByIndex}.
     */
    function tokenOfOwnerByIndex(address owner, uint256 index) public view virtual override returns (uint256) {
        require(index < ERC721.balanceOf(owner), "ERC721Enumerable: owner index out of bounds");
        return _ownedTokens[owner][index];
    }

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

    /**
     * @dev See {IERC721Enumerable-tokenByIndex}.
     */
    function tokenByIndex(uint256 index) public view virtual override returns (uint256) {
        require(index < ERC721Enumerable.totalSupply(), "ERC721Enumerable: global index out of bounds");
        return _allTokens[index];
    }

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

        if (from == address(0)) {
            _addTokenToAllTokensEnumeration(tokenId);
        } else if (from != to) {
            _removeTokenFromOwnerEnumeration(from, tokenId);
        }
        if (to == address(0)) {
            _removeTokenFromAllTokensEnumeration(tokenId);
        } else if (to != from) {
            _addTokenToOwnerEnumeration(to, tokenId);
        }
    }

    /**
     * @dev Private function to add a token to this extension's ownership-tracking data structures.
     * @param to address representing the new owner of the given token ID
     * @param tokenId uint256 ID of the token to be added to the tokens list of the given address
     */
    function _addTokenToOwnerEnumeration(address to, uint256 tokenId) private {
        uint256 length = ERC721.balanceOf(to);
        _ownedTokens[to][length] = tokenId;
        _ownedTokensIndex[tokenId] = length;
    }

    /**
     * @dev Private function to add a token to this extension's token tracking data structures.
     * @param tokenId uint256 ID of the token to be added to the tokens list
     */
    function _addTokenToAllTokensEnumeration(uint256 tokenId) private {
        _allTokensIndex[tokenId] = _allTokens.length;
        _allTokens.push(tokenId);
    }

    /**
     * @dev Private function to remove a token from this extension's ownership-tracking data structures. Note that
     * while the token is not assigned a new owner, the `_ownedTokensIndex` mapping is _not_ updated: this allows for
     * gas optimizations e.g. when performing a transfer operation (avoiding double writes).
     * This has O(1) time complexity, but alters the order of the _ownedTokens array.
     * @param from address representing the previous owner of the given token ID
     * @param tokenId uint256 ID of the token to be removed from the tokens list of the given address
     */
    function _removeTokenFromOwnerEnumeration(address from, uint256 tokenId) private {
        // To prevent a gap in from's tokens array, we store the last token in the index of the token to delete, and
        // then delete the last slot (swap and pop).

        uint256 lastTokenIndex = ERC721.balanceOf(from) - 1;
        uint256 tokenIndex = _ownedTokensIndex[tokenId];

        // When the token to delete is the last token, the swap operation is unnecessary
        if (tokenIndex != lastTokenIndex) {
            uint256 lastTokenId = _ownedTokens[from][lastTokenIndex];

            _ownedTokens[from][tokenIndex] = lastTokenId; // Move the last token to the slot of the to-delete token
            _ownedTokensIndex[lastTokenId] = tokenIndex; // Update the moved token's index
        }

        // This also deletes the contents at the last position of the array
        delete _ownedTokensIndex[tokenId];
        delete _ownedTokens[from][lastTokenIndex];
    }

    /**
     * @dev Private function to remove a token from this extension's token tracking data structures.
     * This has O(1) time complexity, but alters the order of the _allTokens array.
     * @param tokenId uint256 ID of the token to be removed from the tokens list
     */
    function _removeTokenFromAllTokensEnumeration(uint256 tokenId) private {
        // To prevent a gap in the tokens array, we store the last token in the index of the token to delete, and
        // then delete the last slot (swap and pop).

        uint256 lastTokenIndex = _allTokens.length - 1;
        uint256 tokenIndex = _allTokensIndex[tokenId];

        // When the token to delete is the last token, the swap operation is unnecessary. However, since this occurs so
        // rarely (when the last minted token is burnt) that we still do the swap here to avoid the gas cost of adding
        // an 'if' statement (like in _removeTokenFromOwnerEnumeration)
        uint256 lastTokenId = _allTokens[lastTokenIndex];

        _allTokens[tokenIndex] = lastTokenId; // Move the last token to the slot of the to-delete token
        _allTokensIndex[lastTokenId] = tokenIndex; // Update the moved token's index

        // This also deletes the contents at the last position of the array
        delete _allTokensIndex[tokenId];
        _allTokens.pop();
    }
}

File 6 of 21 : ERC721Lending.sol
// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts v4.4.1 (token/ERC721/extensions/ERC721URIStorage.sol)

pragma solidity ^0.8.13;

import "@openzeppelin/contracts/token/ERC721/ERC721.sol";
import "@openzeppelin/contracts/security/ReentrancyGuard.sol";

/**
 * @notice Implementation of ERC-721 NFT lending. The code below was written by using, as a
 *   starting point, the code made public by the Meta Angels NFT team (thank you to that team
 *   for making their code available for other projects to use!)
 *   The code has been modified in several ways, most importantly, that in the original
 *   implementation it was included in the main contract, whereas here we have abstracted the
 *   functionality into its own parent contract. Also, some additional events have been added,
 *   and checking whether loans are paused has been moved to a Modifier. In addition a function
 *   has been added to allow a borrower to initiate the return of a loan (rather than only 
 *   allowing for the original lender to 'recall' the loan.)
 *   Note that when lending, the meaning of terms like 'owner' become ambiguous, particularly
 *   because once a token is lent, as far as the ERC721 standard is concerned, the borrower is
 *   technically the owner. (In other words, the function 'ownerOf()' reqired by EIP-721 will
 *   return the address of the borrower while a token is lent. In the comments and variable names
 *   below we have tried to disambiguate by refering wherever possible to the original/rightful
 *   owner as the address that truly owns the NFT (the address that is able to recall the loan
 *   whenever they want.) However it is important to understand that once a token is loaned, to
 *   the outside world it will appear to be 'owned' by the borrower. From that perspective, the
 *   'owner' is the current borrower.
 * @dev if you would like to use this code and add a function that enumerates the tokens
 *   loaned out by a particular address (eg. it could be a function called
 *   loanedTokensByAddress(address rightfulOwner) ), you'll need to modify this contract so it
 *   inherits from ERC721Enumerable (because such a function will need access to the
 *   'totalSupply()' provided by the Enumerable contract. For the sake of simplicity, this
 *   contract does not currently implement a function that generates the enumeration of loaned
 *   tokens. However, note that a child contract can readily implement such a function, if it
 *   inherits from ERC721Enumerable.
 */
abstract contract ERC721Lending is ERC721, ReentrancyGuard {
    using Strings for uint256;

    mapping (address => uint256) public totalLoanedPerAddress;
    /**
    * @notice The mapping below keeps track of the original/rightful owner of each token, in other words,
    *   the address that truly owns the token (and has simply lent it out.) This is the address
    *   that is allowed to retrieve the token (to end the loan.)
    */
    mapping (uint256 => address) public mapFromTokenIdToRightfulOwner;
    uint256 internal counterGlobalLoans = 0;

    /**
     * @notice A variable that servers two purposes. 1) To allow the 'outside world' to easily query
     *   whether lendig is currently paused (or not), and 2) to hold the current state so that
     *   certain parts of the code can make decisons about the actions that are allowed (or not.)
     *   NOTE that when lending is paused, this restricts NEW loans from happening, but it does not
     *   restrict owners from reclaiming their loans, or from borrowers returning their borrowed tokens.
     */
    bool public loansAreCurrentlyPaused = false;

    /**
     * @notice Emitted when a loan is made.
     * @param from is the owner of the token (who is making the loan.)
     * @param to is the recipient of the loan.
     * @param item is the tokenID representing the token being lent.
     */
    event Loan(address indexed from, address indexed to, uint item);
    /**
     * @notice Emitted when a loan is recalled by its rightful/original owner.
     * @param byOriginalOwner is the original and rightful owner of the token.
     * @param fromBorrower is the address the token was lent out to.
     * @param item is the tokenID representing the token that was lent.
     */
    event LoanReclaimed(address indexed byOriginalOwner, address indexed fromBorrower, uint item);
    /**
     * @notice Emitted when a loan is returned by the borrower.
     * @param byBorrower is the address that token has been lent to.
     * @param toOriginalOwner is the original and rightful owner of the token.
     * @param item is the tokenID representing the token that was lent.
     */
    event LoanReturned(address indexed byBorrower, address indexed toOriginalOwner, uint item);
    /**
     * @notice Emitted when the pausing of loans is triggered.
     * @param account is the address that paused lending.
     */
    event LendingPaused(address account);
    /**
     * @notice Emitted when UNpausing of loans is triggered.
     * @param account is the address that UNpaused lending.
     */
    event LendingUnpaused(address account);


    /**
     * @notice Enables an owner to loan one of their tokens to another address. The loan is effectively
     *   a complete transfer of ownership. However, what makes it a 'loan' are a set of checks that do
     *   not allow the new owner to do certain things (such as further transfers of the token), and the
     *   ability of the lender to recall the token back into their ownership.
     * @param tokenId is the integer ID of the token to loan.
     * @param receiver is the address that the token will be loaned to.
     */
    function loan(address receiver, uint256 tokenId) external nonReentrant allowIfLendingNotPaused {
        require(msg.sender == ownerOf(tokenId), "ERC721Lending: Trying to lend a token that is not owned.");
        require(msg.sender != receiver, "ERC721Lending: Lending to self (the current owner's address) is not permitted.");
        require(receiver != address(0), "ERC721Lending: Loans to the zero 0x0 address are not permitted.");
        require(mapFromTokenIdToRightfulOwner[tokenId] == address(0), "ERC721Lending: Trying to lend a token that is already on loan.");

        // Transfer the token
        safeTransferFrom(msg.sender, receiver, tokenId);

        // Add it to the mapping (of loaned tokens, and who their original/rightful owners are.)
        mapFromTokenIdToRightfulOwner[tokenId] = msg.sender;

        // Add to the owner's loan balance
        uint256 loansByAddress = totalLoanedPerAddress[msg.sender];
        totalLoanedPerAddress[msg.sender] = loansByAddress + 1;
        counterGlobalLoans = counterGlobalLoans + 1;

        emit Loan(msg.sender, receiver, tokenId);
    }

    /**
     * @notice Allow the rightful owner of a token to reclaim it, if it is currently on loan.
     * @dev Notice that (in contrast to the loan() function), this function has to use the _safeTransfer()
     *   function as opposed to safeTransferFrom(). The difference between these functions is that
     *   safeTransferFrom requires taht msg.sender _isApprovedOrOwner, whereas _sefTransfer() does not. In
     *   this case, the current owner as far as teh ERC721 contract is concerned is the borrower, so
     *   safeTransferFrom() cannot be used.
     * @param tokenId is the integer ID of the token that should be retrieved.
     */
    function reclaimLoan(uint256 tokenId) external nonReentrant {
        address rightfulOwner = mapFromTokenIdToRightfulOwner[tokenId];
        require(msg.sender == rightfulOwner, "ERC721Lending: Only the original/rightful owner can recall a loaned token.");

        address borrowerAddress = ownerOf(tokenId);

        // Remove it from the array of loaned out tokens
        delete mapFromTokenIdToRightfulOwner[tokenId];

        // Subtract from the rightful owner's loan balance
        uint256 loansByAddress = totalLoanedPerAddress[rightfulOwner];
        totalLoanedPerAddress[rightfulOwner] = loansByAddress - 1;

        // Decrease the global counter
        counterGlobalLoans = counterGlobalLoans - 1;
        
        // Transfer the token back. (_safeTransfer() requires four parameters, so it is necessary to
        // pass an empty string as the 'data'.)
        _safeTransfer(borrowerAddress, rightfulOwner, tokenId, "");

        emit LoanReclaimed(rightfulOwner, borrowerAddress, tokenId);
    }

    /**
     * @notice Allow the borrower to return the loaned token.
     * @param tokenId is the integer ID of the token that should be retrieved.
     */
    function returnLoanByBorrower(uint256 tokenId) external nonReentrant {
        address borrowerAddress = ownerOf(tokenId);
        require(msg.sender == borrowerAddress, "ERC721Lending: Only the borrower can return the token.");

        address rightfulOwner = mapFromTokenIdToRightfulOwner[tokenId];

        // Remove it from the array of loaned out tokens
        delete mapFromTokenIdToRightfulOwner[tokenId];

        // Subtract from the rightful owner's loan balance
        uint256 loansByAddress = totalLoanedPerAddress[rightfulOwner];
        totalLoanedPerAddress[rightfulOwner] = loansByAddress - 1;

        // Decrease the global counter
        counterGlobalLoans = counterGlobalLoans - 1;
        
        // Transfer the token back
        safeTransferFrom(borrowerAddress, rightfulOwner, tokenId);

        emit LoanReturned(borrowerAddress, rightfulOwner, tokenId);
    }

    /**
     * @notice Queries the number of tokens that are currently on loan.
     * @return The total number of tokens presently loaned.
     */
    function totalLoaned() public view returns (uint256) {
        return counterGlobalLoans;
    }

    /**
     * @notice Function retrieves the number of tokens that an address currently has on loan.
     * @param rightfulOwner is the original/rightful owner of a token or set of tokens.
     * @return The total number of tokens presently loaned by a specific original owner.
     */
    function loanedBalanceOf(address rightfulOwner) public view returns (uint256) {
        require(rightfulOwner != address(0), "ERC721Lending: Balance query for the zero address");
        return totalLoanedPerAddress[rightfulOwner];
    }

    /**
     * @notice Function to pause lending.
     * @dev The function is internal, so it should be called by child contracts, which allows
     *   them to implement their own restrictions, such as Access Control.
     */
    function _pauseLending() internal allowIfLendingNotPaused {
        loansAreCurrentlyPaused = true;
        emit LendingPaused(msg.sender);
    }

    /**
     * @notice Function to UNpause lending.
     * @dev The function is internal, so it should be called by child contracts, which allows
     *   them to implement their own restrictions, such as Access Control.
     */
    function _unpauseLending() internal {
        require(loansAreCurrentlyPaused, "ERC721Lending: Lending of tokens is already in unpaused state.");
        loansAreCurrentlyPaused = false;
        emit LendingUnpaused(msg.sender);
    }

    /**
     * @notice This hook is arguably the most important part of this contract. It is the piece
     *   of code that ensures a borrower cannot transfer the token.
     * @dev Hook that is called before any token transfer. This includes minting
     *   and burning.
     */
    function _beforeTokenTransfer(address from, address to, uint256 tokenId)
        internal
        virtual
        override(ERC721)
    {
        super._beforeTokenTransfer(from, to, tokenId);
        require(mapFromTokenIdToRightfulOwner[tokenId] == address(0), "ERC721Lending: Cannot transfer token on loan.");
    }

    /**
     * @dev Modifier to make a function callable only if lending is not paused.
     */
    modifier allowIfLendingNotPaused() {
        require(!loansAreCurrentlyPaused, "ERC721Lending: Lending of tokens is currently paused.");
        _;
    }

}

File 7 of 21 : URIManager.sol
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.13;

import "@openzeppelin/contracts/utils/Strings.sol";

/**
 * @dev Contract module which abstracts some aspects of URI management away from the main contract.
 *   This contract:
 *
 *      - Provides an adjustable 'default' URI for the location of the metadata (of tokens
 *        in the collection). Typically this is a folder in IPFS or some sort of web server.
 *
 *      - Enables eventual freezing of all metadata, which is useful if a project wants to start
 *        with centralized metadata and eventually move it to a decentralized location and then 'freeze'
 *        it there for posterity.
 */
abstract contract URIManager {
    using Strings for uint256;

    string private _defaultBaseURI;
    string private _contractURI;
    bool private _URIsAreForeverFrozen;

    /**
     * @dev Initializes the contract in unfrozen state with a particular
     * baseURI (a location containing the matadata for each NFT) and a particular
     * contractURI (a file containing collection-wide data, such as the description,
     * name, image, etc. of the collection.)
     */
    constructor(string memory initialBaseURI, string memory initialContractURI) {
        _setBaseURI(initialBaseURI);
        _setContractURI(initialContractURI);
        _URIsAreForeverFrozen = false;
    }
    
    function _setBaseURI(string memory _uri) internal {
        _defaultBaseURI = _uri;
    }

    function _setContractURI(string memory _newContractURI) internal {
        _contractURI = _newContractURI;
    }

    function _getBaseURI() internal view returns (string memory) {
        return _defaultBaseURI;
    }

    function _buildTokenURI(uint256 tokenId) internal view returns (string memory) {
        // return a concatenation of the baseURI (of the collection), with the tokenID, and the file extension.
        return string(abi.encodePacked(_getBaseURI(), tokenId.toString(), ".json"));
    }

    /**
     * @dev Opensea states that a contract may have a contractURI() function, which
     *  returns metadata for the contract as a whole.
     */
    function contractURI() public view returns (string memory) {
        return _contractURI;
    }

    /**
     * @dev Returns true if the metadata URIs have been finalized forever.
     */
    function areURIsForeverFrozen() public view virtual returns (bool) {
        return _URIsAreForeverFrozen;
    }

    /**
     * @dev Modifier to make a function callable only if the URIs have not been frozen forever.
     *
     * Requirements:
     *
     * - The contract must not be paused.
     */
    modifier allowIfNotFrozen() {
        require(!areURIsForeverFrozen(), "URIManager: URIs have been frozen forever");
        _;
    }

    /**
     * @dev Freezes all future changes of the URIs.
     *
     * Requirements:
     *
     * - The URIs must not be frozen already.
     */
    function _freezeURIsForever() internal virtual allowIfNotFrozen {
        _URIsAreForeverFrozen = true;
    }
}

File 8 of 21 : ERC2981GlobalRoyalties.sol
// SPDX-License-Identifier: MIT
pragma solidity ^0.8.13;

import "@openzeppelin/contracts/interfaces/IERC2981.sol";
import "@openzeppelin/contracts/utils/introspection/ERC165.sol";

/**
 * @title Implementation ERC2981 Ethereum NFT Royalty Standard
 * @notice Implementation of the NFT Royalty Standard, a standardized way to retrieve royalty payment information.
 *   Implementation based off of OpenZeppelin's ERC2981.sol, with some customization by Real Vision's web3
 *   team. (The customization primarily revolves around simplifying the contract so that royalties are only
 *   set for all tokens in the collection, rather than allowing for specific tokens to have custom royalties.)
 *   Our sincere Affen gratitude to the hard work, and collaborative spirit of both OpenZeppelin and Real Vision.
 *   IMPORTANT: ERC-2981 only specifies a way to signal royalty information and does not enforce its payment. See
 *   https://eips.ethereum.org/EIPS/eip-2981#optional-royalty-payments[Rationale] in the EIP. Marketplaces are
 *   expected to voluntarily pay royalties together with sales, but note that this standard is not yet
 *   widely supported.
 * @dev The 'Global' word in the name of this contract is there to signify that this contract deliberately does
 *   not implement royalties at the level of each token - it only allows for royalty destination and amount to
 *   be set for ALL tokens in the collection. ALSO NOTE that this contract is IERC2981, and yet, it does not
 *   implement the only function that is required by IERC2981: royaltyInfo(). This task is left to the descendants
 *   of this contract to implement.
 */
abstract contract ERC2981GlobalRoyalties is IERC2981, ERC165 {

    address private _royaltyDestination;
    uint16 private _royaltyInBips;
    uint16 private _bipsBasedFeeDenominator = 10000;
    
    /**
     * @dev See {IERC165-supportsInterface}.
     */
    function supportsInterface(bytes4 interfaceId)
        public
        view
        virtual
        override(ERC165, IERC165)
        returns (bool)
    {
        return interfaceId == type(IERC2981).interfaceId || super.supportsInterface(interfaceId);
    }

    /**
     * @notice Function to set the royalty amount using Basis Points.
     * @dev See OpenZeppelin's docuentation for explanation of their choice of basis points and denominator.
     * @param _newRoyaltyInBips is the amount (as basis points) that the royalty will be set to. For
     *   example, make this parameter 200, to set a royalty of 2%)
     */
    function _setRoyaltyAmountInBips(uint16 _newRoyaltyInBips) internal {
        require(_newRoyaltyInBips <= _bipsBasedFeeDenominator, "Royalty fee will exceed salePrice");
        _royaltyInBips = _newRoyaltyInBips;
    }

    /**
     * @notice Function to set the royalty destination.
     * @param _newRoyaltyDestination is the address that royalties should be sent to.
     */
    function _setRoyaltyDestination(address _newRoyaltyDestination) internal {
        _royaltyDestination = _newRoyaltyDestination;
    }


    /**
     * @notice 
     * @dev The two functions below (royaltyInfo() and _globalRoyaltyInfo() offer the developer a
     *   choice of ways to implement the compulsory (to meet the requirements of the Interface of EIP2981)
     *   function called royaltyInfo() in descendant contracts.
     *   (Both options require overriding the royaltyInfo() declaration of this contract.)
     *   1 - the first option is to override royaltyInfo() and implement the contents of the
     *       function (in the child contract) from scratch in whatever way the developer sees fit.
     *   2 - the second option is to override, but instead of implementing from scratch, 
     *       inside the override (in the child), simply call the internal function _globalRoyaltyInfo()
     *       which already has a working implementation coded below.
     *   As for the parameters and return value, please refer to the official documentation of
     *   eip-2981 for the best explanation.
     */
    function royaltyInfo(uint256 _tokenId, uint256 _salePrice) public view virtual
        returns (address, uint256);

    /**
     * @notice An internal function that can optionally be used by descendant contracts as a ready-made
     *   way to implement the mandatory royaltyInfo() function. The function calculates where and how
     *   much royalty should be sent based on the current global settings of the collection.
     * @dev A descendant contract, in the implementation of royaltyInfo() can simply call this function
     *   if it suits the intended purposes. HOWEVER please NOTE those contracts
     *   (that inherit from this contract) should make sure that the 'royaltyInfo()' function they
     *   implement includes a 'tokenId' parameter in order to comply with EIP2981.
     *   To understand why (within the function) the denominator is 10,000, please see the definition of
     *   the unambiguous financial term: 'basis points' (bips)
     * @param _salePrice is the price that a token is being sold for. A tokenId is not required for this
     *   function because this implementation of eip-2981 only keeps 'global' settings of royalties
     *   for the collections as whole (rather than keeping settings for individual tokens.)
     * @return two values: 1) the royalty destination, and 2) the royalty amount, as required by eip-2981
     */
    function _globalRoyaltyInfo(uint256 _salePrice)
        internal
        view
        returns (address, uint256)
    {
        uint256 royaltyAmount = (_salePrice * _royaltyInBips) / _bipsBasedFeeDenominator;
        return (_royaltyDestination, royaltyAmount);
    }
}

File 9 of 21 : ERC721Burnable.sol
// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts (last updated v4.7.0) (token/ERC721/extensions/ERC721Burnable.sol)

pragma solidity ^0.8.0;

import "../ERC721.sol";
import "../../../utils/Context.sol";

/**
 * @title ERC721 Burnable Token
 * @dev ERC721 Token that can be burned (destroyed).
 */
abstract contract ERC721Burnable is Context, ERC721 {
    /**
     * @dev Burns `tokenId`. See {ERC721-_burn}.
     *
     * Requirements:
     *
     * - The caller must own `tokenId` or be an approved operator.
     */
    function burn(uint256 tokenId) public virtual {
        //solhint-disable-next-line max-line-length
        require(_isApprovedOrOwner(_msgSender(), tokenId), "ERC721: caller is not token owner nor approved");
        _burn(tokenId);
    }
}

File 10 of 21 : Context.sol
// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts v4.4.1 (utils/Context.sol)

pragma solidity ^0.8.0;

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

    function _msgData() internal view virtual returns (bytes calldata) {
        return msg.data;
    }
}

File 11 of 21 : IAccessControl.sol
// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts v4.4.1 (access/IAccessControl.sol)

pragma solidity ^0.8.0;

/**
 * @dev External interface of AccessControl declared to support ERC165 detection.
 */
interface IAccessControl {
    /**
     * @dev Emitted when `newAdminRole` is set as ``role``'s admin role, replacing `previousAdminRole`
     *
     * `DEFAULT_ADMIN_ROLE` is the starting admin for all roles, despite
     * {RoleAdminChanged} not being emitted signaling this.
     *
     * _Available since v3.1._
     */
    event RoleAdminChanged(bytes32 indexed role, bytes32 indexed previousAdminRole, bytes32 indexed newAdminRole);

    /**
     * @dev Emitted when `account` is granted `role`.
     *
     * `sender` is the account that originated the contract call, an admin role
     * bearer except when using {AccessControl-_setupRole}.
     */
    event RoleGranted(bytes32 indexed role, address indexed account, address indexed sender);

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

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

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

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

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

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

File 12 of 21 : Strings.sol
// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts (last updated v4.7.0) (utils/Strings.sol)

pragma solidity ^0.8.0;

/**
 * @dev String operations.
 */
library Strings {
    bytes16 private constant _HEX_SYMBOLS = "0123456789abcdef";
    uint8 private constant _ADDRESS_LENGTH = 20;

    /**
     * @dev Converts a `uint256` to its ASCII `string` decimal representation.
     */
    function toString(uint256 value) internal pure returns (string memory) {
        // Inspired by OraclizeAPI's implementation - MIT licence
        // https://github.com/oraclize/ethereum-api/blob/b42146b063c7d6ee1358846c198246239e9360e8/oraclizeAPI_0.4.25.sol

        if (value == 0) {
            return "0";
        }
        uint256 temp = value;
        uint256 digits;
        while (temp != 0) {
            digits++;
            temp /= 10;
        }
        bytes memory buffer = new bytes(digits);
        while (value != 0) {
            digits -= 1;
            buffer[digits] = bytes1(uint8(48 + uint256(value % 10)));
            value /= 10;
        }
        return string(buffer);
    }

    /**
     * @dev Converts a `uint256` to its ASCII `string` hexadecimal representation.
     */
    function toHexString(uint256 value) internal pure returns (string memory) {
        if (value == 0) {
            return "0x00";
        }
        uint256 temp = value;
        uint256 length = 0;
        while (temp != 0) {
            length++;
            temp >>= 8;
        }
        return toHexString(value, length);
    }

    /**
     * @dev Converts a `uint256` to its ASCII `string` hexadecimal representation with fixed length.
     */
    function toHexString(uint256 value, uint256 length) internal pure returns (string memory) {
        bytes memory buffer = new bytes(2 * length + 2);
        buffer[0] = "0";
        buffer[1] = "x";
        for (uint256 i = 2 * length + 1; i > 1; --i) {
            buffer[i] = _HEX_SYMBOLS[value & 0xf];
            value >>= 4;
        }
        require(value == 0, "Strings: hex length insufficient");
        return string(buffer);
    }

    /**
     * @dev Converts an `address` with fixed length of 20 bytes to its not checksummed ASCII `string` hexadecimal representation.
     */
    function toHexString(address addr) internal pure returns (string memory) {
        return toHexString(uint256(uint160(addr)), _ADDRESS_LENGTH);
    }
}

File 13 of 21 : ERC165.sol
// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts v4.4.1 (utils/introspection/ERC165.sol)

pragma solidity ^0.8.0;

import "./IERC165.sol";

/**
 * @dev Implementation of the {IERC165} interface.
 *
 * Contracts that want to implement ERC165 should inherit from this contract and override {supportsInterface} to check
 * for the additional interface id that will be supported. For example:
 *
 * ```solidity
 * function supportsInterface(bytes4 interfaceId) public view virtual override returns (bool) {
 *     return interfaceId == type(MyInterface).interfaceId || super.supportsInterface(interfaceId);
 * }
 * ```
 *
 * Alternatively, {ERC165Storage} provides an easier to use but more expensive implementation.
 */
abstract contract ERC165 is IERC165 {
    /**
     * @dev See {IERC165-supportsInterface}.
     */
    function supportsInterface(bytes4 interfaceId) public view virtual override returns (bool) {
        return interfaceId == type(IERC165).interfaceId;
    }
}

File 14 of 21 : IERC165.sol
// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts v4.4.1 (utils/introspection/IERC165.sol)

pragma solidity ^0.8.0;

/**
 * @dev Interface of the ERC165 standard, as defined in the
 * https://eips.ethereum.org/EIPS/eip-165[EIP].
 *
 * Implementers can declare support of contract interfaces, which can then be
 * queried by others ({ERC165Checker}).
 *
 * For an implementation, see {ERC165}.
 */
interface IERC165 {
    /**
     * @dev Returns true if this contract implements the interface defined by
     * `interfaceId`. See the corresponding
     * https://eips.ethereum.org/EIPS/eip-165#how-interfaces-are-identified[EIP section]
     * to learn more about how these ids are created.
     *
     * This function call must use less than 30 000 gas.
     */
    function supportsInterface(bytes4 interfaceId) external view returns (bool);
}

File 15 of 21 : Address.sol
// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts (last updated v4.7.0) (utils/Address.sol)

pragma solidity ^0.8.1;

/**
 * @dev Collection of functions related to the address type
 */
library Address {
    /**
     * @dev Returns true if `account` is a contract.
     *
     * [IMPORTANT]
     * ====
     * It is unsafe to assume that an address for which this function returns
     * false is an externally-owned account (EOA) and not a contract.
     *
     * Among others, `isContract` will return false for the following
     * types of addresses:
     *
     *  - an externally-owned account
     *  - a contract in construction
     *  - an address where a contract will be created
     *  - an address where a contract lived, but was destroyed
     * ====
     *
     * [IMPORTANT]
     * ====
     * You shouldn't rely on `isContract` to protect against flash loan attacks!
     *
     * Preventing calls from contracts is highly discouraged. It breaks composability, breaks support for smart wallets
     * like Gnosis Safe, and does not provide security since it can be circumvented by calling from a contract
     * constructor.
     * ====
     */
    function isContract(address account) internal view returns (bool) {
        // This method relies on extcodesize/address.code.length, which returns 0
        // for contracts in construction, since the code is only stored at the end
        // of the constructor execution.

        return account.code.length > 0;
    }

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

        (bool success, ) = recipient.call{value: amount}("");
        require(success, "Address: unable to send value, recipient may have reverted");
    }

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

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

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

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

        (bool success, bytes memory returndata) = target.call{value: value}(data);
        return verifyCallResult(success, returndata, errorMessage);
    }

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

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

        (bool success, bytes memory returndata) = target.staticcall(data);
        return verifyCallResult(success, returndata, errorMessage);
    }

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

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

        (bool success, bytes memory returndata) = target.delegatecall(data);
        return verifyCallResult(success, returndata, errorMessage);
    }

    /**
     * @dev Tool to verifies that a low level call was successful, and revert if it wasn't, either by bubbling the
     * revert reason using the provided one.
     *
     * _Available since v4.3._
     */
    function verifyCallResult(
        bool success,
        bytes memory returndata,
        string memory errorMessage
    ) internal pure returns (bytes memory) {
        if (success) {
            return returndata;
        } else {
            // Look for revert reason and bubble it up if present
            if (returndata.length > 0) {
                // The easiest way to bubble the revert reason is using memory via assembly
                /// @solidity memory-safe-assembly
                assembly {
                    let returndata_size := mload(returndata)
                    revert(add(32, returndata), returndata_size)
                }
            } else {
                revert(errorMessage);
            }
        }
    }
}

File 16 of 21 : IERC721.sol
// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts (last updated v4.7.0) (token/ERC721/IERC721.sol)

pragma solidity ^0.8.0;

import "../../utils/introspection/IERC165.sol";

/**
 * @dev Required interface of an ERC721 compliant contract.
 */
interface IERC721 is IERC165 {
    /**
     * @dev Emitted when `tokenId` token is transferred from `from` to `to`.
     */
    event Transfer(address indexed from, address indexed to, uint256 indexed tokenId);

    /**
     * @dev Emitted when `owner` enables `approved` to manage the `tokenId` token.
     */
    event Approval(address indexed owner, address indexed approved, uint256 indexed tokenId);

    /**
     * @dev Emitted when `owner` enables or disables (`approved`) `operator` to manage all of its assets.
     */
    event ApprovalForAll(address indexed owner, address indexed operator, bool approved);

    /**
     * @dev Returns the number of tokens in ``owner``'s account.
     */
    function balanceOf(address owner) external view returns (uint256 balance);

    /**
     * @dev Returns the owner of the `tokenId` token.
     *
     * Requirements:
     *
     * - `tokenId` must exist.
     */
    function ownerOf(uint256 tokenId) external view returns (address owner);

    /**
     * @dev Safely transfers `tokenId` token from `from` to `to`.
     *
     * Requirements:
     *
     * - `from` cannot be the zero address.
     * - `to` cannot be the zero address.
     * - `tokenId` token must exist and be owned by `from`.
     * - If the caller is not `from`, it must be approved to move this token by either {approve} or {setApprovalForAll}.
     * - If `to` refers to a smart contract, it must implement {IERC721Receiver-onERC721Received}, which is called upon a safe transfer.
     *
     * Emits a {Transfer} event.
     */
    function safeTransferFrom(
        address from,
        address to,
        uint256 tokenId,
        bytes calldata data
    ) external;

    /**
     * @dev Safely transfers `tokenId` token from `from` to `to`, checking first that contract recipients
     * are aware of the ERC721 protocol to prevent tokens from being forever locked.
     *
     * Requirements:
     *
     * - `from` cannot be the zero address.
     * - `to` cannot be the zero address.
     * - `tokenId` token must exist and be owned by `from`.
     * - If the caller is not `from`, it must have been allowed to move this token by either {approve} or {setApprovalForAll}.
     * - If `to` refers to a smart contract, it must implement {IERC721Receiver-onERC721Received}, which is called upon a safe transfer.
     *
     * Emits a {Transfer} event.
     */
    function safeTransferFrom(
        address from,
        address to,
        uint256 tokenId
    ) external;

    /**
     * @dev Transfers `tokenId` token from `from` to `to`.
     *
     * WARNING: Usage of this method is discouraged, use {safeTransferFrom} whenever possible.
     *
     * Requirements:
     *
     * - `from` cannot be the zero address.
     * - `to` cannot be the zero address.
     * - `tokenId` token must be owned by `from`.
     * - If the caller is not `from`, it must be approved to move this token by either {approve} or {setApprovalForAll}.
     *
     * Emits a {Transfer} event.
     */
    function transferFrom(
        address from,
        address to,
        uint256 tokenId
    ) external;

    /**
     * @dev Gives permission to `to` to transfer `tokenId` token to another account.
     * The approval is cleared when the token is transferred.
     *
     * Only a single account can be approved at a time, so approving the zero address clears previous approvals.
     *
     * Requirements:
     *
     * - The caller must own the token or be an approved operator.
     * - `tokenId` must exist.
     *
     * Emits an {Approval} event.
     */
    function approve(address to, uint256 tokenId) external;

    /**
     * @dev Approve or remove `operator` as an operator for the caller.
     * Operators can call {transferFrom} or {safeTransferFrom} for any token owned by the caller.
     *
     * Requirements:
     *
     * - The `operator` cannot be the caller.
     *
     * Emits an {ApprovalForAll} event.
     */
    function setApprovalForAll(address operator, bool _approved) external;

    /**
     * @dev Returns the account approved for `tokenId` token.
     *
     * Requirements:
     *
     * - `tokenId` must exist.
     */
    function getApproved(uint256 tokenId) external view returns (address operator);

    /**
     * @dev Returns if the `operator` is allowed to manage all of the assets of `owner`.
     *
     * See {setApprovalForAll}
     */
    function isApprovedForAll(address owner, address operator) external view returns (bool);
}

File 17 of 21 : IERC721Receiver.sol
// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts (last updated v4.6.0) (token/ERC721/IERC721Receiver.sol)

pragma solidity ^0.8.0;

/**
 * @title ERC721 token receiver interface
 * @dev Interface for any contract that wants to support safeTransfers
 * from ERC721 asset contracts.
 */
interface IERC721Receiver {
    /**
     * @dev Whenever an {IERC721} `tokenId` token is transferred to this contract via {IERC721-safeTransferFrom}
     * by `operator` from `from`, this function is called.
     *
     * It must return its Solidity selector to confirm the token transfer.
     * If any other value is returned or the interface is not implemented by the recipient, the transfer will be reverted.
     *
     * The selector can be obtained in Solidity with `IERC721Receiver.onERC721Received.selector`.
     */
    function onERC721Received(
        address operator,
        address from,
        uint256 tokenId,
        bytes calldata data
    ) external returns (bytes4);
}

File 18 of 21 : IERC721Metadata.sol
// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts v4.4.1 (token/ERC721/extensions/IERC721Metadata.sol)

pragma solidity ^0.8.0;

import "../IERC721.sol";

/**
 * @title ERC-721 Non-Fungible Token Standard, optional metadata extension
 * @dev See https://eips.ethereum.org/EIPS/eip-721
 */
interface IERC721Metadata is IERC721 {
    /**
     * @dev Returns the token collection name.
     */
    function name() external view returns (string memory);

    /**
     * @dev Returns the token collection symbol.
     */
    function symbol() external view returns (string memory);

    /**
     * @dev Returns the Uniform Resource Identifier (URI) for `tokenId` token.
     */
    function tokenURI(uint256 tokenId) external view returns (string memory);
}

File 19 of 21 : IERC721Enumerable.sol
// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts (last updated v4.5.0) (token/ERC721/extensions/IERC721Enumerable.sol)

pragma solidity ^0.8.0;

import "../IERC721.sol";

/**
 * @title ERC-721 Non-Fungible Token Standard, optional enumeration extension
 * @dev See https://eips.ethereum.org/EIPS/eip-721
 */
interface IERC721Enumerable is IERC721 {
    /**
     * @dev Returns the total amount of tokens stored by the contract.
     */
    function totalSupply() external view returns (uint256);

    /**
     * @dev Returns a token ID owned by `owner` at a given `index` of its token list.
     * Use along with {balanceOf} to enumerate all of ``owner``'s tokens.
     */
    function tokenOfOwnerByIndex(address owner, uint256 index) external view returns (uint256);

    /**
     * @dev Returns a token ID at a given `index` of all the tokens stored by the contract.
     * Use along with {totalSupply} to enumerate all tokens.
     */
    function tokenByIndex(uint256 index) external view returns (uint256);
}

File 20 of 21 : ReentrancyGuard.sol
// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts v4.4.1 (security/ReentrancyGuard.sol)

pragma solidity ^0.8.0;

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

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

    uint256 private _status;

    constructor() {
        _status = _NOT_ENTERED;
    }

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

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

        _;

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

File 21 of 21 : IERC2981.sol
// SPDX-License-Identifier: MIT
// OpenZeppelin Contracts (last updated v4.6.0) (interfaces/IERC2981.sol)

pragma solidity ^0.8.0;

import "../utils/introspection/IERC165.sol";

/**
 * @dev Interface for the NFT Royalty Standard.
 *
 * A standardized way to retrieve royalty payment information for non-fungible tokens (NFTs) to enable universal
 * support for royalty payments across all NFT marketplaces and ecosystem participants.
 *
 * _Available since v4.5._
 */
interface IERC2981 is IERC165 {
    /**
     * @dev Returns how much royalty is owed and to whom, based on a sale price that may be denominated in any unit of
     * exchange. The royalty amount is denominated and should be paid in that same unit of exchange.
     */
    function royaltyInfo(uint256 tokenId, uint256 salePrice)
        external
        view
        returns (address receiver, uint256 royaltyAmount);
}

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

Contract Security Audit

Contract ABI

[{"inputs":[{"internalType":"string","name":"name","type":"string"},{"internalType":"string","name":"symbol","type":"string"},{"internalType":"string","name":"initialBaseURI","type":"string"},{"internalType":"string","name":"initialContractURI","type":"string"}],"stateMutability":"nonpayable","type":"constructor"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"owner","type":"address"},{"indexed":true,"internalType":"address","name":"approved","type":"address"},{"indexed":true,"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"Approval","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"owner","type":"address"},{"indexed":true,"internalType":"address","name":"operator","type":"address"},{"indexed":false,"internalType":"bool","name":"approved","type":"bool"}],"name":"ApprovalForAll","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"account","type":"address"}],"name":"LendingPaused","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"account","type":"address"}],"name":"LendingUnpaused","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":"item","type":"uint256"}],"name":"Loan","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"byOriginalOwner","type":"address"},{"indexed":true,"internalType":"address","name":"fromBorrower","type":"address"},{"indexed":false,"internalType":"uint256","name":"item","type":"uint256"}],"name":"LoanReclaimed","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"byBorrower","type":"address"},{"indexed":true,"internalType":"address","name":"toOriginalOwner","type":"address"},{"indexed":false,"internalType":"uint256","name":"item","type":"uint256"}],"name":"LoanReturned","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":"address","name":"account","type":"address"}],"name":"Paused","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"bytes32","name":"role","type":"bytes32"},{"indexed":true,"internalType":"bytes32","name":"previousAdminRole","type":"bytes32"},{"indexed":true,"internalType":"bytes32","name":"newAdminRole","type":"bytes32"}],"name":"RoleAdminChanged","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"bytes32","name":"role","type":"bytes32"},{"indexed":true,"internalType":"address","name":"account","type":"address"},{"indexed":true,"internalType":"address","name":"sender","type":"address"}],"name":"RoleGranted","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"bytes32","name":"role","type":"bytes32"},{"indexed":true,"internalType":"address","name":"account","type":"address"},{"indexed":true,"internalType":"address","name":"sender","type":"address"}],"name":"RoleRevoked","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"from","type":"address"},{"indexed":true,"internalType":"address","name":"to","type":"address"},{"indexed":true,"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"Transfer","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"account","type":"address"}],"name":"Unpaused","type":"event"},{"inputs":[],"name":"DEFAULT_ADMIN_ROLE","outputs":[{"internalType":"bytes32","name":"","type":"bytes32"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"ROLE_METADATA_FREEZER","outputs":[{"internalType":"bytes32","name":"","type":"bytes32"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"ROLE_METADATA_UPDATER","outputs":[{"internalType":"bytes32","name":"","type":"bytes32"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"ROLE_MINTER","outputs":[{"internalType":"bytes32","name":"","type":"bytes32"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"ROLE_PAUSER","outputs":[{"internalType":"bytes32","name":"","type":"bytes32"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"ROLE_ROYALTY_SETTING","outputs":[{"internalType":"bytes32","name":"","type":"bytes32"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"approve","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"areURIsForeverFrozen","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"owner","type":"address"}],"name":"balanceOf","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"burn","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"contractURI","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"freezeURIsForever","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"getApproved","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"bytes32","name":"role","type":"bytes32"}],"name":"getRoleAdmin","outputs":[{"internalType":"bytes32","name":"","type":"bytes32"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"bytes32","name":"role","type":"bytes32"},{"internalType":"address","name":"account","type":"address"}],"name":"grantRole","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"bytes32","name":"role","type":"bytes32"},{"internalType":"address","name":"account","type":"address"}],"name":"hasRole","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"owner","type":"address"},{"internalType":"address","name":"operator","type":"address"}],"name":"isApprovedForAll","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"receiver","type":"address"},{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"loan","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"rightfulOwner","type":"address"}],"name":"loanedBalanceOf","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"rightfulOwner","type":"address"}],"name":"loanedTokensByAddress","outputs":[{"internalType":"uint256[]","name":"","type":"uint256[]"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"loansAreCurrentlyPaused","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"}],"name":"mapFromTokenIdToRightfulOwner","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"name","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"numTokensMinted","outputs":[{"internalType":"uint8","name":"","type":"uint8"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"tokenOwner","type":"address"}],"name":"ownedTokensByAddress","outputs":[{"internalType":"uint256[]","name":"","type":"uint256[]"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"owner","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"ownerOf","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"pause","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"pauseLending","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"paused","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"reclaimLoan","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"bytes32","name":"role","type":"bytes32"},{"internalType":"address","name":"account","type":"address"}],"name":"renounceRole","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"returnLoanByBorrower","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"bytes32","name":"role","type":"bytes32"},{"internalType":"address","name":"account","type":"address"}],"name":"revokeRole","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"tokenId","type":"uint256"},{"internalType":"uint256","name":"salePrice","type":"uint256"}],"name":"royaltyInfo","outputs":[{"internalType":"address","name":"","type":"address"},{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"safeMint","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"from","type":"address"},{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"safeTransferFrom","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"from","type":"address"},{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"tokenId","type":"uint256"},{"internalType":"bytes","name":"data","type":"bytes"}],"name":"safeTransferFrom","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"operator","type":"address"},{"internalType":"bool","name":"approved","type":"bool"}],"name":"setApprovalForAll","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"string","name":"newURI","type":"string"}],"name":"setBaseURI","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"string","name":"newContractURI","type":"string"}],"name":"setContractURI","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"honoraryOwner","type":"address"}],"name":"setHonoraryOwner","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint16","name":"newRoyaltyInBips","type":"uint16"}],"name":"setRoyaltyAmountInBips","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"newRoyaltyDestination","type":"address"}],"name":"setRoyaltyDestination","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"bytes4","name":"interfaceId","type":"bytes4"}],"name":"supportsInterface","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"symbol","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"index","type":"uint256"}],"name":"tokenByIndex","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"owner","type":"address"},{"internalType":"uint256","name":"index","type":"uint256"}],"name":"tokenOfOwnerByIndex","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"tokenURI","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"totalLoaned","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"totalLoanedPerAddress","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"totalSupply","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"from","type":"address"},{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"transferFrom","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"unpause","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"unpauseLending","outputs":[],"stateMutability":"nonpayable","type":"function"}]

6080604052600c805461ffff60b01b191661027160b41b17905560006013556014805460ff191690553480156200003557600080fd5b5060405162003fda38038062003fda833981016040819052620000589162000670565b81818585816000908051906020019062000074929190620004f0565b5080516200008a906001906020840190620004f0565b5050600a805460ff1916905550620000a282620000e0565b620000ad81620000f9565b5050600f805460ff191690556001601055620000cb6000336200010e565b620000d633620001b2565b5050505062000896565b8051620000f590600d906020840190620004f0565b5050565b8051620000f590600e906020840190620004f0565b6000828152600b602090815260408083206001600160a01b038516845290915290205460ff16620000f5576000828152600b602090815260408083206001600160a01b03851684529091529020805460ff191660011790556200016e3390565b6001600160a01b0316816001600160a01b0316837f2f8788117e7eff1d82e926ec794901d17c78024a50270940304540a733656f0d60405160405180910390a45050565b6000620001bf8162000284565b6001600160a01b038216620002295760405162461bcd60e51b815260206004820152602560248201527f4e6577206f776e65722063616e6e6f7420626520746865207a65726f206164646044820152643932b9b99760d91b60648201526084015b60405180910390fd5b601480546001600160a01b03848116610100818102610100600160a81b031985161790945560405193909204169182907f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e090600090a3505050565b62000290813362000293565b50565b6000828152600b602090815260408083206001600160a01b038516845290915290205460ff16620000f557620002df816001600160a01b031660146200033060201b62001bf71760201c565b620002f583602062001bf762000330821b17811c565b6040516020016200030892919062000729565b60408051601f198184030181529082905262461bcd60e51b82526200022091600401620007a2565b6060600062000341836002620007ed565b6200034e9060026200080f565b6001600160401b0381111562000368576200036862000596565b6040519080825280601f01601f19166020018201604052801562000393576020820181803683370190505b509050600360fc1b81600081518110620003b157620003b16200082a565b60200101906001600160f81b031916908160001a905350600f60fb1b81600181518110620003e357620003e36200082a565b60200101906001600160f81b031916908160001a905350600062000409846002620007ed565b620004169060016200080f565b90505b600181111562000498576f181899199a1a9b1b9c1cb0b131b232b360811b85600f16601081106200044e576200044e6200082a565b1a60f81b8282815181106200046757620004676200082a565b60200101906001600160f81b031916908160001a90535060049490941c93620004908162000840565b905062000419565b508315620004e95760405162461bcd60e51b815260206004820181905260248201527f537472696e67733a20686578206c656e67746820696e73756666696369656e74604482015260640162000220565b9392505050565b828054620004fe906200085a565b90600052602060002090601f0160209004810192826200052257600085556200056d565b82601f106200053d57805160ff19168380011785556200056d565b828001600101855582156200056d579182015b828111156200056d57825182559160200191906001019062000550565b506200057b9291506200057f565b5090565b5b808211156200057b576000815560010162000580565b634e487b7160e01b600052604160045260246000fd5b60005b83811015620005c9578181015183820152602001620005af565b83811115620005d9576000848401525b50505050565b600082601f830112620005f157600080fd5b81516001600160401b03808211156200060e576200060e62000596565b604051601f8301601f19908116603f0116810190828211818310171562000639576200063962000596565b816040528381528660208588010111156200065357600080fd5b62000666846020830160208901620005ac565b9695505050505050565b600080600080608085870312156200068757600080fd5b84516001600160401b03808211156200069f57600080fd5b620006ad88838901620005df565b95506020870151915080821115620006c457600080fd5b620006d288838901620005df565b94506040870151915080821115620006e957600080fd5b620006f788838901620005df565b935060608701519150808211156200070e57600080fd5b506200071d87828801620005df565b91505092959194509250565b7f416363657373436f6e74726f6c3a206163636f756e742000000000000000000081526000835162000763816017850160208801620005ac565b7001034b99036b4b9b9b4b733903937b6329607d1b601791840191820152835162000796816028840160208801620005ac565b01602801949350505050565b6020815260008251806020840152620007c3816040850160208701620005ac565b601f01601f19169190910160400192915050565b634e487b7160e01b600052601160045260246000fd5b60008160001904831182151516156200080a576200080a620007d7565b500290565b60008219821115620008255762000825620007d7565b500190565b634e487b7160e01b600052603260045260246000fd5b600081620008525762000852620007d7565b506000190190565b600181811c908216806200086f57607f821691505b6020821081036200089057634e487b7160e01b600052602260045260246000fd5b50919050565b61373480620008a66000396000f3fe608060405234801561001057600080fd5b506004361061035d5760003560e01c80637f48e368116101d3578063acd36b3911610104578063dba028c2116100a2578063e8a3d4851161007c578063e8a3d485146107bc578063e985e9c5146107c4578063ed5a2ec114610800578063f2a808d61461081357600080fd5b8063dba028c214610781578063dde36719146107a1578063ddead0eb146107b457600080fd5b8063b88d4fde116100de578063b88d4fde14610721578063c87b56dd14610734578063d418a9d314610747578063d547741f1461076e57600080fd5b8063acd36b39146106dd578063b047eb0214610704578063b4193d8a1461071957600080fd5b806395d89b4111610171578063a14481941161014b578063a144819414610686578063a217fddf14610699578063a22cb465146106a1578063a583c928146106b457600080fd5b806395d89b411461066b57806397c47d7c146106735780639e6e5cfc1461067e57600080fd5b806391d14854116101ad57806391d148541461060b57806392afc33a1461061e578063938e3d7b1461064557806393b068a41461065857600080fd5b80637f48e368146105c45780638456cb59146105eb5780638da5cb5b146105f357600080fd5b80633a838c2a116102ad57806355f804b31161024b5780635e3dda6f116102255780635e3dda6f1461056b5780635f9be2ac1461057e5780636352211e1461059e57806370a08231146105b157600080fd5b806355f804b31461053a5780635c975abb1461054d5780635e03a6aa1461055857600080fd5b806342842e0e1161028757806342842e0e146104ee57806342966c68146105015780634f6ccce71461051457806355b7c4431461052757600080fd5b80633a838c2a146104b85780633f4ba83a146104c05780634230baee146104c857600080fd5b806323b872dd1161031a5780632f2ff15d116102f45780632f2ff15d1461046c5780632f745c591461047f57806336568abe146104925780633746e948146104a557600080fd5b806323b872dd14610404578063248a9ca3146104175780632a55205a1461043a57600080fd5b806301ffc9a71461036257806306fdde031461038a578063081812fc1461039f578063095ea7b3146103ca57806318160ddd146103df578063185cc751146103f1575b600080fd5b610375610370366004612f27565b610820565b60405190151581526020015b60405180910390f35b610392610831565b6040516103819190612f9c565b6103b26103ad366004612faf565b6108c3565b6040516001600160a01b039091168152602001610381565b6103dd6103d8366004612fe4565b6108ea565b005b6008545b604051908152602001610381565b6103dd6103ff366004612fe4565b610a04565b6103dd61041236600461300e565b610d58565b6103e3610425366004612faf565b6000908152600b602052604090206001015490565b61044d61044836600461304a565b610d8a565b604080516001600160a01b039093168352602083019190915201610381565b6103dd61047a36600461306c565b610e15565b6103e361048d366004612fe4565b610e3a565b6103dd6104a036600461306c565b610ed0565b6103dd6104b3366004613098565b610f4e565b6013546103e3565b6103dd610f81565b6014546104dc90600160a81b900460ff1681565b60405160ff9091168152602001610381565b6103dd6104fc36600461300e565b610fa4565b6103dd61050f366004612faf565b610fbf565b6103e3610522366004612faf565b610fed565b6103dd6105353660046130bc565b611080565b6103dd6105483660046130d7565b611157565b600a5460ff16610375565b6103e36105663660046130bc565b6111e3565b6103dd610579366004612faf565b611227565b6103e361058c3660046130bc565b60116020526000908152604090205481565b6103b26105ac366004612faf565b6113d8565b6103e36105bf3660046130bc565b611438565b6103e37f58376ed7a9634ebd5569e965099321a98f915ee1aad1399d411b22996681698e81565b6103dd6114be565b6014546103b29061010090046001600160a01b031681565b61037561061936600461306c565b6114de565b6103e37faeaef46186eb59f884e36929b6d682a6ae35e1e43d8f05f058dcefb92b60146181565b6103dd6106533660046130d7565b611509565b6103dd610666366004612faf565b611595565b610392611766565b600f5460ff16610375565b6103dd611775565b6103dd610694366004612fe4565b611795565b6103e3600081565b6103dd6106af366004613149565b611892565b6103b26106c2366004612faf565b6012602052600090815260409020546001600160a01b031681565b6103e37fd9f17c44520e5d33006f7504376dd1fae046f60c67eb4e7800f24debd368918081565b6103e36000805160206136df83398151915281565b6103dd61189d565b6103dd61072f36600461319b565b6118bd565b610392610742366004612faf565b6118f5565b6103e37f7840fe4b17ccfbb599243b7bfe0ed6cbbe3a2279e52b2f14ffdef903de88d25f81565b6103dd61077c36600461306c565b611965565b61079461078f3660046130bc565b61198a565b6040516103819190613277565b6103dd6107af3660046130bc565b611a2c565b6103dd611a75565b610392611aca565b6103756107d23660046132bb565b6001600160a01b03918216600090815260056020908152604080832093909416825291909152205460ff1690565b61079461080e3660046130bc565b611ad9565b6014546103759060ff1681565b600061082b82611d9a565b92915050565b606060008054610840906132e5565b80601f016020809104026020016040519081016040528092919081815260200182805461086c906132e5565b80156108b95780601f1061088e576101008083540402835291602001916108b9565b820191906000526020600020905b81548152906001019060200180831161089c57829003601f168201915b5050505050905090565b60006108ce82611dbf565b506000908152600460205260409020546001600160a01b031690565b60006108f5826113d8565b9050806001600160a01b0316836001600160a01b0316036109675760405162461bcd60e51b815260206004820152602160248201527f4552433732313a20617070726f76616c20746f2063757272656e74206f776e656044820152603960f91b60648201526084015b60405180910390fd5b336001600160a01b0382161480610983575061098381336107d2565b6109f55760405162461bcd60e51b815260206004820152603e60248201527f4552433732313a20617070726f76652063616c6c6572206973206e6f7420746f60448201527f6b656e206f776e6572206e6f7220617070726f76656420666f7220616c6c0000606482015260840161095e565b6109ff8383611e1e565b505050565b600260105403610a565760405162461bcd60e51b815260206004820152601f60248201527f5265656e7472616e637947756172643a207265656e7472616e742063616c6c00604482015260640161095e565b600260105560145460ff1615610a7e5760405162461bcd60e51b815260040161095e9061331f565b610a87816113d8565b6001600160a01b0316336001600160a01b031614610b0d5760405162461bcd60e51b815260206004820152603860248201527f4552433732314c656e64696e673a20547279696e6720746f206c656e6420612060448201527f746f6b656e2074686174206973206e6f74206f776e65642e0000000000000000606482015260840161095e565b6001600160a01b0382163303610ba25760405162461bcd60e51b815260206004820152604e60248201527f4552433732314c656e64696e673a204c656e64696e6720746f2073656c66202860448201527f7468652063757272656e74206f776e657227732061646472657373292069732060648201526d3737ba103832b936b4ba3a32b21760911b608482015260a40161095e565b6001600160a01b038216610c1e5760405162461bcd60e51b815260206004820152603f60248201527f4552433732314c656e64696e673a204c6f616e7320746f20746865207a65726f60448201527f20307830206164647265737320617265206e6f74207065726d69747465642e00606482015260840161095e565b6000818152601260205260409020546001600160a01b031615610ca95760405162461bcd60e51b815260206004820152603e60248201527f4552433732314c656e64696e673a20547279696e6720746f206c656e6420612060448201527f746f6b656e207468617420697320616c7265616479206f6e206c6f616e2e0000606482015260840161095e565b610cb4338383610fa4565b600081815260126020908152604080832080546001600160a01b0319163390811790915583526011909152902054610ced81600161338a565b33600090815260116020526040902055601354610d0b90600161338a565b6013556040518281526001600160a01b0384169033907f4d699c2a3f40be9f50773ce2da5d80769decfefe6c4d557178af6e4a412d29a79060200160405180910390a35050600160105550565b610d63335b82611e8c565b610d7f5760405162461bcd60e51b815260040161095e906133a2565b6109ff838383611f0b565b60008281526002602052604081205481906001600160a01b0316610e015760405162461bcd60e51b815260206004820152602860248201527f526f79616c74792072657175657374656420666f72206e6f6e2d6578697374696044820152673733903a37b5b2b760c11b606482015260840161095e565b610e0a836120b2565b915091509250929050565b6000828152600b6020526040902060010154610e30816120fc565b6109ff8383612106565b6000610e4583611438565b8210610ea75760405162461bcd60e51b815260206004820152602b60248201527f455243373231456e756d657261626c653a206f776e657220696e646578206f7560448201526a74206f6620626f756e647360a81b606482015260840161095e565b506001600160a01b03919091166000908152600660209081526040808320938352929052205490565b6001600160a01b0381163314610f405760405162461bcd60e51b815260206004820152602f60248201527f416363657373436f6e74726f6c3a2063616e206f6e6c792072656e6f756e636560448201526e103937b632b9903337b91039b2b63360891b606482015260840161095e565b610f4a828261218c565b5050565b7f7840fe4b17ccfbb599243b7bfe0ed6cbbe3a2279e52b2f14ffdef903de88d25f610f78816120fc565b610f4a826121f3565b6000805160206136df833981519152610f99816120fc565b610fa1612280565b50565b6109ff838383604051806020016040528060008152506118bd565b610fc833610d5d565b610fe45760405162461bcd60e51b815260040161095e906133a2565b610fa1816122d3565b6000610ff860085490565b821061105b5760405162461bcd60e51b815260206004820152602c60248201527f455243373231456e756d657261626c653a20676c6f62616c20696e646578206f60448201526b7574206f6620626f756e647360a01b606482015260840161095e565b6008828154811061106e5761106e6133f0565b90600052602060002001549050919050565b600061108b816120fc565b6001600160a01b0382166110ef5760405162461bcd60e51b815260206004820152602560248201527f4e6577206f776e65722063616e6e6f7420626520746865207a65726f206164646044820152643932b9b99760d91b606482015260840161095e565b601480546001600160a01b0384811661010081810274ffffffffffffffffffffffffffffffffffffffff001985161790945560405193909204169182907f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e090600090a3505050565b7f58376ed7a9634ebd5569e965099321a98f915ee1aad1399d411b22996681698e611181816120fc565b600f5460ff16156111a45760405162461bcd60e51b815260040161095e90613406565b6109ff83838080601f01602080910402602001604051908101604052809392919081815260200183838082843760009201919091525061237a92505050565b60006001600160a01b03821661120b5760405162461bcd60e51b815260040161095e9061344f565b506001600160a01b031660009081526011602052604090205490565b6002601054036112795760405162461bcd60e51b815260206004820152601f60248201527f5265656e7472616e637947756172643a207265656e7472616e742063616c6c00604482015260640161095e565b60026010556000611289826113d8565b9050336001600160a01b038216146113095760405162461bcd60e51b815260206004820152603660248201527f4552433732314c656e64696e673a204f6e6c792074686520626f72726f77657260448201527f2063616e2072657475726e2074686520746f6b656e2e00000000000000000000606482015260840161095e565b600082815260126020908152604080832080546001600160a01b031981169091556001600160a01b031680845260119092529091205461134a6001826134a0565b6001600160a01b038316600090815260116020526040902055601354611372906001906134a0565b601355611380838386610fa4565b816001600160a01b0316836001600160a01b03167f6bc15f6a04fe5ed4600fa4ff46729f772622c11381fc7b57f3e3454c04acb57d866040516113c591815260200190565b60405180910390a3505060016010555050565b6000818152600260205260408120546001600160a01b03168061082b5760405162461bcd60e51b8152602060048201526018602482015277115490cdcc8c4e881a5b9d985b1a59081d1bdad95b88125160421b604482015260640161095e565b60006001600160a01b0382166114a25760405162461bcd60e51b815260206004820152602960248201527f4552433732313a2061646472657373207a65726f206973206e6f7420612076616044820152683634b21037bbb732b960b91b606482015260840161095e565b506001600160a01b031660009081526003602052604090205490565b6000805160206136df8339815191526114d6816120fc565b610fa161238d565b6000918252600b602090815260408084206001600160a01b0393909316845291905290205460ff1690565b7f58376ed7a9634ebd5569e965099321a98f915ee1aad1399d411b22996681698e611533816120fc565b600f5460ff16156115565760405162461bcd60e51b815260040161095e90613406565b6109ff83838080601f0160208091040260200160405190810160405280939291908181526020018383808284376000920191909152506123ca92505050565b6002601054036115e75760405162461bcd60e51b815260206004820152601f60248201527f5265656e7472616e637947756172643a207265656e7472616e742063616c6c00604482015260640161095e565b60026010556000818152601260205260409020546001600160a01b031633811461168c5760405162461bcd60e51b815260206004820152604a60248201527f4552433732314c656e64696e673a204f6e6c7920746865206f726967696e616c60448201527f2f726967687466756c206f776e65722063616e20726563616c6c2061206c6f616064820152693732b2103a37b5b2b71760b11b608482015260a40161095e565b6000611697836113d8565b600084815260126020908152604080832080546001600160a01b03191690556001600160a01b038616835260119091529020549091506116d86001826134a0565b6001600160a01b038416600090815260116020526040902055601354611700906001906134a0565b601381905550611721828486604051806020016040528060008152506123dd565b816001600160a01b0316836001600160a01b03167fb2e0d97c4257c67ef0c641013f79ba97b620da743cef521b451b68b2713d2093866040516113c591815260200190565b606060018054610840906132e5565b6000805160206136df83398151915261178d816120fc565b610fa1612410565b7faeaef46186eb59f884e36929b6d682a6ae35e1e43d8f05f058dcefb92b6014616117bf816120fc565b60145460fa600160a81b90910460ff16106118515760405162461bcd60e51b815260206004820152604660248201527f546865206d6178696d756d206e756d626572206f6620746f6b656e732074686160448201527f742063616e2065766572206265206d696e74656420686173206265656e20726560648201526530b1b432b21760d11b608482015260a40161095e565b6001601460158282829054906101000a900460ff1661187091906134b7565b92506101000a81548160ff021916908360ff1602179055506109ff8383612470565b610f4a33838361248a565b6000805160206136df8339815191526118b5816120fc565b610fa1612558565b6118c73383611e8c565b6118e35760405162461bcd60e51b815260040161095e906133a2565b6118ef848484846123dd565b50505050565b6000818152600260205260409020546060906001600160a01b031661195c5760405162461bcd60e51b815260206004820152601f60248201527f55524920717565727920666f72206e6f6e6578697374656e7420746f6b656e00604482015260640161095e565b61082b8261260a565b6000828152600b6020526040902060010154611980816120fc565b6109ff838361218c565b6060600061199783611438565b905060008167ffffffffffffffff8111156119b4576119b4613185565b6040519080825280602002602001820160405280156119dd578160200160208202803683370190505b50905060005b82811015611a24576119f58582610e3a565b828281518110611a0757611a076133f0565b602090810291909101015280611a1c816134dc565b9150506119e3565b509392505050565b7f7840fe4b17ccfbb599243b7bfe0ed6cbbe3a2279e52b2f14ffdef903de88d25f611a56816120fc565b600c80546001600160a01b0319166001600160a01b0384161790555050565b7fd9f17c44520e5d33006f7504376dd1fae046f60c67eb4e7800f24debd3689180611a9f816120fc565b600f5460ff1615611ac25760405162461bcd60e51b815260040161095e90613406565b610fa1612644565b6060600e8054610840906132e5565b60606001600160a01b038216611b015760405162461bcd60e51b815260040161095e9061344f565b6000611b0c836111e3565b90506000611b1960085490565b90506000808367ffffffffffffffff811115611b3757611b37613185565b604051908082528060200260200182016040528015611b60578160200160208202803683370190505b5090508315611bee576000805b8582108015611b7b57508481105b15611beb57611b8981610fed565b6000818152601260205260409020549094506001600160a01b03808a16911603611bd95783838381518110611bc057611bc06133f0565b602090810291909101015281611bd5816134dc565b9250505b80611be3816134dc565b915050611b6d565b50505b95945050505050565b60606000611c068360026134f5565b611c1190600261338a565b67ffffffffffffffff811115611c2957611c29613185565b6040519080825280601f01601f191660200182016040528015611c53576020820181803683370190505b509050600360fc1b81600081518110611c6e57611c6e6133f0565b60200101906001600160f81b031916908160001a905350600f60fb1b81600181518110611c9d57611c9d6133f0565b60200101906001600160f81b031916908160001a9053506000611cc18460026134f5565b611ccc90600161338a565b90505b6001811115611d44576f181899199a1a9b1b9c1cb0b131b232b360811b85600f1660108110611d0057611d006133f0565b1a60f81b828281518110611d1657611d166133f0565b60200101906001600160f81b031916908160001a90535060049490941c93611d3d81613514565b9050611ccf565b508315611d935760405162461bcd60e51b815260206004820181905260248201527f537472696e67733a20686578206c656e67746820696e73756666696369656e74604482015260640161095e565b9392505050565b60006001600160e01b0319821663152a902d60e11b148061082b575061082b82612676565b6000818152600260205260409020546001600160a01b0316610fa15760405162461bcd60e51b8152602060048201526018602482015277115490cdcc8c4e881a5b9d985b1a59081d1bdad95b88125160421b604482015260640161095e565b600081815260046020526040902080546001600160a01b0319166001600160a01b0384169081179091558190611e53826113d8565b6001600160a01b03167f8c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b92560405160405180910390a45050565b600080611e98836113d8565b9050806001600160a01b0316846001600160a01b03161480611edf57506001600160a01b0380821660009081526005602090815260408083209388168352929052205460ff165b80611f035750836001600160a01b0316611ef8846108c3565b6001600160a01b0316145b949350505050565b826001600160a01b0316611f1e826113d8565b6001600160a01b031614611f825760405162461bcd60e51b815260206004820152602560248201527f4552433732313a207472616e736665722066726f6d20696e636f72726563742060448201526437bbb732b960d91b606482015260840161095e565b6001600160a01b038216611fe45760405162461bcd60e51b8152602060048201526024808201527f4552433732313a207472616e7366657220746f20746865207a65726f206164646044820152637265737360e01b606482015260840161095e565b611fef83838361269b565b611ffa600082611e1e565b6001600160a01b03831660009081526003602052604081208054600192906120239084906134a0565b90915550506001600160a01b038216600090815260036020526040812080546001929061205190849061338a565b909155505060008181526002602052604080822080546001600160a01b0319166001600160a01b0386811691821790925591518493918716917fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef91a4505050565b600c546000908190819061ffff600160b01b82048116916120dc91600160a01b90910416866134f5565b6120e69190613541565b600c546001600160a01b03169590945092505050565b610fa181336126ae565b61211082826114de565b610f4a576000828152600b602090815260408083206001600160a01b03851684529091529020805460ff191660011790556121483390565b6001600160a01b0316816001600160a01b0316837f2f8788117e7eff1d82e926ec794901d17c78024a50270940304540a733656f0d60405160405180910390a45050565b61219682826114de565b15610f4a576000828152600b602090815260408083206001600160a01b0385168085529252808320805460ff1916905551339285917ff6391f5c32d9c69d2a47ea670b442974b53935d1edc7fd64eb21e047a839171b9190a45050565b600c5461ffff600160b01b9091048116908216111561225e5760405162461bcd60e51b815260206004820152602160248201527f526f79616c7479206665652077696c6c206578636565642073616c65507269636044820152606560f81b606482015260840161095e565b600c805461ffff909216600160a01b0261ffff60a01b19909216919091179055565b612288612712565b600a805460ff191690557f5db9ee0a495bf2e6ff9c91a7834c1ba4fdd244a5e8aa4e537bd38aeae4b073aa335b6040516001600160a01b0390911681526020015b60405180910390a1565b60006122de826113d8565b90506122ec8160008461269b565b6122f7600083611e1e565b6001600160a01b03811660009081526003602052604081208054600192906123209084906134a0565b909155505060008281526002602052604080822080546001600160a01b0319169055518391906001600160a01b038416907fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef908390a45050565b8051610f4a90600d906020840190612e78565b61239561275d565b600a805460ff191660011790557f62e78cea01bee320cd4e420270b5ea74000d11b0c9f74754ebdbfc544b05a2586122b53390565b8051610f4a90600e906020840190612e78565b6123e8848484611f0b565b6123f4848484846127a3565b6118ef5760405162461bcd60e51b815260040161095e90613555565b60145460ff16156124335760405162461bcd60e51b815260040161095e9061331f565b6014805460ff191660011790556040513381527f973debc148447544f7c98f7d98b9c83992c795313da2f99e314166de48ac0d11906020016122c9565b610f4a8282604051806020016040528060008152506128a4565b816001600160a01b0316836001600160a01b0316036124eb5760405162461bcd60e51b815260206004820152601960248201527f4552433732313a20617070726f766520746f2063616c6c657200000000000000604482015260640161095e565b6001600160a01b03838116600081815260056020908152604080832094871680845294825291829020805460ff191686151590811790915591519182527f17307eab39ab6107e8899845ad3d59bd9653f200f220920489ca2b5937696c31910160405180910390a3505050565b60145460ff166125d05760405162461bcd60e51b815260206004820152603e60248201527f4552433732314c656e64696e673a204c656e64696e67206f6620746f6b656e7360448201527f20697320616c726561647920696e20756e7061757365642073746174652e0000606482015260840161095e565b6014805460ff191690556040513381527f0d844c498c8f00ba188127e29f77aea468f9c8e768bd6ddb88472fe704b2a456906020016122c9565b60606126146128d7565b61261d836128e6565b60405160200161262e9291906135a7565b6040516020818303038152906040529050919050565b600f5460ff16156126675760405162461bcd60e51b815260040161095e90613406565b600f805460ff19166001179055565b60006001600160e01b03198216637965db0b60e01b148061082b575061082b826129e7565b6126a361275d565b6109ff838383612a0c565b6126b882826114de565b610f4a576126d0816001600160a01b03166014611bf7565b6126db836020611bf7565b6040516020016126ec9291906135e6565b60408051601f198184030181529082905262461bcd60e51b825261095e91600401612f9c565b600a5460ff1661275b5760405162461bcd60e51b815260206004820152601460248201527314185d5cd8589b194e881b9bdd081c185d5cd95960621b604482015260640161095e565b565b600a5460ff161561275b5760405162461bcd60e51b815260206004820152601060248201526f14185d5cd8589b194e881c185d5cd95960821b604482015260640161095e565b60006001600160a01b0384163b1561289957604051630a85bd0160e11b81526001600160a01b0385169063150b7a02906127e790339089908890889060040161365b565b6020604051808303816000875af1925050508015612822575060408051601f3d908101601f1916820190925261281f91810190613697565b60015b61287f573d808015612850576040519150601f19603f3d011682016040523d82523d6000602084013e612855565b606091505b5080516000036128775760405162461bcd60e51b815260040161095e90613555565b805181602001fd5b6001600160e01b031916630a85bd0160e11b149050611f03565b506001949350505050565b6128ae8383612a92565b6128bb60008484846127a3565b6109ff5760405162461bcd60e51b815260040161095e90613555565b6060600d8054610840906132e5565b60608160000361290d5750506040805180820190915260018152600360fc1b602082015290565b8160005b81156129375780612921816134dc565b91506129309050600a83613541565b9150612911565b60008167ffffffffffffffff81111561295257612952613185565b6040519080825280601f01601f19166020018201604052801561297c576020820181803683370190505b5090505b8415611f03576129916001836134a0565b915061299e600a866136b4565b6129a990603061338a565b60f81b8183815181106129be576129be6133f0565b60200101906001600160f81b031916908160001a9053506129e0600a86613541565b9450612980565b60006001600160e01b0319821663780e9d6360e01b148061082b575061082b82612be0565b612a17838383612c30565b6000818152601260205260409020546001600160a01b0316156109ff5760405162461bcd60e51b815260206004820152602d60248201527f4552433732314c656e64696e673a2043616e6e6f74207472616e73666572207460448201526c37b5b2b71037b7103637b0b71760991b606482015260840161095e565b6001600160a01b038216612ae85760405162461bcd60e51b815260206004820181905260248201527f4552433732313a206d696e7420746f20746865207a65726f2061646472657373604482015260640161095e565b6000818152600260205260409020546001600160a01b031615612b4d5760405162461bcd60e51b815260206004820152601c60248201527f4552433732313a20746f6b656e20616c7265616479206d696e74656400000000604482015260640161095e565b612b596000838361269b565b6001600160a01b0382166000908152600360205260408120805460019290612b8290849061338a565b909155505060008181526002602052604080822080546001600160a01b0319166001600160a01b03861690811790915590518392907fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef908290a45050565b60006001600160e01b031982166380ac58cd60e01b1480612c1157506001600160e01b03198216635b5e139f60e01b145b8061082b57506301ffc9a760e01b6001600160e01b031983161461082b565b6001600160a01b038316612c8b57612c8681600880546000838152600960205260408120829055600182018355919091527ff3f7a9fe364faab93b216da50a3214154f22a0a2b415b23a84c8169e8b636ee30155565b612cae565b816001600160a01b0316836001600160a01b031614612cae57612cae8382612ce8565b6001600160a01b038216612cc5576109ff81612d85565b826001600160a01b0316826001600160a01b0316146109ff576109ff8282612e34565b60006001612cf584611438565b612cff91906134a0565b600083815260076020526040902054909150808214612d52576001600160a01b03841660009081526006602090815260408083208584528252808320548484528184208190558352600790915290208190555b5060009182526007602090815260408084208490556001600160a01b039094168352600681528383209183525290812055565b600854600090612d97906001906134a0565b60008381526009602052604081205460088054939450909284908110612dbf57612dbf6133f0565b906000526020600020015490508060088381548110612de057612de06133f0565b6000918252602080832090910192909255828152600990915260408082208490558582528120556008805480612e1857612e186136c8565b6001900381819060005260206000200160009055905550505050565b6000612e3f83611438565b6001600160a01b039093166000908152600660209081526040808320868452825280832085905593825260079052919091209190915550565b828054612e84906132e5565b90600052602060002090601f016020900481019282612ea65760008555612eec565b82601f10612ebf57805160ff1916838001178555612eec565b82800160010185558215612eec579182015b82811115612eec578251825591602001919060010190612ed1565b50612ef8929150612efc565b5090565b5b80821115612ef85760008155600101612efd565b6001600160e01b031981168114610fa157600080fd5b600060208284031215612f3957600080fd5b8135611d9381612f11565b60005b83811015612f5f578181015183820152602001612f47565b838111156118ef5750506000910152565b60008151808452612f88816020860160208601612f44565b601f01601f19169290920160200192915050565b602081526000611d936020830184612f70565b600060208284031215612fc157600080fd5b5035919050565b80356001600160a01b0381168114612fdf57600080fd5b919050565b60008060408385031215612ff757600080fd5b61300083612fc8565b946020939093013593505050565b60008060006060848603121561302357600080fd5b61302c84612fc8565b925061303a60208501612fc8565b9150604084013590509250925092565b6000806040838503121561305d57600080fd5b50508035926020909101359150565b6000806040838503121561307f57600080fd5b8235915061308f60208401612fc8565b90509250929050565b6000602082840312156130aa57600080fd5b813561ffff81168114611d9357600080fd5b6000602082840312156130ce57600080fd5b611d9382612fc8565b600080602083850312156130ea57600080fd5b823567ffffffffffffffff8082111561310257600080fd5b818501915085601f83011261311657600080fd5b81358181111561312557600080fd5b86602082850101111561313757600080fd5b60209290920196919550909350505050565b6000806040838503121561315c57600080fd5b61316583612fc8565b91506020830135801515811461317a57600080fd5b809150509250929050565b634e487b7160e01b600052604160045260246000fd5b600080600080608085870312156131b157600080fd5b6131ba85612fc8565b93506131c860208601612fc8565b925060408501359150606085013567ffffffffffffffff808211156131ec57600080fd5b818701915087601f83011261320057600080fd5b81358181111561321257613212613185565b604051601f8201601f19908116603f0116810190838211818310171561323a5761323a613185565b816040528281528a602084870101111561325357600080fd5b82602086016020830137600060208483010152809550505050505092959194509250565b6020808252825182820181905260009190848201906040850190845b818110156132af57835183529284019291840191600101613293565b50909695505050505050565b600080604083850312156132ce57600080fd5b6132d783612fc8565b915061308f60208401612fc8565b600181811c908216806132f957607f821691505b60208210810361331957634e487b7160e01b600052602260045260246000fd5b50919050565b60208082526035908201527f4552433732314c656e64696e673a204c656e64696e67206f6620746f6b656e736040820152741034b99031bab93932b73a363c903830bab9b2b21760591b606082015260800190565b634e487b7160e01b600052601160045260246000fd5b6000821982111561339d5761339d613374565b500190565b6020808252602e908201527f4552433732313a2063616c6c6572206973206e6f7420746f6b656e206f776e6560408201526d1c881b9bdc88185c1c1c9bdd995960921b606082015260800190565b634e487b7160e01b600052603260045260246000fd5b60208082526029908201527f5552494d616e616765723a20555249732068617665206265656e2066726f7a6560408201526837103337b932bb32b960b91b606082015260800190565b60208082526031908201527f4552433732314c656e64696e673a2042616c616e636520717565727920666f7260408201527020746865207a65726f206164647265737360781b606082015260800190565b6000828210156134b2576134b2613374565b500390565b600060ff821660ff84168060ff038211156134d4576134d4613374565b019392505050565b6000600182016134ee576134ee613374565b5060010190565b600081600019048311821515161561350f5761350f613374565b500290565b60008161352357613523613374565b506000190190565b634e487b7160e01b600052601260045260246000fd5b6000826135505761355061352b565b500490565b60208082526032908201527f4552433732313a207472616e7366657220746f206e6f6e20455243373231526560408201527131b2b4bb32b91034b6b83632b6b2b73a32b960711b606082015260800190565b600083516135b9818460208801612f44565b8351908301906135cd818360208801612f44565b64173539b7b760d91b9101908152600501949350505050565b7f416363657373436f6e74726f6c3a206163636f756e742000000000000000000081526000835161361e816017850160208801612f44565b7001034b99036b4b9b9b4b733903937b6329607d1b601791840191820152835161364f816028840160208801612f44565b01602801949350505050565b60006001600160a01b0380871683528086166020840152508360408301526080606083015261368d6080830184612f70565b9695505050505050565b6000602082840312156136a957600080fd5b8151611d9381612f11565b6000826136c3576136c361352b565b500690565b634e487b7160e01b600052603160045260246000fdfe372d55e37651a7c6e1940a3fb8628e4b6122a3c1a8b2b70aee13e07228604567a264697066735822122030f13e4ab7faf105aff824bf91a10d6f01e5713a1c3643a22fb13f7606bebb0964736f6c634300080d0033000000000000000000000000000000000000000000000000000000000000008000000000000000000000000000000000000000000000000000000000000000c0000000000000000000000000000000000000000000000000000000000000010000000000000000000000000000000000000000000000000000000000000001a0000000000000000000000000000000000000000000000000000000000000000c416666654d6974576166666500000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000003414d570000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000006268747470733a2f2f6177732d73616d2d616666656e2d64622d7075626c69632d616666656e2d6d657461646174612e73332e65752d63656e7472616c2d312e616d617a6f6e6177732e636f6d2f616666656d697477616666653732312f6a736f6e2f000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000006c68747470733a2f2f6177732d73616d2d616666656e2d64622d7075626c69632d616666656e2d6d657461646174612e73332e65752d63656e7472616c2d312e616d617a6f6e6177732e636f6d2f616666656d697477616666653732312f636f6c6c656374696f6e2e6a736f6e0000000000000000000000000000000000000000

Deployed Bytecode

0x608060405234801561001057600080fd5b506004361061035d5760003560e01c80637f48e368116101d3578063acd36b3911610104578063dba028c2116100a2578063e8a3d4851161007c578063e8a3d485146107bc578063e985e9c5146107c4578063ed5a2ec114610800578063f2a808d61461081357600080fd5b8063dba028c214610781578063dde36719146107a1578063ddead0eb146107b457600080fd5b8063b88d4fde116100de578063b88d4fde14610721578063c87b56dd14610734578063d418a9d314610747578063d547741f1461076e57600080fd5b8063acd36b39146106dd578063b047eb0214610704578063b4193d8a1461071957600080fd5b806395d89b4111610171578063a14481941161014b578063a144819414610686578063a217fddf14610699578063a22cb465146106a1578063a583c928146106b457600080fd5b806395d89b411461066b57806397c47d7c146106735780639e6e5cfc1461067e57600080fd5b806391d14854116101ad57806391d148541461060b57806392afc33a1461061e578063938e3d7b1461064557806393b068a41461065857600080fd5b80637f48e368146105c45780638456cb59146105eb5780638da5cb5b146105f357600080fd5b80633a838c2a116102ad57806355f804b31161024b5780635e3dda6f116102255780635e3dda6f1461056b5780635f9be2ac1461057e5780636352211e1461059e57806370a08231146105b157600080fd5b806355f804b31461053a5780635c975abb1461054d5780635e03a6aa1461055857600080fd5b806342842e0e1161028757806342842e0e146104ee57806342966c68146105015780634f6ccce71461051457806355b7c4431461052757600080fd5b80633a838c2a146104b85780633f4ba83a146104c05780634230baee146104c857600080fd5b806323b872dd1161031a5780632f2ff15d116102f45780632f2ff15d1461046c5780632f745c591461047f57806336568abe146104925780633746e948146104a557600080fd5b806323b872dd14610404578063248a9ca3146104175780632a55205a1461043a57600080fd5b806301ffc9a71461036257806306fdde031461038a578063081812fc1461039f578063095ea7b3146103ca57806318160ddd146103df578063185cc751146103f1575b600080fd5b610375610370366004612f27565b610820565b60405190151581526020015b60405180910390f35b610392610831565b6040516103819190612f9c565b6103b26103ad366004612faf565b6108c3565b6040516001600160a01b039091168152602001610381565b6103dd6103d8366004612fe4565b6108ea565b005b6008545b604051908152602001610381565b6103dd6103ff366004612fe4565b610a04565b6103dd61041236600461300e565b610d58565b6103e3610425366004612faf565b6000908152600b602052604090206001015490565b61044d61044836600461304a565b610d8a565b604080516001600160a01b039093168352602083019190915201610381565b6103dd61047a36600461306c565b610e15565b6103e361048d366004612fe4565b610e3a565b6103dd6104a036600461306c565b610ed0565b6103dd6104b3366004613098565b610f4e565b6013546103e3565b6103dd610f81565b6014546104dc90600160a81b900460ff1681565b60405160ff9091168152602001610381565b6103dd6104fc36600461300e565b610fa4565b6103dd61050f366004612faf565b610fbf565b6103e3610522366004612faf565b610fed565b6103dd6105353660046130bc565b611080565b6103dd6105483660046130d7565b611157565b600a5460ff16610375565b6103e36105663660046130bc565b6111e3565b6103dd610579366004612faf565b611227565b6103e361058c3660046130bc565b60116020526000908152604090205481565b6103b26105ac366004612faf565b6113d8565b6103e36105bf3660046130bc565b611438565b6103e37f58376ed7a9634ebd5569e965099321a98f915ee1aad1399d411b22996681698e81565b6103dd6114be565b6014546103b29061010090046001600160a01b031681565b61037561061936600461306c565b6114de565b6103e37faeaef46186eb59f884e36929b6d682a6ae35e1e43d8f05f058dcefb92b60146181565b6103dd6106533660046130d7565b611509565b6103dd610666366004612faf565b611595565b610392611766565b600f5460ff16610375565b6103dd611775565b6103dd610694366004612fe4565b611795565b6103e3600081565b6103dd6106af366004613149565b611892565b6103b26106c2366004612faf565b6012602052600090815260409020546001600160a01b031681565b6103e37fd9f17c44520e5d33006f7504376dd1fae046f60c67eb4e7800f24debd368918081565b6103e36000805160206136df83398151915281565b6103dd61189d565b6103dd61072f36600461319b565b6118bd565b610392610742366004612faf565b6118f5565b6103e37f7840fe4b17ccfbb599243b7bfe0ed6cbbe3a2279e52b2f14ffdef903de88d25f81565b6103dd61077c36600461306c565b611965565b61079461078f3660046130bc565b61198a565b6040516103819190613277565b6103dd6107af3660046130bc565b611a2c565b6103dd611a75565b610392611aca565b6103756107d23660046132bb565b6001600160a01b03918216600090815260056020908152604080832093909416825291909152205460ff1690565b61079461080e3660046130bc565b611ad9565b6014546103759060ff1681565b600061082b82611d9a565b92915050565b606060008054610840906132e5565b80601f016020809104026020016040519081016040528092919081815260200182805461086c906132e5565b80156108b95780601f1061088e576101008083540402835291602001916108b9565b820191906000526020600020905b81548152906001019060200180831161089c57829003601f168201915b5050505050905090565b60006108ce82611dbf565b506000908152600460205260409020546001600160a01b031690565b60006108f5826113d8565b9050806001600160a01b0316836001600160a01b0316036109675760405162461bcd60e51b815260206004820152602160248201527f4552433732313a20617070726f76616c20746f2063757272656e74206f776e656044820152603960f91b60648201526084015b60405180910390fd5b336001600160a01b0382161480610983575061098381336107d2565b6109f55760405162461bcd60e51b815260206004820152603e60248201527f4552433732313a20617070726f76652063616c6c6572206973206e6f7420746f60448201527f6b656e206f776e6572206e6f7220617070726f76656420666f7220616c6c0000606482015260840161095e565b6109ff8383611e1e565b505050565b600260105403610a565760405162461bcd60e51b815260206004820152601f60248201527f5265656e7472616e637947756172643a207265656e7472616e742063616c6c00604482015260640161095e565b600260105560145460ff1615610a7e5760405162461bcd60e51b815260040161095e9061331f565b610a87816113d8565b6001600160a01b0316336001600160a01b031614610b0d5760405162461bcd60e51b815260206004820152603860248201527f4552433732314c656e64696e673a20547279696e6720746f206c656e6420612060448201527f746f6b656e2074686174206973206e6f74206f776e65642e0000000000000000606482015260840161095e565b6001600160a01b0382163303610ba25760405162461bcd60e51b815260206004820152604e60248201527f4552433732314c656e64696e673a204c656e64696e6720746f2073656c66202860448201527f7468652063757272656e74206f776e657227732061646472657373292069732060648201526d3737ba103832b936b4ba3a32b21760911b608482015260a40161095e565b6001600160a01b038216610c1e5760405162461bcd60e51b815260206004820152603f60248201527f4552433732314c656e64696e673a204c6f616e7320746f20746865207a65726f60448201527f20307830206164647265737320617265206e6f74207065726d69747465642e00606482015260840161095e565b6000818152601260205260409020546001600160a01b031615610ca95760405162461bcd60e51b815260206004820152603e60248201527f4552433732314c656e64696e673a20547279696e6720746f206c656e6420612060448201527f746f6b656e207468617420697320616c7265616479206f6e206c6f616e2e0000606482015260840161095e565b610cb4338383610fa4565b600081815260126020908152604080832080546001600160a01b0319163390811790915583526011909152902054610ced81600161338a565b33600090815260116020526040902055601354610d0b90600161338a565b6013556040518281526001600160a01b0384169033907f4d699c2a3f40be9f50773ce2da5d80769decfefe6c4d557178af6e4a412d29a79060200160405180910390a35050600160105550565b610d63335b82611e8c565b610d7f5760405162461bcd60e51b815260040161095e906133a2565b6109ff838383611f0b565b60008281526002602052604081205481906001600160a01b0316610e015760405162461bcd60e51b815260206004820152602860248201527f526f79616c74792072657175657374656420666f72206e6f6e2d6578697374696044820152673733903a37b5b2b760c11b606482015260840161095e565b610e0a836120b2565b915091509250929050565b6000828152600b6020526040902060010154610e30816120fc565b6109ff8383612106565b6000610e4583611438565b8210610ea75760405162461bcd60e51b815260206004820152602b60248201527f455243373231456e756d657261626c653a206f776e657220696e646578206f7560448201526a74206f6620626f756e647360a81b606482015260840161095e565b506001600160a01b03919091166000908152600660209081526040808320938352929052205490565b6001600160a01b0381163314610f405760405162461bcd60e51b815260206004820152602f60248201527f416363657373436f6e74726f6c3a2063616e206f6e6c792072656e6f756e636560448201526e103937b632b9903337b91039b2b63360891b606482015260840161095e565b610f4a828261218c565b5050565b7f7840fe4b17ccfbb599243b7bfe0ed6cbbe3a2279e52b2f14ffdef903de88d25f610f78816120fc565b610f4a826121f3565b6000805160206136df833981519152610f99816120fc565b610fa1612280565b50565b6109ff838383604051806020016040528060008152506118bd565b610fc833610d5d565b610fe45760405162461bcd60e51b815260040161095e906133a2565b610fa1816122d3565b6000610ff860085490565b821061105b5760405162461bcd60e51b815260206004820152602c60248201527f455243373231456e756d657261626c653a20676c6f62616c20696e646578206f60448201526b7574206f6620626f756e647360a01b606482015260840161095e565b6008828154811061106e5761106e6133f0565b90600052602060002001549050919050565b600061108b816120fc565b6001600160a01b0382166110ef5760405162461bcd60e51b815260206004820152602560248201527f4e6577206f776e65722063616e6e6f7420626520746865207a65726f206164646044820152643932b9b99760d91b606482015260840161095e565b601480546001600160a01b0384811661010081810274ffffffffffffffffffffffffffffffffffffffff001985161790945560405193909204169182907f8be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e090600090a3505050565b7f58376ed7a9634ebd5569e965099321a98f915ee1aad1399d411b22996681698e611181816120fc565b600f5460ff16156111a45760405162461bcd60e51b815260040161095e90613406565b6109ff83838080601f01602080910402602001604051908101604052809392919081815260200183838082843760009201919091525061237a92505050565b60006001600160a01b03821661120b5760405162461bcd60e51b815260040161095e9061344f565b506001600160a01b031660009081526011602052604090205490565b6002601054036112795760405162461bcd60e51b815260206004820152601f60248201527f5265656e7472616e637947756172643a207265656e7472616e742063616c6c00604482015260640161095e565b60026010556000611289826113d8565b9050336001600160a01b038216146113095760405162461bcd60e51b815260206004820152603660248201527f4552433732314c656e64696e673a204f6e6c792074686520626f72726f77657260448201527f2063616e2072657475726e2074686520746f6b656e2e00000000000000000000606482015260840161095e565b600082815260126020908152604080832080546001600160a01b031981169091556001600160a01b031680845260119092529091205461134a6001826134a0565b6001600160a01b038316600090815260116020526040902055601354611372906001906134a0565b601355611380838386610fa4565b816001600160a01b0316836001600160a01b03167f6bc15f6a04fe5ed4600fa4ff46729f772622c11381fc7b57f3e3454c04acb57d866040516113c591815260200190565b60405180910390a3505060016010555050565b6000818152600260205260408120546001600160a01b03168061082b5760405162461bcd60e51b8152602060048201526018602482015277115490cdcc8c4e881a5b9d985b1a59081d1bdad95b88125160421b604482015260640161095e565b60006001600160a01b0382166114a25760405162461bcd60e51b815260206004820152602960248201527f4552433732313a2061646472657373207a65726f206973206e6f7420612076616044820152683634b21037bbb732b960b91b606482015260840161095e565b506001600160a01b031660009081526003602052604090205490565b6000805160206136df8339815191526114d6816120fc565b610fa161238d565b6000918252600b602090815260408084206001600160a01b0393909316845291905290205460ff1690565b7f58376ed7a9634ebd5569e965099321a98f915ee1aad1399d411b22996681698e611533816120fc565b600f5460ff16156115565760405162461bcd60e51b815260040161095e90613406565b6109ff83838080601f0160208091040260200160405190810160405280939291908181526020018383808284376000920191909152506123ca92505050565b6002601054036115e75760405162461bcd60e51b815260206004820152601f60248201527f5265656e7472616e637947756172643a207265656e7472616e742063616c6c00604482015260640161095e565b60026010556000818152601260205260409020546001600160a01b031633811461168c5760405162461bcd60e51b815260206004820152604a60248201527f4552433732314c656e64696e673a204f6e6c7920746865206f726967696e616c60448201527f2f726967687466756c206f776e65722063616e20726563616c6c2061206c6f616064820152693732b2103a37b5b2b71760b11b608482015260a40161095e565b6000611697836113d8565b600084815260126020908152604080832080546001600160a01b03191690556001600160a01b038616835260119091529020549091506116d86001826134a0565b6001600160a01b038416600090815260116020526040902055601354611700906001906134a0565b601381905550611721828486604051806020016040528060008152506123dd565b816001600160a01b0316836001600160a01b03167fb2e0d97c4257c67ef0c641013f79ba97b620da743cef521b451b68b2713d2093866040516113c591815260200190565b606060018054610840906132e5565b6000805160206136df83398151915261178d816120fc565b610fa1612410565b7faeaef46186eb59f884e36929b6d682a6ae35e1e43d8f05f058dcefb92b6014616117bf816120fc565b60145460fa600160a81b90910460ff16106118515760405162461bcd60e51b815260206004820152604660248201527f546865206d6178696d756d206e756d626572206f6620746f6b656e732074686160448201527f742063616e2065766572206265206d696e74656420686173206265656e20726560648201526530b1b432b21760d11b608482015260a40161095e565b6001601460158282829054906101000a900460ff1661187091906134b7565b92506101000a81548160ff021916908360ff1602179055506109ff8383612470565b610f4a33838361248a565b6000805160206136df8339815191526118b5816120fc565b610fa1612558565b6118c73383611e8c565b6118e35760405162461bcd60e51b815260040161095e906133a2565b6118ef848484846123dd565b50505050565b6000818152600260205260409020546060906001600160a01b031661195c5760405162461bcd60e51b815260206004820152601f60248201527f55524920717565727920666f72206e6f6e6578697374656e7420746f6b656e00604482015260640161095e565b61082b8261260a565b6000828152600b6020526040902060010154611980816120fc565b6109ff838361218c565b6060600061199783611438565b905060008167ffffffffffffffff8111156119b4576119b4613185565b6040519080825280602002602001820160405280156119dd578160200160208202803683370190505b50905060005b82811015611a24576119f58582610e3a565b828281518110611a0757611a076133f0565b602090810291909101015280611a1c816134dc565b9150506119e3565b509392505050565b7f7840fe4b17ccfbb599243b7bfe0ed6cbbe3a2279e52b2f14ffdef903de88d25f611a56816120fc565b600c80546001600160a01b0319166001600160a01b0384161790555050565b7fd9f17c44520e5d33006f7504376dd1fae046f60c67eb4e7800f24debd3689180611a9f816120fc565b600f5460ff1615611ac25760405162461bcd60e51b815260040161095e90613406565b610fa1612644565b6060600e8054610840906132e5565b60606001600160a01b038216611b015760405162461bcd60e51b815260040161095e9061344f565b6000611b0c836111e3565b90506000611b1960085490565b90506000808367ffffffffffffffff811115611b3757611b37613185565b604051908082528060200260200182016040528015611b60578160200160208202803683370190505b5090508315611bee576000805b8582108015611b7b57508481105b15611beb57611b8981610fed565b6000818152601260205260409020549094506001600160a01b03808a16911603611bd95783838381518110611bc057611bc06133f0565b602090810291909101015281611bd5816134dc565b9250505b80611be3816134dc565b915050611b6d565b50505b95945050505050565b60606000611c068360026134f5565b611c1190600261338a565b67ffffffffffffffff811115611c2957611c29613185565b6040519080825280601f01601f191660200182016040528015611c53576020820181803683370190505b509050600360fc1b81600081518110611c6e57611c6e6133f0565b60200101906001600160f81b031916908160001a905350600f60fb1b81600181518110611c9d57611c9d6133f0565b60200101906001600160f81b031916908160001a9053506000611cc18460026134f5565b611ccc90600161338a565b90505b6001811115611d44576f181899199a1a9b1b9c1cb0b131b232b360811b85600f1660108110611d0057611d006133f0565b1a60f81b828281518110611d1657611d166133f0565b60200101906001600160f81b031916908160001a90535060049490941c93611d3d81613514565b9050611ccf565b508315611d935760405162461bcd60e51b815260206004820181905260248201527f537472696e67733a20686578206c656e67746820696e73756666696369656e74604482015260640161095e565b9392505050565b60006001600160e01b0319821663152a902d60e11b148061082b575061082b82612676565b6000818152600260205260409020546001600160a01b0316610fa15760405162461bcd60e51b8152602060048201526018602482015277115490cdcc8c4e881a5b9d985b1a59081d1bdad95b88125160421b604482015260640161095e565b600081815260046020526040902080546001600160a01b0319166001600160a01b0384169081179091558190611e53826113d8565b6001600160a01b03167f8c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b92560405160405180910390a45050565b600080611e98836113d8565b9050806001600160a01b0316846001600160a01b03161480611edf57506001600160a01b0380821660009081526005602090815260408083209388168352929052205460ff165b80611f035750836001600160a01b0316611ef8846108c3565b6001600160a01b0316145b949350505050565b826001600160a01b0316611f1e826113d8565b6001600160a01b031614611f825760405162461bcd60e51b815260206004820152602560248201527f4552433732313a207472616e736665722066726f6d20696e636f72726563742060448201526437bbb732b960d91b606482015260840161095e565b6001600160a01b038216611fe45760405162461bcd60e51b8152602060048201526024808201527f4552433732313a207472616e7366657220746f20746865207a65726f206164646044820152637265737360e01b606482015260840161095e565b611fef83838361269b565b611ffa600082611e1e565b6001600160a01b03831660009081526003602052604081208054600192906120239084906134a0565b90915550506001600160a01b038216600090815260036020526040812080546001929061205190849061338a565b909155505060008181526002602052604080822080546001600160a01b0319166001600160a01b0386811691821790925591518493918716917fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef91a4505050565b600c546000908190819061ffff600160b01b82048116916120dc91600160a01b90910416866134f5565b6120e69190613541565b600c546001600160a01b03169590945092505050565b610fa181336126ae565b61211082826114de565b610f4a576000828152600b602090815260408083206001600160a01b03851684529091529020805460ff191660011790556121483390565b6001600160a01b0316816001600160a01b0316837f2f8788117e7eff1d82e926ec794901d17c78024a50270940304540a733656f0d60405160405180910390a45050565b61219682826114de565b15610f4a576000828152600b602090815260408083206001600160a01b0385168085529252808320805460ff1916905551339285917ff6391f5c32d9c69d2a47ea670b442974b53935d1edc7fd64eb21e047a839171b9190a45050565b600c5461ffff600160b01b9091048116908216111561225e5760405162461bcd60e51b815260206004820152602160248201527f526f79616c7479206665652077696c6c206578636565642073616c65507269636044820152606560f81b606482015260840161095e565b600c805461ffff909216600160a01b0261ffff60a01b19909216919091179055565b612288612712565b600a805460ff191690557f5db9ee0a495bf2e6ff9c91a7834c1ba4fdd244a5e8aa4e537bd38aeae4b073aa335b6040516001600160a01b0390911681526020015b60405180910390a1565b60006122de826113d8565b90506122ec8160008461269b565b6122f7600083611e1e565b6001600160a01b03811660009081526003602052604081208054600192906123209084906134a0565b909155505060008281526002602052604080822080546001600160a01b0319169055518391906001600160a01b038416907fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef908390a45050565b8051610f4a90600d906020840190612e78565b61239561275d565b600a805460ff191660011790557f62e78cea01bee320cd4e420270b5ea74000d11b0c9f74754ebdbfc544b05a2586122b53390565b8051610f4a90600e906020840190612e78565b6123e8848484611f0b565b6123f4848484846127a3565b6118ef5760405162461bcd60e51b815260040161095e90613555565b60145460ff16156124335760405162461bcd60e51b815260040161095e9061331f565b6014805460ff191660011790556040513381527f973debc148447544f7c98f7d98b9c83992c795313da2f99e314166de48ac0d11906020016122c9565b610f4a8282604051806020016040528060008152506128a4565b816001600160a01b0316836001600160a01b0316036124eb5760405162461bcd60e51b815260206004820152601960248201527f4552433732313a20617070726f766520746f2063616c6c657200000000000000604482015260640161095e565b6001600160a01b03838116600081815260056020908152604080832094871680845294825291829020805460ff191686151590811790915591519182527f17307eab39ab6107e8899845ad3d59bd9653f200f220920489ca2b5937696c31910160405180910390a3505050565b60145460ff166125d05760405162461bcd60e51b815260206004820152603e60248201527f4552433732314c656e64696e673a204c656e64696e67206f6620746f6b656e7360448201527f20697320616c726561647920696e20756e7061757365642073746174652e0000606482015260840161095e565b6014805460ff191690556040513381527f0d844c498c8f00ba188127e29f77aea468f9c8e768bd6ddb88472fe704b2a456906020016122c9565b60606126146128d7565b61261d836128e6565b60405160200161262e9291906135a7565b6040516020818303038152906040529050919050565b600f5460ff16156126675760405162461bcd60e51b815260040161095e90613406565b600f805460ff19166001179055565b60006001600160e01b03198216637965db0b60e01b148061082b575061082b826129e7565b6126a361275d565b6109ff838383612a0c565b6126b882826114de565b610f4a576126d0816001600160a01b03166014611bf7565b6126db836020611bf7565b6040516020016126ec9291906135e6565b60408051601f198184030181529082905262461bcd60e51b825261095e91600401612f9c565b600a5460ff1661275b5760405162461bcd60e51b815260206004820152601460248201527314185d5cd8589b194e881b9bdd081c185d5cd95960621b604482015260640161095e565b565b600a5460ff161561275b5760405162461bcd60e51b815260206004820152601060248201526f14185d5cd8589b194e881c185d5cd95960821b604482015260640161095e565b60006001600160a01b0384163b1561289957604051630a85bd0160e11b81526001600160a01b0385169063150b7a02906127e790339089908890889060040161365b565b6020604051808303816000875af1925050508015612822575060408051601f3d908101601f1916820190925261281f91810190613697565b60015b61287f573d808015612850576040519150601f19603f3d011682016040523d82523d6000602084013e612855565b606091505b5080516000036128775760405162461bcd60e51b815260040161095e90613555565b805181602001fd5b6001600160e01b031916630a85bd0160e11b149050611f03565b506001949350505050565b6128ae8383612a92565b6128bb60008484846127a3565b6109ff5760405162461bcd60e51b815260040161095e90613555565b6060600d8054610840906132e5565b60608160000361290d5750506040805180820190915260018152600360fc1b602082015290565b8160005b81156129375780612921816134dc565b91506129309050600a83613541565b9150612911565b60008167ffffffffffffffff81111561295257612952613185565b6040519080825280601f01601f19166020018201604052801561297c576020820181803683370190505b5090505b8415611f03576129916001836134a0565b915061299e600a866136b4565b6129a990603061338a565b60f81b8183815181106129be576129be6133f0565b60200101906001600160f81b031916908160001a9053506129e0600a86613541565b9450612980565b60006001600160e01b0319821663780e9d6360e01b148061082b575061082b82612be0565b612a17838383612c30565b6000818152601260205260409020546001600160a01b0316156109ff5760405162461bcd60e51b815260206004820152602d60248201527f4552433732314c656e64696e673a2043616e6e6f74207472616e73666572207460448201526c37b5b2b71037b7103637b0b71760991b606482015260840161095e565b6001600160a01b038216612ae85760405162461bcd60e51b815260206004820181905260248201527f4552433732313a206d696e7420746f20746865207a65726f2061646472657373604482015260640161095e565b6000818152600260205260409020546001600160a01b031615612b4d5760405162461bcd60e51b815260206004820152601c60248201527f4552433732313a20746f6b656e20616c7265616479206d696e74656400000000604482015260640161095e565b612b596000838361269b565b6001600160a01b0382166000908152600360205260408120805460019290612b8290849061338a565b909155505060008181526002602052604080822080546001600160a01b0319166001600160a01b03861690811790915590518392907fddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef908290a45050565b60006001600160e01b031982166380ac58cd60e01b1480612c1157506001600160e01b03198216635b5e139f60e01b145b8061082b57506301ffc9a760e01b6001600160e01b031983161461082b565b6001600160a01b038316612c8b57612c8681600880546000838152600960205260408120829055600182018355919091527ff3f7a9fe364faab93b216da50a3214154f22a0a2b415b23a84c8169e8b636ee30155565b612cae565b816001600160a01b0316836001600160a01b031614612cae57612cae8382612ce8565b6001600160a01b038216612cc5576109ff81612d85565b826001600160a01b0316826001600160a01b0316146109ff576109ff8282612e34565b60006001612cf584611438565b612cff91906134a0565b600083815260076020526040902054909150808214612d52576001600160a01b03841660009081526006602090815260408083208584528252808320548484528184208190558352600790915290208190555b5060009182526007602090815260408084208490556001600160a01b039094168352600681528383209183525290812055565b600854600090612d97906001906134a0565b60008381526009602052604081205460088054939450909284908110612dbf57612dbf6133f0565b906000526020600020015490508060088381548110612de057612de06133f0565b6000918252602080832090910192909255828152600990915260408082208490558582528120556008805480612e1857612e186136c8565b6001900381819060005260206000200160009055905550505050565b6000612e3f83611438565b6001600160a01b039093166000908152600660209081526040808320868452825280832085905593825260079052919091209190915550565b828054612e84906132e5565b90600052602060002090601f016020900481019282612ea65760008555612eec565b82601f10612ebf57805160ff1916838001178555612eec565b82800160010185558215612eec579182015b82811115612eec578251825591602001919060010190612ed1565b50612ef8929150612efc565b5090565b5b80821115612ef85760008155600101612efd565b6001600160e01b031981168114610fa157600080fd5b600060208284031215612f3957600080fd5b8135611d9381612f11565b60005b83811015612f5f578181015183820152602001612f47565b838111156118ef5750506000910152565b60008151808452612f88816020860160208601612f44565b601f01601f19169290920160200192915050565b602081526000611d936020830184612f70565b600060208284031215612fc157600080fd5b5035919050565b80356001600160a01b0381168114612fdf57600080fd5b919050565b60008060408385031215612ff757600080fd5b61300083612fc8565b946020939093013593505050565b60008060006060848603121561302357600080fd5b61302c84612fc8565b925061303a60208501612fc8565b9150604084013590509250925092565b6000806040838503121561305d57600080fd5b50508035926020909101359150565b6000806040838503121561307f57600080fd5b8235915061308f60208401612fc8565b90509250929050565b6000602082840312156130aa57600080fd5b813561ffff81168114611d9357600080fd5b6000602082840312156130ce57600080fd5b611d9382612fc8565b600080602083850312156130ea57600080fd5b823567ffffffffffffffff8082111561310257600080fd5b818501915085601f83011261311657600080fd5b81358181111561312557600080fd5b86602082850101111561313757600080fd5b60209290920196919550909350505050565b6000806040838503121561315c57600080fd5b61316583612fc8565b91506020830135801515811461317a57600080fd5b809150509250929050565b634e487b7160e01b600052604160045260246000fd5b600080600080608085870312156131b157600080fd5b6131ba85612fc8565b93506131c860208601612fc8565b925060408501359150606085013567ffffffffffffffff808211156131ec57600080fd5b818701915087601f83011261320057600080fd5b81358181111561321257613212613185565b604051601f8201601f19908116603f0116810190838211818310171561323a5761323a613185565b816040528281528a602084870101111561325357600080fd5b82602086016020830137600060208483010152809550505050505092959194509250565b6020808252825182820181905260009190848201906040850190845b818110156132af57835183529284019291840191600101613293565b50909695505050505050565b600080604083850312156132ce57600080fd5b6132d783612fc8565b915061308f60208401612fc8565b600181811c908216806132f957607f821691505b60208210810361331957634e487b7160e01b600052602260045260246000fd5b50919050565b60208082526035908201527f4552433732314c656e64696e673a204c656e64696e67206f6620746f6b656e736040820152741034b99031bab93932b73a363c903830bab9b2b21760591b606082015260800190565b634e487b7160e01b600052601160045260246000fd5b6000821982111561339d5761339d613374565b500190565b6020808252602e908201527f4552433732313a2063616c6c6572206973206e6f7420746f6b656e206f776e6560408201526d1c881b9bdc88185c1c1c9bdd995960921b606082015260800190565b634e487b7160e01b600052603260045260246000fd5b60208082526029908201527f5552494d616e616765723a20555249732068617665206265656e2066726f7a6560408201526837103337b932bb32b960b91b606082015260800190565b60208082526031908201527f4552433732314c656e64696e673a2042616c616e636520717565727920666f7260408201527020746865207a65726f206164647265737360781b606082015260800190565b6000828210156134b2576134b2613374565b500390565b600060ff821660ff84168060ff038211156134d4576134d4613374565b019392505050565b6000600182016134ee576134ee613374565b5060010190565b600081600019048311821515161561350f5761350f613374565b500290565b60008161352357613523613374565b506000190190565b634e487b7160e01b600052601260045260246000fd5b6000826135505761355061352b565b500490565b60208082526032908201527f4552433732313a207472616e7366657220746f206e6f6e20455243373231526560408201527131b2b4bb32b91034b6b83632b6b2b73a32b960711b606082015260800190565b600083516135b9818460208801612f44565b8351908301906135cd818360208801612f44565b64173539b7b760d91b9101908152600501949350505050565b7f416363657373436f6e74726f6c3a206163636f756e742000000000000000000081526000835161361e816017850160208801612f44565b7001034b99036b4b9b9b4b733903937b6329607d1b601791840191820152835161364f816028840160208801612f44565b01602801949350505050565b60006001600160a01b0380871683528086166020840152508360408301526080606083015261368d6080830184612f70565b9695505050505050565b6000602082840312156136a957600080fd5b8151611d9381612f11565b6000826136c3576136c361352b565b500690565b634e487b7160e01b600052603160045260246000fdfe372d55e37651a7c6e1940a3fb8628e4b6122a3c1a8b2b70aee13e07228604567a264697066735822122030f13e4ab7faf105aff824bf91a10d6f01e5713a1c3643a22fb13f7606bebb0964736f6c634300080d0033

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

000000000000000000000000000000000000000000000000000000000000008000000000000000000000000000000000000000000000000000000000000000c0000000000000000000000000000000000000000000000000000000000000010000000000000000000000000000000000000000000000000000000000000001a0000000000000000000000000000000000000000000000000000000000000000c416666654d6974576166666500000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000003414d570000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000006268747470733a2f2f6177732d73616d2d616666656e2d64622d7075626c69632d616666656e2d6d657461646174612e73332e65752d63656e7472616c2d312e616d617a6f6e6177732e636f6d2f616666656d697477616666653732312f6a736f6e2f000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000006c68747470733a2f2f6177732d73616d2d616666656e2d64622d7075626c69632d616666656e2d6d657461646174612e73332e65752d63656e7472616c2d312e616d617a6f6e6177732e636f6d2f616666656d697477616666653732312f636f6c6c656374696f6e2e6a736f6e0000000000000000000000000000000000000000

-----Decoded View---------------
Arg [0] : name (string): AffeMitWaffe
Arg [1] : symbol (string): AMW
Arg [2] : initialBaseURI (string): https://aws-sam-affen-db-public-affen-metadata.s3.eu-central-1.amazonaws.com/affemitwaffe721/json/
Arg [3] : initialContractURI (string): https://aws-sam-affen-db-public-affen-metadata.s3.eu-central-1.amazonaws.com/affemitwaffe721/collection.json

-----Encoded View---------------
18 Constructor Arguments found :
Arg [0] : 0000000000000000000000000000000000000000000000000000000000000080
Arg [1] : 00000000000000000000000000000000000000000000000000000000000000c0
Arg [2] : 0000000000000000000000000000000000000000000000000000000000000100
Arg [3] : 00000000000000000000000000000000000000000000000000000000000001a0
Arg [4] : 000000000000000000000000000000000000000000000000000000000000000c
Arg [5] : 416666654d697457616666650000000000000000000000000000000000000000
Arg [6] : 0000000000000000000000000000000000000000000000000000000000000003
Arg [7] : 414d570000000000000000000000000000000000000000000000000000000000
Arg [8] : 0000000000000000000000000000000000000000000000000000000000000062
Arg [9] : 68747470733a2f2f6177732d73616d2d616666656e2d64622d7075626c69632d
Arg [10] : 616666656e2d6d657461646174612e73332e65752d63656e7472616c2d312e61
Arg [11] : 6d617a6f6e6177732e636f6d2f616666656d697477616666653732312f6a736f
Arg [12] : 6e2f000000000000000000000000000000000000000000000000000000000000
Arg [13] : 000000000000000000000000000000000000000000000000000000000000006c
Arg [14] : 68747470733a2f2f6177732d73616d2d616666656e2d64622d7075626c69632d
Arg [15] : 616666656e2d6d657461646174612e73332e65752d63656e7472616c2d312e61
Arg [16] : 6d617a6f6e6177732e636f6d2f616666656d697477616666653732312f636f6c
Arg [17] : 6c656374696f6e2e6a736f6e0000000000000000000000000000000000000000


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

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