ERC-721
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
Token Contract
Balance
1 AMWLoading...
Loading
Loading...
Loading
Loading...
Loading
# | Exchange | Pair | Price | 24H Volume | % Volume |
---|
Contract Name:
AffeMitWaffe
Compiler Version
v0.8.13+commit.abaa5c0e
Optimization Enabled:
Yes with 250 runs
Other Settings:
default evmVersion
Contract Source Code (Solidity Standard Json-Input format)
// 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); } }
// 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()); } }
// 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()); } } }
// 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 {} }
// 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(); } }
// 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."); _; } }
// 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; } }
// 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); } }
// 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); } }
// 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; } }
// 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; }
// 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); } }
// 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; } }
// 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); }
// 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); } } } }
// 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); }
// 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); }
// 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); }
// 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); }
// 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; } }
// 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); }
{ "optimizer": { "enabled": true, "runs": 250 }, "outputSelection": { "*": { "*": [ "evm.bytecode", "evm.deployedBytecode", "devdoc", "userdoc", "metadata", "abi" ] } }, "libraries": {} }
Contract Security Audit
- No Contract Security Audit Submitted- Submit Audit Here
[{"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"}]
Contract Creation Code
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.