Overview
ETH Balance
0 ETH
ETH Value
$0.00More Info
Private Name Tags
ContractCreator
Transaction Hash |
Method
|
Block
|
From
|
To
|
|||||
---|---|---|---|---|---|---|---|---|---|
Latest 1 internal transaction
Parent Transaction Hash | Block | From | To | |||
---|---|---|---|---|---|---|
415575 | 30 days ago | Contract Creation | 0 ETH |
Loading...
Loading
This contract may be a proxy contract. Click on More Options and select Is this a proxy? to confirm and enable the "Read as Proxy" & "Write as Proxy" tabs.
Contract Name:
ERC721General
Compiler Version
v0.8.10+commit.fc410830
ZkSolc Version
v1.5.6
Optimization Enabled:
Yes with Mode z
Other Settings:
default evmVersion
Contract Source Code (Solidity Standard Json-Input format)
// SPDX-License-Identifier: UNLICENSED pragma solidity 0.8.10; import "./ERC721Base.sol"; import "../metadata/MetadataEncryption.sol"; import "../tokenManager/interfaces/IPostTransfer.sol"; import "../tokenManager/interfaces/IPostBurn.sol"; import "./interfaces/IERC721GeneralMint.sol"; import "./ERC721GeneralBase.sol"; /** * @title Generalized ERC721 * @author highlight.xyz * @notice Generalized NFT smart contract */ contract ERC721General is ERC721GeneralBase { using EnumerableSet for EnumerableSet.AddressSet; /** * @notice Initialize the contract * @param creator Creator/owner of contract * @param _contractURI Contract metadata * @param defaultRoyalty Default royalty object for contract (optional) * @param _defaultTokenManager Default token manager for contract (optional) * @param _name Name of token edition * @param _symbol Symbol of the token edition * @param trustedForwarder Trusted minimal forwarder * @param initialMinter Initial minter to register * @param newBaseURI Base URI for contract * @param _limitSupply Initial limit supply * @param useMarketplaceFiltererRegistry Denotes whether to use marketplace filterer registry * @param _observability Observability contract address */ function initialize( address creator, string memory _contractURI, IRoyaltyManager.Royalty memory defaultRoyalty, address _defaultTokenManager, string memory _name, string memory _symbol, address trustedForwarder, address initialMinter, string memory newBaseURI, uint256 _limitSupply, bool useMarketplaceFiltererRegistry, address _observability ) external initializer { _initialize( creator, _contractURI, defaultRoyalty, _defaultTokenManager, _name, _symbol, trustedForwarder, initialMinter, newBaseURI, _limitSupply, useMarketplaceFiltererRegistry, _observability ); } /** * @notice Initialize the contract * @param creator Creator/owner of contract * @param data Data to initialize the contract * @ param _contractURI Contract metadata * @ param defaultRoyalty Default royalty object for contract (optional) * @ param _defaultTokenManager Default token manager for contract (optional) * @ param _name Name of token edition * @ param _symbol Symbol of the token edition * @ param trustedForwarder Trusted minimal forwarder * @ param initialMinter Initial minter to register * @ param newBaseURI Base URI for contract * @ param _limitSupply Initial limit supply * @ param useMarketplaceFiltererRegistry Denotes whether to use marketplace filterer registry * @ param _observability Observability contract address */ function initialize(address creator, bytes memory data) external initializer { ( string memory _contractURI, IRoyaltyManager.Royalty memory defaultRoyalty, address _defaultTokenManager, string memory _name, string memory _symbol, address trustedForwarder, address initialMinter, string memory newBaseURI, uint256 _limitSupply, bool useMarketplaceFiltererRegistry, address _observability ) = abi.decode( data, ( string, IRoyaltyManager.Royalty, address, string, string, address, address, string, uint256, bool, address ) ); _initialize( creator, _contractURI, defaultRoyalty, _defaultTokenManager, _name, _symbol, trustedForwarder, initialMinter, newBaseURI, _limitSupply, useMarketplaceFiltererRegistry, _observability ); } /** * @notice Initialize the contract * @param creator Creator/owner of contract * @param _contractURI Contract metadata * @param defaultRoyalty Default royalty object for contract (optional) * @param _defaultTokenManager Default token manager for contract (optional) * @param _name Name of token edition * @param _symbol Symbol of the token edition * @param trustedForwarder Trusted minimal forwarder * @param initialMinter Initial minter to register * @param newBaseURI Base URI for contract * @param _limitSupply Initial limit supply * @param useMarketplaceFiltererRegistry Denotes whether to use marketplace filterer registry * @param _observability Observability contract address */ function _initialize( address creator, string memory _contractURI, IRoyaltyManager.Royalty memory defaultRoyalty, address _defaultTokenManager, string memory _name, string memory _symbol, address trustedForwarder, address initialMinter, string memory newBaseURI, uint256 _limitSupply, bool useMarketplaceFiltererRegistry, address _observability ) private { __ERC721URIStorage_init(); __ERC721Base_initialize(creator, defaultRoyalty, _defaultTokenManager); __ERC2771ContextUpgradeable__init__(trustedForwarder); __ERC721_init(_name, _symbol); // deprecate but keep input for backwards-compatibility: // __MarketplaceFilterer__init__(useMarketplaceFiltererRegistry); _minters.add(initialMinter); contractURI = _contractURI; IObservability(_observability).emitSeriesDeployed(address(this)); observability = IObservability(_observability); if (bytes(newBaseURI).length > 0) { _setBaseURI(newBaseURI); // don't emit on observability contract here } if (_limitSupply > 0) { limitSupply = _limitSupply; // don't emit on observability contract here } } }
// SPDX-License-Identifier: UNLICENSED pragma solidity 0.8.10; /** * @dev Utilities for metadata encryption and decryption * @author highlight.xyz */ abstract contract MetadataEncryption { /// @dev See: https://ethereum.stackexchange.com/questions/69825/decrypt-message-on-chain function encryptDecrypt(bytes memory data, bytes calldata key) public pure returns (bytes memory result) { // Store data length on stack for later use uint256 length = data.length; // solhint-disable-next-line no-inline-assembly assembly { // Set result to free memory pointer result := mload(0x40) // Increase free memory pointer by lenght + 32 mstore(0x40, add(add(result, length), 32)) // Set result length mstore(result, length) } // Iterate over the data stepping by 32 bytes for (uint256 i = 0; i < length; i += 32) { // Generate hash of the key and offset bytes32 hash = keccak256(abi.encodePacked(key, i)); bytes32 chunk; // solhint-disable-next-line no-inline-assembly assembly { // Read 32-bytes data chunk chunk := mload(add(data, add(i, 32))) } // XOR the chunk with hash chunk ^= hash; // solhint-disable-next-line no-inline-assembly assembly { // Write 32-byte encrypted chunk mstore(add(result, add(i, 32)), chunk) } } } function _sliceUint(bytes memory bs, uint256 start) internal pure returns (uint256) { require(bs.length >= start + 32, "slicing out of range"); uint256 x; assembly { x := mload(add(bs, add(0x20, start))) } return x; } }
// SPDX-License-Identifier: UNLICENSED pragma solidity 0.8.10; import "../royaltyManager/interfaces/IRoyaltyManager.sol"; import "../tokenManager/interfaces/ITokenManager.sol"; import "../utils/Ownable.sol"; import "../utils/ERC2981/IERC2981Upgradeable.sol"; import "../utils/ERC165/ERC165CheckerUpgradeable.sol"; import "../metatx/ERC2771ContextUpgradeable.sol"; import "../observability/IObservability.sol"; import "@openzeppelin/contracts/utils/structs/EnumerableSet.sol"; import "@openzeppelin/contracts-upgradeable/access/OwnableUpgradeable.sol"; import "@openzeppelin/contracts-upgradeable/security/ReentrancyGuardUpgradeable.sol"; import { UUPSUpgradeable } from "@openzeppelin/contracts/proxy/utils/UUPSUpgradeable.sol"; /** * @title Base ERC721 * @author highlight.xyz * @notice Core piece of Highlight NFT contracts (v2) */ abstract contract ERC721Base is OwnableUpgradeable, IERC2981Upgradeable, ERC2771ContextUpgradeable, ReentrancyGuardUpgradeable, UUPSUpgradeable { using EnumerableSet for EnumerableSet.AddressSet; using ERC165CheckerUpgradeable for address; /** * @notice Throw when token or royalty manager is invalid */ error InvalidManager(); /** * @notice Throw when token or royalty manager does not exist */ error ManagerDoesNotExist(); /** * @notice Throw when sender is unauthorized to perform action */ error Unauthorized(); /** * @notice Throw when sender is not a minter */ error NotMinter(); /** * @notice Throw when token manager or royalty manager swap is blocked */ error ManagerSwapBlocked(); /** * @notice Throw when token manager or royalty manager remove is blocked */ error ManagerRemoveBlocked(); /** * @notice Throw when setting default or granular royalty is blocked */ error RoyaltySetBlocked(); /** * @notice Throw when royalty BPS is invalid */ error RoyaltyBPSInvalid(); /** * @notice Throw when minter registration is invalid */ error MinterRegistrationInvalid(); /** * @notice Set of minters allowed to mint on contract */ EnumerableSet.AddressSet internal _minters; /** * @notice Global token/edition manager default */ address public defaultManager; /** * @notice Token/edition managers per token grouping. * Edition ID if implemented by Editions contract, and token ID if implemented by General contract. */ mapping(uint256 => address) internal _managers; /** * @notice Default royalty for entire contract */ IRoyaltyManager.Royalty internal _defaultRoyalty; /** * @notice Royalty per token grouping. * Edition ID if implemented by Editions contract, and token ID if implemented by General contract. */ mapping(uint256 => IRoyaltyManager.Royalty) internal _royalties; /** * @notice Royalty manager - optional contract that defines the conditions around setting royalties */ address public royaltyManager; /** * @notice Freezes minting on smart contract forever */ uint8 internal _mintFrozen; /** * @notice Observability contract */ IObservability public observability; /** * @notice Emitted when minter is registered or unregistered * @param minter Minter that was changed * @param registered True if the minter was registered, false if unregistered */ event MinterRegistrationChanged(address indexed minter, bool indexed registered); /** * @notice Emitted when token managers are set for token/edition ids * @param _ids Edition / token ids * @param _tokenManagers Token managers to set for tokens / editions */ event GranularTokenManagersSet(uint256[] _ids, address[] _tokenManagers); /** * @notice Emitted when token managers are removed for token/edition ids * @param _ids Edition / token ids to remove token managers for */ event GranularTokenManagersRemoved(uint256[] _ids); /** * @notice Emitted when default token manager changed * @param newDefaultTokenManager New default token manager. Zero address if old one was removed */ event DefaultTokenManagerChanged(address indexed newDefaultTokenManager); /** * @notice Emitted when default royalty is set * @param recipientAddress Royalty recipient * @param royaltyPercentageBPS Percentage of sale (in basis points) owed to royalty recipient */ event DefaultRoyaltySet(address indexed recipientAddress, uint16 indexed royaltyPercentageBPS); /** * @notice Emitted when royalties are set for edition / token ids * @param ids Token / edition ids * @param _newRoyalties New royalties for each token / edition */ event GranularRoyaltiesSet(uint256[] ids, IRoyaltyManager.Royalty[] _newRoyalties); /** * @notice Emitted when royalty manager is updated * @param newRoyaltyManager New royalty manager. Zero address if old one was removed */ event RoyaltyManagerChanged(address indexed newRoyaltyManager); /** * @notice Emitted when mints are frozen permanently */ event MintsFrozen(); /** * @notice Restricts calls to minters */ modifier onlyMinter() { if (!_minters.contains(_msgSender())) { _revert(NotMinter.selector); } _; } /** * @notice Restricts calls if input royalty bps is over 10000 */ modifier royaltyValid(uint16 _royaltyBPS) { if (!_royaltyBPSValid(_royaltyBPS)) { _revert(RoyaltyBPSInvalid.selector); } _; } /** * @notice Registers a minter * @param minter New minter */ function registerMinter(address minter) external onlyOwner { if (!_minters.add(minter)) { _revert(MinterRegistrationInvalid.selector); } emit MinterRegistrationChanged(minter, true); observability.emitMinterRegistrationChanged(minter, true); } /** * @notice Unregisters a minter * @param minter Minter to unregister */ function unregisterMinter(address minter) external onlyOwner { if (!_minters.remove(minter)) { _revert(MinterRegistrationInvalid.selector); } emit MinterRegistrationChanged(minter, false); observability.emitMinterRegistrationChanged(minter, false); } /** * @notice Sets granular token managers if current token manager(s) allow it * @param _ids Edition / token ids * @param _tokenManagers Token managers to set for tokens / editions */ function setGranularTokenManagers( uint256[] calldata _ids, address[] calldata _tokenManagers ) external nonReentrant { address msgSender = _msgSender(); address tempOwner = owner(); uint256 idsLength = _ids.length; for (uint256 i = 0; i < idsLength; i++) { if (!_isValidTokenManager(_tokenManagers[i])) { _revert(InvalidManager.selector); } address currentTokenManager = tokenManager(_ids[i]); if (currentTokenManager == address(0)) { if (msgSender != tempOwner) { _revert(Unauthorized.selector); } } else { if (!ITokenManager(currentTokenManager).canSwap(msgSender, _ids[i], _managers[i])) { _revert(ManagerSwapBlocked.selector); } } _managers[_ids[i]] = _tokenManagers[i]; } emit GranularTokenManagersSet(_ids, _tokenManagers); observability.emitGranularTokenManagersSet(_ids, _tokenManagers); } /** * @notice Remove granular token managers * @param _ids Edition / token ids to remove token managers for */ function removeGranularTokenManagers(uint256[] calldata _ids) external nonReentrant { address msgSender = _msgSender(); uint256 idsLength = _ids.length; for (uint256 i = 0; i < idsLength; i++) { address currentTokenManager = _managers[_ids[i]]; if (currentTokenManager == address(0)) { _revert(ManagerDoesNotExist.selector); } if (!ITokenManager(currentTokenManager).canRemoveItself(msgSender, _ids[i])) { _revert(ManagerRemoveBlocked.selector); } _managers[_ids[i]] = address(0); } emit GranularTokenManagersRemoved(_ids); observability.emitGranularTokenManagersRemoved(_ids); } /** * @notice Set default token manager if current token manager allows it * @param _defaultTokenManager New default token manager */ function setDefaultTokenManager(address _defaultTokenManager) external nonReentrant { if (!_isValidTokenManager(_defaultTokenManager)) { _revert(InvalidManager.selector); } address msgSender = _msgSender(); address currentTokenManager = defaultManager; if (currentTokenManager == address(0)) { if (msgSender != owner()) { _revert(Unauthorized.selector); } } else { if (!ITokenManager(currentTokenManager).canSwap(msgSender, 0, _defaultTokenManager)) { _revert(ManagerSwapBlocked.selector); } } defaultManager = _defaultTokenManager; emit DefaultTokenManagerChanged(_defaultTokenManager); observability.emitDefaultTokenManagerChanged(_defaultTokenManager); } /** * @notice Removes default token manager if current token manager allows it */ function removeDefaultTokenManager() external nonReentrant { address msgSender = _msgSender(); address currentTokenManager = defaultManager; if (currentTokenManager == address(0)) { _revert(ManagerDoesNotExist.selector); } if (!ITokenManager(currentTokenManager).canRemoveItself(msgSender, 0)) { _revert(ManagerRemoveBlocked.selector); } defaultManager = address(0); emit DefaultTokenManagerChanged(address(0)); observability.emitDefaultTokenManagerChanged(address(0)); } /** * @notice Sets default royalty if royalty manager allows it * @param _royalty New default royalty */ function setDefaultRoyalty( IRoyaltyManager.Royalty calldata _royalty ) external nonReentrant royaltyValid(_royalty.royaltyPercentageBPS) { address msgSender = _msgSender(); address _royaltyManager = royaltyManager; if (_royaltyManager == address(0)) { if (msgSender != owner()) { _revert(Unauthorized.selector); } } else { if (!IRoyaltyManager(_royaltyManager).canSetDefaultRoyalty(_royalty, msgSender)) { _revert(RoyaltySetBlocked.selector); } } _defaultRoyalty = _royalty; emit DefaultRoyaltySet(_royalty.recipientAddress, _royalty.royaltyPercentageBPS); observability.emitDefaultRoyaltySet(_royalty.recipientAddress, _royalty.royaltyPercentageBPS); } /** * @notice Sets granular royalties (per token-grouping) if royalty manager allows it * @param ids Token / edition ids * @param _newRoyalties New royalties for each token / edition */ function setGranularRoyalties( uint256[] calldata ids, IRoyaltyManager.Royalty[] calldata _newRoyalties ) external nonReentrant { address msgSender = _msgSender(); address tempOwner = owner(); address _royaltyManager = royaltyManager; uint256 idsLength = ids.length; if (_royaltyManager == address(0)) { if (msgSender != tempOwner) { _revert(Unauthorized.selector); } for (uint256 i = 0; i < idsLength; i++) { if (!_royaltyBPSValid(_newRoyalties[i].royaltyPercentageBPS)) { _revert(RoyaltyBPSInvalid.selector); } _royalties[ids[i]] = _newRoyalties[i]; } } else { for (uint256 i = 0; i < idsLength; i++) { if (!_royaltyBPSValid(_newRoyalties[i].royaltyPercentageBPS)) { _revert(RoyaltyBPSInvalid.selector); } if (!IRoyaltyManager(_royaltyManager).canSetGranularRoyalty(ids[i], _newRoyalties[i], msgSender)) { _revert(RoyaltySetBlocked.selector); } _royalties[ids[i]] = _newRoyalties[i]; } } emit GranularRoyaltiesSet(ids, _newRoyalties); observability.emitGranularRoyaltiesSet(ids, _newRoyalties); } /** * @notice Sets royalty manager if current one allows it * @param _royaltyManager New royalty manager */ function setRoyaltyManager(address _royaltyManager) external nonReentrant { if (!_isValidRoyaltyManager(_royaltyManager)) { _revert(InvalidManager.selector); } address msgSender = _msgSender(); address currentRoyaltyManager = royaltyManager; if (currentRoyaltyManager == address(0)) { if (msgSender != owner()) { _revert(Unauthorized.selector); } } else { if (!IRoyaltyManager(currentRoyaltyManager).canSwap(_royaltyManager, msgSender)) { _revert(ManagerSwapBlocked.selector); } } royaltyManager = _royaltyManager; emit RoyaltyManagerChanged(_royaltyManager); observability.emitRoyaltyManagerChanged(_royaltyManager); } /** * @notice Removes royalty manager if current one allows it */ function removeRoyaltyManager() external nonReentrant { address msgSender = _msgSender(); address currentRoyaltyManager = royaltyManager; if (currentRoyaltyManager == address(0)) { _revert(ManagerDoesNotExist.selector); } if (!IRoyaltyManager(currentRoyaltyManager).canRemoveItself(msgSender)) { _revert(ManagerRemoveBlocked.selector); } royaltyManager = address(0); emit RoyaltyManagerChanged(address(0)); observability.emitRoyaltyManagerChanged(address(0)); } /** * @notice Freeze mints on contract forever */ function freezeMints() external onlyOwner nonReentrant { _mintFrozen = 1; emit MintsFrozen(); observability.emitMintsFrozen(); } /** * @notice Return allowed minters on contract */ function minters() external view returns (address[] memory) { return _minters.values(); } /** * @notice Conforms to ERC-2981. Editions should overwrite to return royalty for entire edition * @param _tokenGroupingId Token id if on general, and edition id if on editions * @param _salePrice Sale price of token */ function royaltyInfo( uint256 _tokenGroupingId, uint256 _salePrice ) public view virtual override returns (address receiver, uint256 royaltyAmount) { IRoyaltyManager.Royalty memory royalty = _royalties[_tokenGroupingId]; if (royalty.recipientAddress == address(0)) { royalty = _defaultRoyalty; } receiver = royalty.recipientAddress; royaltyAmount = (_salePrice * uint256(royalty.royaltyPercentageBPS)) / 10000; } /** * @notice Returns the token manager for the id passed in. * @param id Token ID or Edition ID for Editions implementing contracts */ function tokenManager(uint256 id) public view returns (address manager) { manager = defaultManager; address granularManager = _managers[id]; if (granularManager != address(0)) { manager = granularManager; } } /** * @notice Initializes the contract, setting the creator as the initial owner. * @param _creator Contract creator * @param defaultRoyalty Default royalty for the contract * @param _defaultTokenManager Default token manager for the contract */ function __ERC721Base_initialize( address _creator, IRoyaltyManager.Royalty memory defaultRoyalty, address _defaultTokenManager ) internal onlyInitializing royaltyValid(defaultRoyalty.royaltyPercentageBPS) { __Ownable_init(); __ReentrancyGuard_init(); _transferOwnership(_creator); if (defaultRoyalty.recipientAddress != address(0)) { _defaultRoyalty = defaultRoyalty; } if (_defaultTokenManager != address(0)) { defaultManager = _defaultTokenManager; } } /** * @notice See {UUPSUpgradeable-_authorizeUpgrade} * @param // New implementation to upgrade to */ function _authorizeUpgrade(address) internal override { revert("Not upgradeable"); } /** * @notice Returns true if address is a valid tokenManager * @param _tokenManager Token manager being checked */ function _isValidTokenManager(address _tokenManager) internal view returns (bool) { return _tokenManager.supportsInterface(type(ITokenManager).interfaceId); } /** * @notice Returns true if address is a valid royaltyManager * @param _royaltyManager Royalty manager being checked */ function _isValidRoyaltyManager(address _royaltyManager) internal view returns (bool) { return _royaltyManager.supportsInterface(type(IRoyaltyManager).interfaceId); } /** * @notice Used for meta-transactions */ function _msgSender() internal view virtual override(ContextUpgradeable, ERC2771ContextUpgradeable) returns (address sender) { return ERC2771ContextUpgradeable._msgSender(); } /** * @dev For more efficient reverts. */ function _revert(bytes4 errorSelector) internal pure virtual { assembly { mstore(0x00, errorSelector) revert(0x00, 0x04) } } /** * @notice Used for meta-transactions */ function _msgData() internal view virtual override(ContextUpgradeable, ERC2771ContextUpgradeable) returns (bytes calldata) { return ERC2771ContextUpgradeable._msgData(); } /** * @notice Returns true if royalty bps passed in is valid (<= 10000) * @param _royaltyBPS Royalty basis points */ function _royaltyBPSValid(uint16 _royaltyBPS) private pure returns (bool) { return _royaltyBPS <= 10000; } }
// SPDX-License-Identifier: MIT pragma solidity 0.8.10; /** * @author highlight.xyz * @notice If token managers implement this, transfer actions will call * postSafeTransferFrom or postTransferFrom on the token manager. */ interface IPostTransfer { /** * @notice Hook called by community after safe transfers, if token manager of transferred token implements this * interface. * @param operator Operator transferring tokens * @param from Token(s) sender * @param to Token(s) recipient * @param id Transferred token's id * @param data Arbitrary data */ function postSafeTransferFrom(address operator, address from, address to, uint256 id, bytes memory data) external; /** * @notice Hook called by community after transfers, if token manager of transferred token implements * this interface. * @param operator Operator transferring tokens * @param from Token(s) sender * @param to Token(s) recipient * @param id Transferred token's id */ function postTransferFrom(address operator, address from, address to, uint256 id) external; }
// SPDX-License-Identifier: UNLICENSED pragma solidity 0.8.10; import "./ERC721Base.sol"; import "../metadata/MetadataEncryption.sol"; import "../tokenManager/interfaces/IPostTransfer.sol"; import "../tokenManager/interfaces/IPostBurn.sol"; import "./interfaces/IERC721GeneralMint.sol"; import "../utils/ERC721/ERC721URIStorageUpgradeable.sol"; import "./MarketplaceFilterer/MarketplaceFiltererAbridged.sol"; /** * @title Generalized Base ERC721 * @author highlight.xyz * @notice Generalized Base NFT smart contract */ abstract contract ERC721GeneralBase is ERC721Base, ERC721URIStorageUpgradeable, IERC721GeneralMint, MarketplaceFiltererAbridged { using EnumerableSet for EnumerableSet.AddressSet; /** * @notice Throw when attempting to mint, while mint is frozen */ error MintFrozen(); /** * @notice Throw when requested token is not in range within bounds of limit supply */ error TokenNotInRange(); /** * @notice Throw when new supply is over limit supply */ error OverLimitSupply(); /** * @notice Throw when array lengths are mismatched */ error MismatchedArrayLengths(); /** * @notice Throw when string is empty */ error EmptyString(); /** * @notice Contract metadata */ string public contractURI; /** * @notice Total tokens minted */ uint256 public supply; /** * @notice Limit the supply to take advantage of over-promising in summation with multiple mint vectors */ uint256 public limitSupply; /** * @notice Emitted when uris are set for tokens * @param ids IDs of tokens to set uris for * @param uris Uris to set on tokens */ event TokenURIsSet(uint256[] ids, string[] uris); /** * @notice Emitted when limit supply is set * @param newLimitSupply Limit supply to set */ event LimitSupplySet(uint256 indexed newLimitSupply); /** * @notice See {IERC721GeneralMint-mintOneToOneRecipient} */ function mintOneToOneRecipient(address recipient) external onlyMinter nonReentrant returns (uint256) { if (_mintFrozen == 1) { _revert(MintFrozen.selector); } uint256 tempSupply = supply; tempSupply++; _requireLimitSupply(tempSupply); _mint(recipient, tempSupply); supply = tempSupply; return tempSupply; } /** * @notice See {IERC721GeneralMint-mintAmountToOneRecipient} */ function mintAmountToOneRecipient(address recipient, uint256 amount) external onlyMinter nonReentrant { if (_mintFrozen == 1) { _revert(MintFrozen.selector); } uint256 tempSupply = supply; // cache for (uint256 i = 0; i < amount; i++) { tempSupply++; _mint(recipient, tempSupply); } _requireLimitSupply(tempSupply); supply = tempSupply; } /** * @notice See {IERC721GeneralMint-mintOneToMultipleRecipients} */ function mintOneToMultipleRecipients(address[] calldata recipients) external onlyMinter nonReentrant { if (_mintFrozen == 1) { _revert(MintFrozen.selector); } uint256 recipientsLength = recipients.length; uint256 tempSupply = supply; // cache for (uint256 i = 0; i < recipientsLength; i++) { tempSupply++; _mint(recipients[i], tempSupply); } _requireLimitSupply(tempSupply); supply = tempSupply; } /** * @notice See {IERC721GeneralMint-mintSameAmountToMultipleRecipients} */ function mintSameAmountToMultipleRecipients( address[] calldata recipients, uint256 amount ) external onlyMinter nonReentrant { if (_mintFrozen == 1) { _revert(MintFrozen.selector); } uint256 recipientsLength = recipients.length; uint256 tempSupply = supply; // cache for (uint256 i = 0; i < recipientsLength; i++) { for (uint256 j = 0; j < amount; j++) { tempSupply++; _mint(recipients[i], tempSupply); } } _requireLimitSupply(tempSupply); supply = tempSupply; } /** * @notice See {IERC721GeneralMint-mintSpecificTokenToOneRecipient} */ function mintSpecificTokenToOneRecipient(address recipient, uint256 tokenId) external onlyMinter nonReentrant { if (_mintFrozen == 1) { _revert(MintFrozen.selector); } uint256 tempSupply = supply; tempSupply++; uint256 _limitSupply = limitSupply; if (_limitSupply != 0) { if (tokenId > _limitSupply) { _revert(TokenNotInRange.selector); } } _mint(recipient, tokenId); supply = tempSupply; } /** * @notice See {IERC721GeneralMint-mintSpecificTokensToOneRecipient} */ function mintSpecificTokensToOneRecipient( address recipient, uint256[] calldata tokenIds ) external onlyMinter nonReentrant { if (_mintFrozen == 1) { _revert(MintFrozen.selector); } uint256 tempSupply = supply; uint256 tokenIdsLength = tokenIds.length; uint256 _limitSupply = limitSupply; if (_limitSupply == 0) { // don't check that token id is within range, since _limitSupply being 0 implies unlimited range for (uint256 i = 0; i < tokenIdsLength; i++) { _mint(recipient, tokenIds[i]); tempSupply++; } } else { // check that token id is within range for (uint256 i = 0; i < tokenIdsLength; i++) { if (tokenIds[i] > _limitSupply) { _revert(TokenNotInRange.selector); } _mint(recipient, tokenIds[i]); tempSupply++; } } } /** * @notice Override base URI system for select tokens, with custom per-token metadata * @param ids IDs of tokens to override base uri system for with custom uris * @param uris Custom uris */ function setTokenURIs(uint256[] calldata ids, string[] calldata uris) external nonReentrant { uint256 idsLength = ids.length; if (idsLength != uris.length) { _revert(MismatchedArrayLengths.selector); } for (uint256 i = 0; i < idsLength; i++) { _setTokenURI(ids[i], uris[i]); } emit TokenURIsSet(ids, uris); observability.emitTokenURIsSet(ids, uris); } /** * @notice Set base uri * @param newBaseURI New base uri to set */ function setBaseURI(string calldata newBaseURI) external nonReentrant { if (bytes(newBaseURI).length == 0) { _revert(EmptyString.selector); } address _manager = defaultManager; if (_manager == address(0)) { if (_msgSender() != owner()) { _revert(Unauthorized.selector); } } else { if (!ITokenManager(_manager).canUpdateMetadata(_msgSender(), 0, bytes(newBaseURI))) { _revert(Unauthorized.selector); } } _setBaseURI(newBaseURI); observability.emitBaseUriSet(newBaseURI); } /** * @notice Set limit supply * @param _limitSupply Limit supply to set */ function setLimitSupply(uint256 _limitSupply) external onlyOwner nonReentrant { // allow it to be 0, for post-mint limitSupply = _limitSupply; emit LimitSupplySet(_limitSupply); observability.emitLimitSupplySet(_limitSupply); } /** * @notice Set contract name * @param newName New name * @param newSymbol New symbol * @param newContractUri New contractURI */ function setContractMetadata( string calldata newName, string calldata newSymbol, string calldata newContractUri ) external onlyOwner { _setContractMetadata(newName, newSymbol); contractURI = newContractUri; observability.emitContractMetadataSet(newName, newSymbol, newContractUri); } /** * @notice Total supply of NFTs on the contract * @dev Won't handle burned (temporary) */ function totalSupply() external view returns (uint256) { return supply; } /** * @notice See {IERC721-setApprovalForAll}. * Overrides default behaviour to check MarketplaceFilterer allowed operators. */ function setApprovalForAll(address operator, bool approved) public override onlyAllowedOperatorApproval(operator) { super.setApprovalForAll(operator, approved); } /** * @notice See {IERC721-approve}. * Overrides default behaviour to check MarketplaceFilterer allowed operators. */ function approve(address operator, uint256 tokenId) public override onlyAllowedOperatorApproval(operator) { super.approve(operator, tokenId); } /** * @notice See {IERC721-burn}. Overrides default behaviour to check associated tokenManager. */ function burn(uint256 tokenId) public nonReentrant { address _manager = tokenManager(tokenId); address msgSender = _msgSender(); if (_manager != address(0) && IERC165Upgradeable(_manager).supportsInterface(type(IPostBurn).interfaceId)) { address owner = ownerOf(tokenId); IPostBurn(_manager).postBurn(msgSender, owner, tokenId); } else { // default to restricting burn to owner or operator if a valid TM isn't present if (!_isApprovedOrOwner(msgSender, tokenId)) { _revert(Unauthorized.selector); } } _burn(tokenId); observability.emitTransfer(msgSender, address(0), tokenId); } /** * @notice Overrides tokenURI to first rotate the token id * @param tokenId ID of token to get uri for */ function tokenURI(uint256 tokenId) public view virtual override returns (string memory) { return ERC721URIStorageUpgradeable.tokenURI(tokenId); } /** * @notice See {IERC165-supportsInterface}. */ function supportsInterface( bytes4 interfaceId ) public view virtual override(IERC165Upgradeable, ERC721Upgradeable) returns (bool) { return ERC721Upgradeable.supportsInterface(interfaceId); } /** * @notice Hook called after transfers * @param from Account token is being transferred from * @param to Account token is being transferred to * @param tokenId ID of token being transferred */ function _afterTokenTransfers(address from, address to, uint256 tokenId) internal override { address msgSender = _msgSender(); if (from != msgSender) { _checkFilterOperator(msgSender); } address _manager = tokenManager(tokenId); if (_manager != address(0) && IERC165Upgradeable(_manager).supportsInterface(type(IPostTransfer).interfaceId)) { IPostTransfer(_manager).postSafeTransferFrom(msgSender, from, to, tokenId, ""); } observability.emitTransfer(from, to, tokenId); } /** * @notice Used for meta-transactions */ function _msgSender() internal view virtual override(ERC721Base, ContextUpgradeable) returns (address sender) { return ERC2771ContextUpgradeable._msgSender(); } /** * @notice Used for meta-transactions */ function _msgData() internal view virtual override(ERC721Base, ContextUpgradeable) returns (bytes calldata) { return ERC2771ContextUpgradeable._msgData(); } /** * @dev For more efficient reverts. */ function _revert( bytes4 errorSelector ) internal pure virtual override(ERC721Upgradeable, ERC721Base, MarketplaceFiltererAbridged) { ERC721Upgradeable._revert(errorSelector); } /** * @notice Override base URI system for select tokens, with custom per-token metadata * @param tokenId Token to set uri for * @param _uri Uri to set on token */ function _setTokenURI(uint256 tokenId, string calldata _uri) private { address _manager = tokenManager(tokenId); address msgSender = _msgSender(); address tempOwner = owner(); if (_manager == address(0)) { if (msgSender != tempOwner) { _revert(Unauthorized.selector); } } else { if (!ITokenManager(_manager).canUpdateMetadata(msgSender, tokenId, bytes(_uri))) { _revert(Unauthorized.selector); } } _tokenURIs[tokenId] = _uri; } /** * @notice Require the new supply of tokens after mint to be less than limit supply * @param newSupply New supply */ function _requireLimitSupply(uint256 newSupply) private view { uint256 _limitSupply = limitSupply; if (_limitSupply != 0 && newSupply > _limitSupply) { _revert(OverLimitSupply.selector); } } }
// SPDX-License-Identifier: MIT pragma solidity 0.8.10; /** * @author highlight.xyz * @notice If token managers implement this, transfer actions will call * postBurn on the token manager. */ interface IPostBurn { /** * @notice Hook called by contract after burn, if token manager of burned token implements this * interface. * @param operator Operator burning tokens * @param sender Msg sender * @param id Burned token's id or id of edition of token that is burned */ function postBurn(address operator, address sender, uint256 id) external; }
// SPDX-License-Identifier: UNLICENSED pragma solidity 0.8.10; /** * @notice General721 mint interface * @author highlight.xyz */ interface IERC721GeneralMint { /** * @notice Mint one token to one recipient * @param recipient Recipient of minted NFT */ function mintOneToOneRecipient(address recipient) external returns (uint256); /** * @notice Mint an amount of tokens to one recipient * @param recipient Recipient of minted NFTs * @param amount Amount of NFTs minted */ function mintAmountToOneRecipient(address recipient, uint256 amount) external; /** * @notice Mint one token to multiple recipients. Useful for use-cases like airdrops * @param recipients Recipients of minted NFTs */ function mintOneToMultipleRecipients(address[] calldata recipients) external; /** * @notice Mint the same amount of tokens to multiple recipients * @param recipients Recipients of minted NFTs * @param amount Amount of NFTs minted to each recipient */ function mintSameAmountToMultipleRecipients(address[] calldata recipients, uint256 amount) external; /** * @notice Mint a chosen token id to a single recipient * @param recipient Recipient of chosen NFT * @param tokenId ID of NFT to mint */ function mintSpecificTokenToOneRecipient(address recipient, uint256 tokenId) external; /** * @notice Mint chosen token ids to a single recipient * @param recipient Recipient of chosen NFT * @param tokenIds IDs of NFTs to mint */ function mintSpecificTokensToOneRecipient(address recipient, uint256[] calldata tokenIds) external; }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.5.0) (metatx/ERC2771Context.sol) pragma solidity 0.8.10; import "@openzeppelin/contracts-upgradeable/proxy/utils/Initializable.sol"; /** * @dev Context variant with ERC2771 support. * Openzeppelin contract slightly modified by ishan@ highlight.xyz to be upgradeable. */ abstract contract ERC2771ContextUpgradeable is Initializable { address private _trustedForwarder; /** * @dev This empty reserved space is put in place to allow future versions to add new * variables without shifting down storage in the inheritance chain. * See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps */ uint256[49] private __gap; function isTrustedForwarder(address forwarder) public view virtual returns (bool) { return forwarder == _trustedForwarder; } function __ERC2771ContextUpgradeable__init__(address trustedForwarder) internal onlyInitializing { _trustedForwarder = trustedForwarder; } function _msgSender() internal view virtual returns (address sender) { if (isTrustedForwarder(msg.sender)) { // The assembly code is more direct than the Solidity version using `abi.decode`. /* solhint-disable no-inline-assembly */ assembly { sender := shr(96, calldataload(sub(calldatasize(), 20))) } /* solhint-enable no-inline-assembly */ } else { return msg.sender; } } function _msgData() internal view virtual returns (bytes calldata) { if (isTrustedForwarder(msg.sender)) { return msg.data[:msg.data.length - 20]; } else { return msg.data; } } }
// SPDX-License-Identifier: UNLICENSED pragma solidity 0.8.10; import "../royaltyManager/interfaces/IRoyaltyManager.sol"; /** * @title IObservability * @author highlight.xyz * @notice Interface to interact with the Highlight observability singleton * @dev Singleton to coalesce select Highlight protocol events */ interface IObservability { /************************** ERC721Base / ERC721MinimizedBase events **************************/ /** * @notice Emitted when minter is registered or unregistered * @param contractAddress Initial contract that emitted event * @param minter Minter that was changed * @param registered True if the minter was registered, false if unregistered */ event MinterRegistrationChanged(address indexed contractAddress, address indexed minter, bool indexed registered); /** * @notice Emitted when token managers are set for token/edition ids * @param contractAddress Initial contract that emitted event * @param _ids Edition / token ids * @param _tokenManagers Token managers to set for tokens / editions */ event GranularTokenManagersSet(address indexed contractAddress, uint256[] _ids, address[] _tokenManagers); /** * @notice Emitted when token managers are removed for token/edition ids * @param contractAddress Initial contract that emitted event * @param _ids Edition / token ids to remove token managers for */ event GranularTokenManagersRemoved(address indexed contractAddress, uint256[] _ids); /** * @notice Emitted when default token manager changed * @param contractAddress Initial contract that emitted event * @param newDefaultTokenManager New default token manager. Zero address if old one was removed */ event DefaultTokenManagerChanged(address indexed contractAddress, address indexed newDefaultTokenManager); /** * @notice Emitted when default royalty is set * @param contractAddress Initial contract that emitted event * @param recipientAddress Royalty recipient * @param royaltyPercentageBPS Percentage of sale (in basis points) owed to royalty recipient */ event DefaultRoyaltySet( address indexed contractAddress, address indexed recipientAddress, uint16 indexed royaltyPercentageBPS ); /** * @notice Emitted when royalties are set for edition / token ids * @param contractAddress Initial contract that emitted event * @param ids Token / edition ids * @param _newRoyalties New royalties for each token / edition */ event GranularRoyaltiesSet(address indexed contractAddress, uint256[] ids, IRoyaltyManager.Royalty[] _newRoyalties); /** * @notice Emitted when royalty manager is updated * @param contractAddress Initial contract that emitted event * @param newRoyaltyManager New royalty manager. Zero address if old one was removed */ event RoyaltyManagerChanged(address indexed contractAddress, address indexed newRoyaltyManager); /** * @notice Emitted when mints are frozen permanently * @param contractAddress Initial contract that emitted event */ event MintsFrozen(address indexed contractAddress); /** * @notice Emitted when contract metadata is set * @param contractAddress Initial contract that emitted event * @param name New name * @param symbol New symbol * @param contractURI New contract uri */ event ContractMetadataSet(address indexed contractAddress, string name, string symbol, string contractURI); /************************** ERC721General events **************************/ /** * @notice Emitted when hashed metadata config is set * @param contractAddress Initial contract that emitted event * @param hashedURIData Hashed uri data * @param hashedRotationData Hashed rotation key * @param _supply Supply of tokens to mint w/ reveal */ event HashedMetadataConfigSet( address indexed contractAddress, bytes hashedURIData, bytes hashedRotationData, uint256 indexed _supply ); /** * @notice Emitted when metadata is revealed * @param contractAddress Initial contract that emitted event * @param key Key used to decode hashed data * @param newRotationKey Actual rotation key to be used */ event Revealed(address indexed contractAddress, bytes key, uint256 newRotationKey); /************************** ERC721GeneralBase events **************************/ /** * @notice Emitted when uris are set for tokens * @param contractAddress Initial contract that emitted event * @param ids IDs of tokens to set uris for * @param uris Uris to set on tokens */ event TokenURIsSet(address indexed contractAddress, uint256[] ids, string[] uris); /** * @notice Emitted when limit supply is set * @param contractAddress Initial contract that emitted event * @param newLimitSupply Limit supply to set */ event LimitSupplySet(address indexed contractAddress, uint256 indexed newLimitSupply); /************************** ERC721StorageUri events **************************/ /** * @notice Emits when a series collection has its base uri set * @param contractAddress Contract with updated base uri * @param newBaseUri New base uri */ event BaseUriSet(address indexed contractAddress, string newBaseUri); /************************** ERC721Editions / ERC721SingleEdition events **************************/ // Not adding EditionCreated, EditionMintedToOneRecipient, EditionMintedToRecipients // EditionCreated - handled by MetadataInitialized // EditionMintedToOneRecipient / EditionMintedToRecipients - handled via mint module events /************************** Deployment events **************************/ /** * @notice Emitted when Generative Series contract is deployed * @param deployer Contract deployer * @param contractAddress Address of contract that was deployed */ event GenerativeSeriesDeployed(address indexed deployer, address indexed contractAddress); /** * @notice Emitted when Series contract is deployed * @param deployer Contract deployer * @param contractAddress Address of contract that was deployed */ event SeriesDeployed(address indexed deployer, address indexed contractAddress); /** * @notice Emitted when MultipleEditions contract is deployed * @param deployer Contract deployer * @param contractAddress Address of contract that was deployed */ event MultipleEditionsDeployed(address indexed deployer, address indexed contractAddress); /** * @notice Emitted when SingleEdition contract is deployed * @param deployer Contract deployer * @param contractAddress Address of contract that was deployed */ event SingleEditionDeployed(address indexed deployer, address indexed contractAddress); /************************** ERC721 events **************************/ /** * @notice Emitted when `tokenId` token is transferred from `from` to `to` on contractAddress * @param contractAddress NFT contract token resides on * @param from Token sender * @param to Token receiver * @param tokenId Token being sent */ event Transfer(address indexed contractAddress, address indexed from, address to, uint256 indexed tokenId); /** * @notice Emit MinterRegistrationChanged */ function emitMinterRegistrationChanged(address minter, bool registered) external; /** * @notice Emit GranularTokenManagersSet */ function emitGranularTokenManagersSet(uint256[] calldata _ids, address[] calldata _tokenManagers) external; /** * @notice Emit GranularTokenManagersRemoved */ function emitGranularTokenManagersRemoved(uint256[] calldata _ids) external; /** * @notice Emit DefaultTokenManagerChanged */ function emitDefaultTokenManagerChanged(address newDefaultTokenManager) external; /** * @notice Emit DefaultRoyaltySet */ function emitDefaultRoyaltySet(address recipientAddress, uint16 royaltyPercentageBPS) external; /** * @notice Emit GranularRoyaltiesSet */ function emitGranularRoyaltiesSet( uint256[] calldata ids, IRoyaltyManager.Royalty[] calldata _newRoyalties ) external; /** * @notice Emit RoyaltyManagerChanged */ function emitRoyaltyManagerChanged(address newRoyaltyManager) external; /** * @notice Emit MintsFrozen */ function emitMintsFrozen() external; /** * @notice Emit ContractMetadataSet */ function emitContractMetadataSet( string calldata name, string calldata symbol, string calldata contractURI ) external; /** * @notice Emit HashedMetadataConfigSet */ function emitHashedMetadataConfigSet( bytes calldata hashedURIData, bytes calldata hashedRotationData, uint256 _supply ) external; /** * @notice Emit Revealed */ function emitRevealed(bytes calldata key, uint256 newRotationKey) external; /** * @notice Emit TokenURIsSet */ function emitTokenURIsSet(uint256[] calldata ids, string[] calldata uris) external; /** * @notice Emit LimitSupplySet */ function emitLimitSupplySet(uint256 newLimitSupply) external; /** * @notice Emit BaseUriSet */ function emitBaseUriSet(string calldata newBaseUri) external; /** * @notice Emit GenerativeSeriesDeployed */ function emitGenerativeSeriesDeployed(address contractAddress) external; /** * @notice Emit SeriesDeployed */ function emitSeriesDeployed(address contractAddress) external; /** * @notice Emit MultipleEditionsDeployed */ function emitMultipleEditionsDeployed(address contractAddress) external; /** * @notice Emit SingleEditionDeployed */ function emitSingleEditionDeployed(address contractAddress) external; /** * @notice Emit Transfer */ function emitTransfer(address from, address to, uint256 tokenId) external; }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.7.0) (access/Ownable.sol) pragma solidity 0.8.10; import "@openzeppelin/contracts/utils/Context.sol"; /** * @dev Contract module which provides a basic access control mechanism, where * there is an account (an owner) that can be granted exclusive access to * specific functions. * * By default, the owner account will be the one that deploys the contract. This * can later be changed with {transferOwnership}. * * This module is used through inheritance. It will make available the modifier * `onlyOwner`, which can be applied to your functions to restrict their use to * the owner. */ /* solhint-disable */ abstract contract Ownable is Context { address private _owner; event OwnershipTransferred(address indexed previousOwner, address indexed newOwner); /** * @dev Initializes the contract setting the deployer as the initial owner. */ constructor() { _transferOwnership(_msgSender()); } /** * @dev Throws if called by any account other than the owner. */ modifier onlyOwner() { _checkOwner(); _; } /** * @dev Returns the address of the current owner. */ function owner() public view virtual returns (address) { return _owner; } /** * @dev Throws if the sender is not the owner. */ function _checkOwner() internal view virtual { require(owner() == _msgSender(), "Ownable: caller is not the owner"); } /** * @dev Leaves the contract without owner. It will not be possible to call * `onlyOwner` functions anymore. Can only be called by the current owner. * * NOTE: Renouncing ownership will leave the contract without an owner, * thereby removing any functionality that is only available to the owner. */ function renounceOwnership() public virtual onlyOwner { _transferOwnership(address(0)); } /** * @dev Transfers ownership of the contract to a new account (`newOwner`). * Can only be called by the current owner. */ function transferOwnership(address newOwner) public virtual onlyOwner { require(newOwner != address(0), "Ownable: new owner is the zero address"); _transferOwnership(newOwner); } /** * @dev Transfers ownership of the contract to a new account (`newOwner`). * Internal function without access restriction. */ function _transferOwnership(address newOwner) internal virtual { address oldOwner = _owner; _owner = newOwner; emit OwnershipTransferred(oldOwner, newOwner); } }
// SPDX-License-Identifier: UNLICENSED pragma solidity 0.8.10; /** * @title ITokenManager * @author highlight.xyz * @notice Enables interfacing with custom token managers */ interface ITokenManager { /** * @notice Returns whether metadata updater is allowed to update * @param sender Updater * @param id Token/edition who's uri is being updated * If id is 0, implementation should decide behaviour for base uri update * @param newData Token's new uri if called by general contract, and any metadata field if called by editions * @return If invocation can update metadata */ function canUpdateMetadata(address sender, uint256 id, bytes calldata newData) external returns (bool); /** * @notice Returns whether token manager can be swapped for another one by invocator * @notice Default token manager implementations should ignore id * @param sender Swapper * @param id Token grouping id (token id or edition id) * @param newTokenManager New token manager being swapped to * @return If invocation can swap token managers */ function canSwap(address sender, uint256 id, address newTokenManager) external returns (bool); /** * @notice Returns whether token manager can be removed * @notice Default token manager implementations should ignore id * @param sender Swapper * @param id Token grouping id (token id or edition id) * @return If invocation can remove token manager */ function canRemoveItself(address sender, uint256 id) external returns (bool); }
// SPDX-License-Identifier: UNLICENSED pragma solidity 0.8.10; /** * @title IRoyaltyManager * @author highlight.xyz * @notice Enables interfacing with custom royalty managers that define conditions on setting royalties for * NFT contracts */ interface IRoyaltyManager { /** * @notice Struct containing values required to adhere to ERC-2981 * @param recipientAddress Royalty recipient - can be EOA, royalty splitter contract, etc. * @param royaltyPercentageBPS Royalty cut, in basis points */ struct Royalty { address recipientAddress; uint16 royaltyPercentageBPS; } /** * @notice Defines conditions around being able to swap royalty manager for another one * @param newRoyaltyManager New royalty manager being swapped in * @param sender msg sender */ function canSwap(address newRoyaltyManager, address sender) external view returns (bool); /** * @notice Defines conditions around being able to remove current royalty manager * @param sender msg sender */ function canRemoveItself(address sender) external view returns (bool); /** * @notice Defines conditions around being able to set granular royalty (per token or per edition) * @param id Edition / token ID whose royalty is being set * @param royalty Royalty being set * @param sender msg sender */ function canSetGranularRoyalty(uint256 id, Royalty calldata royalty, address sender) external view returns (bool); /** * @notice Defines conditions around being able to set default royalty * @param royalty Royalty being set * @param sender msg sender */ function canSetDefaultRoyalty(Royalty calldata royalty, address sender) external view returns (bool); }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts v4.4.1 (utils/introspection/ERC165Checker.sol) pragma solidity 0.8.10; import "./IERC165Upgradeable.sol"; /** * @dev Library used to query support of an interface declared via {IERC165}. * * Note that these functions return the actual result of the query: they do not * `revert` if an interface is not supported. It is up to the caller to decide * what to do in these cases. */ /* solhint-disable */ library ERC165CheckerUpgradeable { // As per the EIP-165 spec, no interface should ever match 0xffffffff bytes4 private constant _INTERFACE_ID_INVALID = 0xffffffff; /** * @dev Returns true if `account` supports the {IERC165} interface, */ function supportsERC165(address account) internal view returns (bool) { // Any contract that implements ERC165 must explicitly indicate support of // InterfaceId_ERC165 and explicitly indicate non-support of InterfaceId_Invalid return supportsERC165InterfaceUnchecked(account, type(IERC165Upgradeable).interfaceId) && !supportsERC165InterfaceUnchecked(account, _INTERFACE_ID_INVALID); } /** * @dev Returns true if `account` supports the interface defined by * `interfaceId`. Support for {IERC165} itself is queried automatically. * * See {IERC165-supportsInterface}. */ function supportsInterface(address account, bytes4 interfaceId) internal view returns (bool) { // query support of both ERC165 as per the spec and support of _interfaceId return supportsERC165(account) && supportsERC165InterfaceUnchecked(account, interfaceId); } /** * @dev Returns a boolean array where each value corresponds to the * interfaces passed in and whether they're supported or not. This allows * you to batch check interfaces for a contract where your expectation * is that some interfaces may not be supported. * * See {IERC165-supportsInterface}. * * _Available since v3.4._ */ function getSupportedInterfaces( address account, bytes4[] memory interfaceIds ) internal view returns (bool[] memory) { // an array of booleans corresponding to interfaceIds and whether they're supported or not bool[] memory interfaceIdsSupported = new bool[](interfaceIds.length); // query support of ERC165 itself if (supportsERC165(account)) { // query support of each interface in interfaceIds for (uint256 i = 0; i < interfaceIds.length; i++) { interfaceIdsSupported[i] = supportsERC165InterfaceUnchecked(account, interfaceIds[i]); } } return interfaceIdsSupported; } /** * @dev Returns true if `account` supports all the interfaces defined in * `interfaceIds`. Support for {IERC165} itself is queried automatically. * * Batch-querying can lead to gas savings by skipping repeated checks for * {IERC165} support. * * See {IERC165-supportsInterface}. */ function supportsAllInterfaces(address account, bytes4[] memory interfaceIds) internal view returns (bool) { // query support of ERC165 itself if (!supportsERC165(account)) { return false; } // query support of each interface in _interfaceIds for (uint256 i = 0; i < interfaceIds.length; i++) { if (!supportsERC165InterfaceUnchecked(account, interfaceIds[i])) { return false; } } // all interfaces supported return true; } /** * @notice Query if a contract implements an interface, does not check ERC165 support * @param account The address of the contract to query for support of an interface * @param interfaceId The interface identifier, as specified in ERC-165 * @return true if the contract at account indicates support of the interface with * identifier interfaceId, false otherwise * @dev Assumes that account contains a contract that supports ERC165, otherwise * the behavior of this method is undefined. This precondition can be checked * with {supportsERC165}. * Interface identification is specified in ERC-165. */ function supportsERC165InterfaceUnchecked(address account, bytes4 interfaceId) internal view returns (bool) { // prepare call bytes memory encodedParams = abi.encodeWithSelector(IERC165Upgradeable.supportsInterface.selector, interfaceId); // perform static call bool success; uint256 returnSize; uint256 returnValue; assembly { success := staticcall(30000, account, add(encodedParams, 0x20), mload(encodedParams), 0x00, 0x20) returnSize := returndatasize() returnValue := mload(0x00) } return success && returnSize >= 0x20 && returnValue > 0; } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.6.0) (interfaces/IERC2981Upgradeable.sol) pragma solidity 0.8.10; import "../ERC165/IERC165Upgradeable.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 IERC2981Upgradeable is IERC165Upgradeable { /** * @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); }
// SPDX-License-Identifier: MIT pragma solidity 0.8.10; import { IOperatorFilterRegistry } from "./interfaces/IOperatorFilterRegistry.sol"; import "@openzeppelin/contracts-upgradeable/access/OwnableUpgradeable.sol"; /** * @title MarketplaceFiltererAbridged * @notice Abstract contract whose constructor automatically registers and subscribes to default subscription from OpenSea, if a valid registry is passed in. Modified from `OperatorFilterer` contract by ishan@ highlight.xyz. * @dev This smart contract is meant to be inherited by token contracts so they can use the following: * - `onlyAllowedOperator` modifier for `transferFrom` and `safeTransferFrom` methods. * - `onlyAllowedOperatorApproval` modifier for `approve` and `setApprovalForAll` methods. */ abstract contract MarketplaceFiltererAbridged is OwnableUpgradeable { error NotAContract(); error OperatorNotAllowed(); /** * @notice MarketplaceFilterer Registry (CORI) */ address private constant _MARKETPLACE_FILTERER_REGISTRY = address(0x000000000000AAeB6D7670E522A718067333cd4E); /** * @notice Default subscription to register collection with on CORI Marketplace filterer registry */ address private constant _DEFAULT_SUBSCRIPTION = address(0x3cc6CddA760b79bAfa08dF41ECFA224f810dCeB6); /** * @notice CORI Marketplace filterer registry. Set to address(0) when not used to avoid extra inter-contract calls. */ address public operatorFiltererRegistry; function __MarketplaceFilterer__init__(bool useFilterer) internal onlyInitializing { // If an inheriting token contract is deployed to a network without the registry deployed, the modifier // will not revert, but the contract will need to be registered with the registry once it is deployed in // order for the modifier to filter addresses. if (useFilterer) { _setRegistryAndSubscription(_MARKETPLACE_FILTERER_REGISTRY, _DEFAULT_SUBSCRIPTION); } } /** * @notice Update the address that the contract will make MarketplaceFilterer checks against. * Also register this contract with that registry. */ function setRegistryAndSubscription(address newRegistry, address subscription) external onlyOwner { _setRegistryAndSubscription(newRegistry, subscription); } modifier onlyAllowedOperatorApproval(address operator) { _checkFilterOperator(operator); _; } function _checkFilterOperator(address operator) internal view { // Check registry code length to facilitate testing in environments without a deployed registry. if (operatorFiltererRegistry != address(0)) { if (!IOperatorFilterRegistry(operatorFiltererRegistry).isOperatorAllowed(address(this), operator)) { _revert(OperatorNotAllowed.selector); } } } function _setRegistryAndSubscription(address newRegistry, address subscription) private { operatorFiltererRegistry = newRegistry; if (newRegistry != address(0)) { if (newRegistry.code.length == 0) { _revert(NotAContract.selector); } IOperatorFilterRegistry(newRegistry).registerAndSubscribe(address(this), subscription); } } /** * @dev For more efficient reverts. */ function _revert(bytes4 errorSelector) internal pure virtual { assembly { mstore(0x00, errorSelector) revert(0x00, 0x04) } } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.7.0) (token/ERC721/extensions/ERC721URIStorage.sol) pragma solidity 0.8.10; import "./ERC721Upgradeable.sol"; import "@openzeppelin/contracts-upgradeable/proxy/utils/Initializable.sol"; import "@openzeppelin/contracts-upgradeable/utils/StringsUpgradeable.sol"; /** * @title Appending URI storage utilities onto template ERC721 contract * @author highlight.xyz and OpenZeppelin * @dev ERC721 token with storage based token URI management. OpenZeppelin template edited by Highlight */ /* solhint-disable */ abstract contract ERC721URIStorageUpgradeable is Initializable, ERC721Upgradeable { /** * @notice Throw when token doesn't exist */ error TokenDoesNotExist(); function __ERC721URIStorage_init() internal onlyInitializing {} function __ERC721URIStorage_init_unchained() internal onlyInitializing {} using StringsUpgradeable for uint256; // Optional mapping for token URIs mapping(uint256 => string) internal _tokenURIs; /** * @dev Hashed rotation key data */ bytes internal _hashedRotationKeyData; /** * @dev Hashed base uri data */ bytes internal _hashedBaseURIData; /** * @dev Rotation key */ uint256 internal _rotationKey; /** * @dev Contract baseURI */ string public baseURI; /** @notice Emitted when base uri is set * @param oldBaseUri Old base uri * @param newBaseURI New base uri */ event BaseURISet(string oldBaseUri, string newBaseURI); /** * @dev Set contract baseURI */ function _setBaseURI(string memory newBaseURI) internal { emit BaseURISet(baseURI, newBaseURI); baseURI = newBaseURI; } /** * @dev See {IERC721Metadata-tokenURI}. */ function tokenURI(uint256 tokenId) public view virtual override returns (string memory) { if (!_exists(tokenId)) { _revert(TokenDoesNotExist.selector); } string memory _tokenURI = _tokenURIs[tokenId]; // If there is no token URI, return the base URI. if (bytes(_tokenURI).length == 0) { return super.tokenURI(tokenId); } return _tokenURI; } /** * @dev See {ERC721-_burn}. This override additionally checks to see if a * token-specific URI was set for the token, and if so, it deletes the token URI from * the storage mapping. */ function _burn(uint256 tokenId) internal virtual override { super._burn(tokenId); if (bytes(_tokenURIs[tokenId]).length != 0) { delete _tokenURIs[tokenId]; } } function _baseURI() internal view override returns (string memory) { return baseURI; } /** * @dev This empty reserved space is put in place to allow future versions to add new * variables without shifting down storage in the inheritance chain. * See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps */ uint256[49] private __gap; }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.9.0) (access/Ownable.sol) pragma solidity ^0.8.0; import "../utils/ContextUpgradeable.sol"; import {Initializable} from "../proxy/utils/Initializable.sol"; /** * @dev Contract module which provides a basic access control mechanism, where * there is an account (an owner) that can be granted exclusive access to * specific functions. * * By default, the owner account will be the one that deploys the contract. This * can later be changed with {transferOwnership}. * * This module is used through inheritance. It will make available the modifier * `onlyOwner`, which can be applied to your functions to restrict their use to * the owner. */ abstract contract OwnableUpgradeable is Initializable, ContextUpgradeable { address private _owner; event OwnershipTransferred(address indexed previousOwner, address indexed newOwner); /** * @dev Initializes the contract setting the deployer as the initial owner. */ function __Ownable_init() internal onlyInitializing { __Ownable_init_unchained(); } function __Ownable_init_unchained() internal onlyInitializing { _transferOwnership(_msgSender()); } /** * @dev Throws if called by any account other than the owner. */ modifier onlyOwner() { _checkOwner(); _; } /** * @dev Returns the address of the current owner. */ function owner() public view virtual returns (address) { return _owner; } /** * @dev Throws if the sender is not the owner. */ function _checkOwner() internal view virtual { require(owner() == _msgSender(), "Ownable: caller is not the owner"); } /** * @dev Leaves the contract without owner. It will not be possible to call * `onlyOwner` functions. Can only be called by the current owner. * * NOTE: Renouncing ownership will leave the contract without an owner, * thereby disabling any functionality that is only available to the owner. */ function renounceOwnership() public virtual onlyOwner { _transferOwnership(address(0)); } /** * @dev Transfers ownership of the contract to a new account (`newOwner`). * Can only be called by the current owner. */ function transferOwnership(address newOwner) public virtual onlyOwner { require(newOwner != address(0), "Ownable: new owner is the zero address"); _transferOwnership(newOwner); } /** * @dev Transfers ownership of the contract to a new account (`newOwner`). * Internal function without access restriction. */ function _transferOwnership(address newOwner) internal virtual { address oldOwner = _owner; _owner = newOwner; emit OwnershipTransferred(oldOwner, newOwner); } /** * @dev This empty reserved space is put in place to allow future versions to add new * variables without shifting down storage in the inheritance chain. * See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps */ uint256[49] private __gap; }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.9.0) (security/ReentrancyGuard.sol) pragma solidity ^0.8.0; import {Initializable} from "../proxy/utils/Initializable.sol"; /** * @dev Contract module that helps prevent reentrant calls to a function. * * Inheriting from `ReentrancyGuard` will make the {nonReentrant} modifier * available, which can be applied to functions to make sure there are no nested * (reentrant) calls to them. * * Note that because there is a single `nonReentrant` guard, functions marked as * `nonReentrant` may not call one another. This can be worked around by making * those functions `private`, and then adding `external` `nonReentrant` entry * points to them. * * TIP: If you would like to learn more about reentrancy and alternative ways * to protect against it, check out our blog post * https://blog.openzeppelin.com/reentrancy-after-istanbul/[Reentrancy After Istanbul]. */ abstract contract ReentrancyGuardUpgradeable is Initializable { // 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; function __ReentrancyGuard_init() internal onlyInitializing { __ReentrancyGuard_init_unchained(); } function __ReentrancyGuard_init_unchained() internal onlyInitializing { _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() { _nonReentrantBefore(); _; _nonReentrantAfter(); } function _nonReentrantBefore() private { // On the first call to nonReentrant, _status will be _NOT_ENTERED require(_status != _ENTERED, "ReentrancyGuard: reentrant call"); // Any calls to nonReentrant after this point will fail _status = _ENTERED; } function _nonReentrantAfter() private { // By storing the original value once again, a refund is triggered (see // https://eips.ethereum.org/EIPS/eip-2200) _status = _NOT_ENTERED; } /** * @dev Returns true if the reentrancy guard is currently set to "entered", which indicates there is a * `nonReentrant` function in the call stack. */ function _reentrancyGuardEntered() internal view returns (bool) { return _status == _ENTERED; } /** * @dev This empty reserved space is put in place to allow future versions to add new * variables without shifting down storage in the inheritance chain. * See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps */ uint256[49] private __gap; }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.9.0) (proxy/utils/UUPSUpgradeable.sol) pragma solidity ^0.8.0; import "../../interfaces/draft-IERC1822.sol"; import "../ERC1967/ERC1967Upgrade.sol"; /** * @dev An upgradeability mechanism designed for UUPS proxies. The functions included here can perform an upgrade of an * {ERC1967Proxy}, when this contract is set as the implementation behind such a proxy. * * A security mechanism ensures that an upgrade does not turn off upgradeability accidentally, although this risk is * reinstated if the upgrade retains upgradeability but removes the security mechanism, e.g. by replacing * `UUPSUpgradeable` with a custom implementation of upgrades. * * The {_authorizeUpgrade} function must be overridden to include access restriction to the upgrade mechanism. * * _Available since v4.1._ */ abstract contract UUPSUpgradeable is IERC1822Proxiable, ERC1967Upgrade { /// @custom:oz-upgrades-unsafe-allow state-variable-immutable state-variable-assignment address private immutable __self = address(this); /** * @dev Check that the execution is being performed through a delegatecall call and that the execution context is * a proxy contract with an implementation (as defined in ERC1967) pointing to self. This should only be the case * for UUPS and transparent proxies that are using the current contract as their implementation. Execution of a * function through ERC1167 minimal proxies (clones) would not normally pass this test, but is not guaranteed to * fail. */ modifier onlyProxy() { require(address(this) != __self, "Function must be called through delegatecall"); require(_getImplementation() == __self, "Function must be called through active proxy"); _; } /** * @dev Check that the execution is not being performed through a delegate call. This allows a function to be * callable on the implementing contract but not through proxies. */ modifier notDelegated() { require(address(this) == __self, "UUPSUpgradeable: must not be called through delegatecall"); _; } /** * @dev Implementation of the ERC1822 {proxiableUUID} function. This returns the storage slot used by the * implementation. It is used to validate the implementation's compatibility when performing an upgrade. * * IMPORTANT: A proxy pointing at a proxiable contract should not be considered proxiable itself, because this risks * bricking a proxy that upgrades to it, by delegating to itself until out of gas. Thus it is critical that this * function revert if invoked through a proxy. This is guaranteed by the `notDelegated` modifier. */ function proxiableUUID() external view virtual override notDelegated returns (bytes32) { return _IMPLEMENTATION_SLOT; } /** * @dev Upgrade the implementation of the proxy to `newImplementation`. * * Calls {_authorizeUpgrade}. * * Emits an {Upgraded} event. * * @custom:oz-upgrades-unsafe-allow-reachable delegatecall */ function upgradeTo(address newImplementation) public virtual onlyProxy { _authorizeUpgrade(newImplementation); _upgradeToAndCallUUPS(newImplementation, new bytes(0), false); } /** * @dev Upgrade the implementation of the proxy to `newImplementation`, and subsequently execute the function call * encoded in `data`. * * Calls {_authorizeUpgrade}. * * Emits an {Upgraded} event. * * @custom:oz-upgrades-unsafe-allow-reachable delegatecall */ function upgradeToAndCall(address newImplementation, bytes memory data) public payable virtual onlyProxy { _authorizeUpgrade(newImplementation); _upgradeToAndCallUUPS(newImplementation, data, true); } /** * @dev Function that should revert when `msg.sender` is not authorized to upgrade the contract. Called by * {upgradeTo} and {upgradeToAndCall}. * * Normally, this function will use an xref:access.adoc[access control] modifier such as {Ownable-onlyOwner}. * * ```solidity * function _authorizeUpgrade(address) internal override onlyOwner {} * ``` */ function _authorizeUpgrade(address newImplementation) internal virtual; }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.9.0) (utils/structs/EnumerableSet.sol) // This file was procedurally generated from scripts/generate/templates/EnumerableSet.js. pragma solidity ^0.8.0; /** * @dev Library for managing * https://en.wikipedia.org/wiki/Set_(abstract_data_type)[sets] of primitive * types. * * Sets have the following properties: * * - Elements are added, removed, and checked for existence in constant time * (O(1)). * - Elements are enumerated in O(n). No guarantees are made on the ordering. * * ```solidity * contract Example { * // Add the library methods * using EnumerableSet for EnumerableSet.AddressSet; * * // Declare a set state variable * EnumerableSet.AddressSet private mySet; * } * ``` * * As of v3.3.0, sets of type `bytes32` (`Bytes32Set`), `address` (`AddressSet`) * and `uint256` (`UintSet`) are supported. * * [WARNING] * ==== * Trying to delete such a structure from storage will likely result in data corruption, rendering the structure * unusable. * See https://github.com/ethereum/solidity/pull/11843[ethereum/solidity#11843] for more info. * * In order to clean an EnumerableSet, you can either remove all elements one by one or create a fresh instance using an * array of EnumerableSet. * ==== */ library EnumerableSet { // To implement this library for multiple types with as little code // repetition as possible, we write it in terms of a generic Set type with // bytes32 values. // The Set implementation uses private functions, and user-facing // implementations (such as AddressSet) are just wrappers around the // underlying Set. // This means that we can only create new EnumerableSets for types that fit // in bytes32. struct Set { // Storage of set values bytes32[] _values; // Position of the value in the `values` array, plus 1 because index 0 // means a value is not in the set. mapping(bytes32 => uint256) _indexes; } /** * @dev Add a value to a set. O(1). * * Returns true if the value was added to the set, that is if it was not * already present. */ function _add(Set storage set, bytes32 value) private returns (bool) { if (!_contains(set, value)) { set._values.push(value); // The value is stored at length-1, but we add 1 to all indexes // and use 0 as a sentinel value set._indexes[value] = set._values.length; return true; } else { return false; } } /** * @dev Removes a value from a set. O(1). * * Returns true if the value was removed from the set, that is if it was * present. */ function _remove(Set storage set, bytes32 value) private returns (bool) { // We read and store the value's index to prevent multiple reads from the same storage slot uint256 valueIndex = set._indexes[value]; if (valueIndex != 0) { // Equivalent to contains(set, value) // To delete an element from the _values array in O(1), we swap the element to delete with the last one in // the array, and then remove the last element (sometimes called as 'swap and pop'). // This modifies the order of the array, as noted in {at}. uint256 toDeleteIndex = valueIndex - 1; uint256 lastIndex = set._values.length - 1; if (lastIndex != toDeleteIndex) { bytes32 lastValue = set._values[lastIndex]; // Move the last value to the index where the value to delete is set._values[toDeleteIndex] = lastValue; // Update the index for the moved value set._indexes[lastValue] = valueIndex; // Replace lastValue's index to valueIndex } // Delete the slot where the moved value was stored set._values.pop(); // Delete the index for the deleted slot delete set._indexes[value]; return true; } else { return false; } } /** * @dev Returns true if the value is in the set. O(1). */ function _contains(Set storage set, bytes32 value) private view returns (bool) { return set._indexes[value] != 0; } /** * @dev Returns the number of values on the set. O(1). */ function _length(Set storage set) private view returns (uint256) { return set._values.length; } /** * @dev Returns the value stored at position `index` in the set. O(1). * * Note that there are no guarantees on the ordering of values inside the * array, and it may change when more values are added or removed. * * Requirements: * * - `index` must be strictly less than {length}. */ function _at(Set storage set, uint256 index) private view returns (bytes32) { return set._values[index]; } /** * @dev Return the entire set in an array * * WARNING: This operation will copy the entire storage to memory, which can be quite expensive. This is designed * to mostly be used by view accessors that are queried without any gas fees. Developers should keep in mind that * this function has an unbounded cost, and using it as part of a state-changing function may render the function * uncallable if the set grows to a point where copying to memory consumes too much gas to fit in a block. */ function _values(Set storage set) private view returns (bytes32[] memory) { return set._values; } // Bytes32Set struct Bytes32Set { Set _inner; } /** * @dev Add a value to a set. O(1). * * Returns true if the value was added to the set, that is if it was not * already present. */ function add(Bytes32Set storage set, bytes32 value) internal returns (bool) { return _add(set._inner, value); } /** * @dev Removes a value from a set. O(1). * * Returns true if the value was removed from the set, that is if it was * present. */ function remove(Bytes32Set storage set, bytes32 value) internal returns (bool) { return _remove(set._inner, value); } /** * @dev Returns true if the value is in the set. O(1). */ function contains(Bytes32Set storage set, bytes32 value) internal view returns (bool) { return _contains(set._inner, value); } /** * @dev Returns the number of values in the set. O(1). */ function length(Bytes32Set storage set) internal view returns (uint256) { return _length(set._inner); } /** * @dev Returns the value stored at position `index` in the set. O(1). * * Note that there are no guarantees on the ordering of values inside the * array, and it may change when more values are added or removed. * * Requirements: * * - `index` must be strictly less than {length}. */ function at(Bytes32Set storage set, uint256 index) internal view returns (bytes32) { return _at(set._inner, index); } /** * @dev Return the entire set in an array * * WARNING: This operation will copy the entire storage to memory, which can be quite expensive. This is designed * to mostly be used by view accessors that are queried without any gas fees. Developers should keep in mind that * this function has an unbounded cost, and using it as part of a state-changing function may render the function * uncallable if the set grows to a point where copying to memory consumes too much gas to fit in a block. */ function values(Bytes32Set storage set) internal view returns (bytes32[] memory) { bytes32[] memory store = _values(set._inner); bytes32[] memory result; /// @solidity memory-safe-assembly assembly { result := store } return result; } // AddressSet struct AddressSet { Set _inner; } /** * @dev Add a value to a set. O(1). * * Returns true if the value was added to the set, that is if it was not * already present. */ function add(AddressSet storage set, address value) internal returns (bool) { return _add(set._inner, bytes32(uint256(uint160(value)))); } /** * @dev Removes a value from a set. O(1). * * Returns true if the value was removed from the set, that is if it was * present. */ function remove(AddressSet storage set, address value) internal returns (bool) { return _remove(set._inner, bytes32(uint256(uint160(value)))); } /** * @dev Returns true if the value is in the set. O(1). */ function contains(AddressSet storage set, address value) internal view returns (bool) { return _contains(set._inner, bytes32(uint256(uint160(value)))); } /** * @dev Returns the number of values in the set. O(1). */ function length(AddressSet storage set) internal view returns (uint256) { return _length(set._inner); } /** * @dev Returns the value stored at position `index` in the set. O(1). * * Note that there are no guarantees on the ordering of values inside the * array, and it may change when more values are added or removed. * * Requirements: * * - `index` must be strictly less than {length}. */ function at(AddressSet storage set, uint256 index) internal view returns (address) { return address(uint160(uint256(_at(set._inner, index)))); } /** * @dev Return the entire set in an array * * WARNING: This operation will copy the entire storage to memory, which can be quite expensive. This is designed * to mostly be used by view accessors that are queried without any gas fees. Developers should keep in mind that * this function has an unbounded cost, and using it as part of a state-changing function may render the function * uncallable if the set grows to a point where copying to memory consumes too much gas to fit in a block. */ function values(AddressSet storage set) internal view returns (address[] memory) { bytes32[] memory store = _values(set._inner); address[] memory result; /// @solidity memory-safe-assembly assembly { result := store } return result; } // UintSet struct UintSet { Set _inner; } /** * @dev Add a value to a set. O(1). * * Returns true if the value was added to the set, that is if it was not * already present. */ function add(UintSet storage set, uint256 value) internal returns (bool) { return _add(set._inner, bytes32(value)); } /** * @dev Removes a value from a set. O(1). * * Returns true if the value was removed from the set, that is if it was * present. */ function remove(UintSet storage set, uint256 value) internal returns (bool) { return _remove(set._inner, bytes32(value)); } /** * @dev Returns true if the value is in the set. O(1). */ function contains(UintSet storage set, uint256 value) internal view returns (bool) { return _contains(set._inner, bytes32(value)); } /** * @dev Returns the number of values in the set. O(1). */ function length(UintSet storage set) internal view returns (uint256) { return _length(set._inner); } /** * @dev Returns the value stored at position `index` in the set. O(1). * * Note that there are no guarantees on the ordering of values inside the * array, and it may change when more values are added or removed. * * Requirements: * * - `index` must be strictly less than {length}. */ function at(UintSet storage set, uint256 index) internal view returns (uint256) { return uint256(_at(set._inner, index)); } /** * @dev Return the entire set in an array * * WARNING: This operation will copy the entire storage to memory, which can be quite expensive. This is designed * to mostly be used by view accessors that are queried without any gas fees. Developers should keep in mind that * this function has an unbounded cost, and using it as part of a state-changing function may render the function * uncallable if the set grows to a point where copying to memory consumes too much gas to fit in a block. */ function values(UintSet storage set) internal view returns (uint256[] memory) { bytes32[] memory store = _values(set._inner); uint256[] memory result; /// @solidity memory-safe-assembly assembly { result := store } return result; } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts v4.4.1 (utils/introspection/IERC165.sol) pragma solidity 0.8.10; /** * @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}. */ /* solhint-disable */ interface IERC165Upgradeable { /** * @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) (token/ERC721/ERC721.sol) pragma solidity 0.8.10; import "./IERC721Upgradeable.sol"; import "./IERC721ReceiverUpgradeable.sol"; import "./IERC721MetadataUpgradeable.sol"; import "@openzeppelin/contracts-upgradeable/utils/AddressUpgradeable.sol"; import "@openzeppelin/contracts-upgradeable/utils/ContextUpgradeable.sol"; import "@openzeppelin/contracts-upgradeable/utils/StringsUpgradeable.sol"; import "../ERC165/ERC165Upgradeable.sol"; import "@openzeppelin/contracts-upgradeable/proxy/utils/Initializable.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}. */ /* solhint-disable */ contract ERC721Upgradeable is Initializable, ContextUpgradeable, ERC165Upgradeable, IERC721Upgradeable, IERC721MetadataUpgradeable { using AddressUpgradeable for address; using StringsUpgradeable 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. */ function __ERC721_init(string memory name_, string memory symbol_) internal onlyInitializing { __ERC721_init_unchained(name_, symbol_); } function __ERC721_init_unchained(string memory name_, string memory symbol_) internal onlyInitializing { _name = name_; _symbol = symbol_; } /** * @dev See {IERC165-supportsInterface}. */ function supportsInterface( bytes4 interfaceId ) public view virtual override(ERC165Upgradeable, IERC165Upgradeable) returns (bool) { return interfaceId == type(IERC721Upgradeable).interfaceId || interfaceId == type(IERC721MetadataUpgradeable).interfaceId || super.supportsInterface(interfaceId); } /** * @dev See {IERC721-balanceOf}. */ function balanceOf(address owner) public view virtual override returns (uint256) { require(owner != address(0), "ERC721: Invalid owner"); return _balances[owner]; } /** * @dev See {IERC721-ownerOf}. */ function ownerOf(uint256 tokenId) public view virtual override returns (address) { address owner = _owners[tokenId]; 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 = ERC721Upgradeable.ownerOf(tokenId); require(to != owner, "ERC721: already owns"); require( _msgSender() == owner || isApprovedForAll(owner, _msgSender()), "ERC721: approve caller is not token owner or 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 unauthorized"); _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 unauthorized"); _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: invalid receiver"); } /** * @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 = ERC721Upgradeable.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: invalid receiver"); } /** * @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: zero address"); require(!_exists(tokenId), "ERC721: token minted"); _balances[to] += 1; _owners[tokenId] = to; emit Transfer(address(0), to, tokenId); } /** * @dev Destroys `tokenId`. * The approval is cleared when the token is burned. * This is an internal function that does not check if the sender is authorized to operate on the token. * * Requirements: * * - `tokenId` must exist. * * Emits a {Transfer} event. */ function _burn(uint256 tokenId) internal virtual { address owner = ERC721Upgradeable.ownerOf(tokenId); // Clear approvals delete _tokenApprovals[tokenId]; _balances[owner] -= 1; delete _owners[tokenId]; emit Transfer(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(ERC721Upgradeable.ownerOf(tokenId) == from, "ERC721: from not owner"); require(to != address(0), "ERC721: to the zero address"); // Clear approvals from the previous owner delete _tokenApprovals[tokenId]; _balances[from] -= 1; _balances[to] += 1; _owners[tokenId] = to; emit Transfer(from, to, tokenId); _afterTokenTransfers(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(ERC721Upgradeable.ownerOf(tokenId), to, tokenId); } /** * @dev Hook that is called after a set of serially-ordered token IDs * have been transferred. * * `startTokenId` - the first token ID to be transferred. * * Calling conditions: * * - `from` and `to` are both non-zero, since this is only invoked ont transfers */ function _afterTokenTransfers(address from, address to, uint256 tokenId) internal virtual {} /** * @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 Added by Highlight to facilitate updating of name and symbol */ function _setContractMetadata(string calldata newName, string calldata newSymbol) internal { _name = newName; _symbol = newSymbol; } /** * @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 IERC721ReceiverUpgradeable(to).onERC721Received(_msgSender(), from, tokenId, data) returns ( bytes4 retval ) { return retval == IERC721ReceiverUpgradeable.onERC721Received.selector; } catch (bytes memory reason) { if (reason.length == 0) { revert("ERC721: invalid receiver"); } else { /// @solidity memory-safe-assembly assembly { revert(add(32, reason), mload(reason)) } } } } else { return true; } } /** * @dev For more efficient reverts. */ function _revert(bytes4 errorSelector) internal pure virtual { assembly { mstore(0x00, errorSelector) revert(0x00, 0x04) } } /** * @dev This empty reserved space is put in place to allow future versions to add new * variables without shifting down storage in the inheritance chain. * See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps */ uint256[44] private __gap; }
// SPDX-License-Identifier: MIT pragma solidity 0.8.10; interface IOperatorFilterRegistry { function isOperatorAllowed(address registrant, address operator) external view returns (bool); function register(address registrant) external; function registerAndSubscribe(address registrant, address subscription) external; function registerAndCopyEntries(address registrant, address registrantToCopy) external; function unregister(address addr) external; function updateOperator(address registrant, address operator, bool filtered) external; function updateOperators(address registrant, address[] calldata operators, bool filtered) external; function updateCodeHash(address registrant, bytes32 codehash, bool filtered) external; function updateCodeHashes(address registrant, bytes32[] calldata codeHashes, bool filtered) external; function subscribe(address registrant, address registrantToSubscribe) external; function unsubscribe(address registrant, bool copyExistingEntries) external; function subscriptionOf(address addr) external returns (address registrant); function subscribers(address registrant) external returns (address[] memory); function subscriberAt(address registrant, uint256 index) external returns (address); function copyEntriesOf(address registrant, address registrantToCopy) external; function isOperatorFiltered(address registrant, address operator) external returns (bool); function isCodeHashOfFiltered(address registrant, address operatorWithCode) external returns (bool); function isCodeHashFiltered(address registrant, bytes32 codeHash) external returns (bool); function filteredOperators(address addr) external returns (address[] memory); function filteredCodeHashes(address addr) external returns (bytes32[] memory); function filteredOperatorAt(address registrant, uint256 index) external returns (address); function filteredCodeHashAt(address registrant, uint256 index) external returns (bytes32); function isRegistered(address addr) external returns (bool); function codeHashOf(address addr) external returns (bytes32); }
// 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 (last updated v4.9.0) (proxy/utils/Initializable.sol) pragma solidity ^0.8.2; import "../../utils/AddressUpgradeable.sol"; /** * @dev This is a base contract to aid in writing upgradeable contracts, or any kind of contract that will be deployed * behind a proxy. Since proxied contracts do not make use of a constructor, it's common to move constructor logic to an * external initializer function, usually called `initialize`. It then becomes necessary to protect this initializer * function so it can only be called once. The {initializer} modifier provided by this contract will have this effect. * * The initialization functions use a version number. Once a version number is used, it is consumed and cannot be * reused. This mechanism prevents re-execution of each "step" but allows the creation of new initialization steps in * case an upgrade adds a module that needs to be initialized. * * For example: * * [.hljs-theme-light.nopadding] * ```solidity * contract MyToken is ERC20Upgradeable { * function initialize() initializer public { * __ERC20_init("MyToken", "MTK"); * } * } * * contract MyTokenV2 is MyToken, ERC20PermitUpgradeable { * function initializeV2() reinitializer(2) public { * __ERC20Permit_init("MyToken"); * } * } * ``` * * TIP: To avoid leaving the proxy in an uninitialized state, the initializer function should be called as early as * possible by providing the encoded function call as the `_data` argument to {ERC1967Proxy-constructor}. * * CAUTION: When used with inheritance, manual care must be taken to not invoke a parent initializer twice, or to ensure * that all initializers are idempotent. This is not verified automatically as constructors are by Solidity. * * [CAUTION] * ==== * Avoid leaving a contract uninitialized. * * An uninitialized contract can be taken over by an attacker. This applies to both a proxy and its implementation * contract, which may impact the proxy. To prevent the implementation contract from being used, you should invoke * the {_disableInitializers} function in the constructor to automatically lock it when it is deployed: * * [.hljs-theme-light.nopadding] * ``` * /// @custom:oz-upgrades-unsafe-allow constructor * constructor() { * _disableInitializers(); * } * ``` * ==== */ abstract contract Initializable { /** * @dev Indicates that the contract has been initialized. * @custom:oz-retyped-from bool */ uint8 private _initialized; /** * @dev Indicates that the contract is in the process of being initialized. */ bool private _initializing; /** * @dev Triggered when the contract has been initialized or reinitialized. */ event Initialized(uint8 version); /** * @dev A modifier that defines a protected initializer function that can be invoked at most once. In its scope, * `onlyInitializing` functions can be used to initialize parent contracts. * * Similar to `reinitializer(1)`, except that functions marked with `initializer` can be nested in the context of a * constructor. * * Emits an {Initialized} event. */ modifier initializer() { bool isTopLevelCall = !_initializing; require( (isTopLevelCall && _initialized < 1) || (!AddressUpgradeable.isContract(address(this)) && _initialized == 1), "Initializable: contract is already initialized" ); _initialized = 1; if (isTopLevelCall) { _initializing = true; } _; if (isTopLevelCall) { _initializing = false; emit Initialized(1); } } /** * @dev A modifier that defines a protected reinitializer function that can be invoked at most once, and only if the * contract hasn't been initialized to a greater version before. In its scope, `onlyInitializing` functions can be * used to initialize parent contracts. * * A reinitializer may be used after the original initialization step. This is essential to configure modules that * are added through upgrades and that require initialization. * * When `version` is 1, this modifier is similar to `initializer`, except that functions marked with `reinitializer` * cannot be nested. If one is invoked in the context of another, execution will revert. * * Note that versions can jump in increments greater than 1; this implies that if multiple reinitializers coexist in * a contract, executing them in the right order is up to the developer or operator. * * WARNING: setting the version to 255 will prevent any future reinitialization. * * Emits an {Initialized} event. */ modifier reinitializer(uint8 version) { require(!_initializing && _initialized < version, "Initializable: contract is already initialized"); _initialized = version; _initializing = true; _; _initializing = false; emit Initialized(version); } /** * @dev Modifier to protect an initialization function so that it can only be invoked by functions with the * {initializer} and {reinitializer} modifiers, directly or indirectly. */ modifier onlyInitializing() { require(_initializing, "Initializable: contract is not initializing"); _; } /** * @dev Locks the contract, preventing any future reinitialization. This cannot be part of an initializer call. * Calling this in the constructor of a contract will prevent that contract from being initialized or reinitialized * to any version. It is recommended to use this to lock implementation contracts that are designed to be called * through proxies. * * Emits an {Initialized} event the first time it is successfully executed. */ function _disableInitializers() internal virtual { require(!_initializing, "Initializable: contract is initializing"); if (_initialized != type(uint8).max) { _initialized = type(uint8).max; emit Initialized(type(uint8).max); } } /** * @dev Returns the highest version that has been initialized. See {reinitializer}. */ function _getInitializedVersion() internal view returns (uint8) { return _initialized; } /** * @dev Returns `true` if the contract is currently initializing. See {onlyInitializing}. */ function _isInitializing() internal view returns (bool) { return _initializing; } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.9.0) (utils/Strings.sol) pragma solidity ^0.8.0; import "./math/MathUpgradeable.sol"; import "./math/SignedMathUpgradeable.sol"; /** * @dev String operations. */ library StringsUpgradeable { bytes16 private constant _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) { unchecked { uint256 length = MathUpgradeable.log10(value) + 1; string memory buffer = new string(length); uint256 ptr; /// @solidity memory-safe-assembly assembly { ptr := add(buffer, add(32, length)) } while (true) { ptr--; /// @solidity memory-safe-assembly assembly { mstore8(ptr, byte(mod(value, 10), _SYMBOLS)) } value /= 10; if (value == 0) break; } return buffer; } } /** * @dev Converts a `int256` to its ASCII `string` decimal representation. */ function toString(int256 value) internal pure returns (string memory) { return string(abi.encodePacked(value < 0 ? "-" : "", toString(SignedMathUpgradeable.abs(value)))); } /** * @dev Converts a `uint256` to its ASCII `string` hexadecimal representation. */ function toHexString(uint256 value) internal pure returns (string memory) { unchecked { return toHexString(value, MathUpgradeable.log256(value) + 1); } } /** * @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] = _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); } /** * @dev Returns true if the two strings are equal. */ function equal(string memory a, string memory b) internal pure returns (bool) { return keccak256(bytes(a)) == keccak256(bytes(b)); } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.7.0) (token/ERC721/IERC721.sol) pragma solidity 0.8.10; import "../ERC165/IERC165Upgradeable.sol"; /** * @dev Required interface of an ERC721 compliant contract. */ /* solhint-disable */ interface IERC721Upgradeable is IERC165Upgradeable { /** * @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.10; /** * @title ERC721 token receiver interface * @dev Interface for any contract that wants to support safeTransfers * from ERC721 asset contracts. */ /* solhint-disable */ interface IERC721ReceiverUpgradeable { /** * @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.10; import "./IERC721Upgradeable.sol"; /** * @title ERC-721 Non-Fungible Token Standard, optional metadata extension * @dev See https://eips.ethereum.org/EIPS/eip-721 */ /* solhint-disable */ interface IERC721MetadataUpgradeable is IERC721Upgradeable { /** * @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 v4.4.1 (utils/introspection/ERC165.sol) pragma solidity 0.8.10; import "./IERC165Upgradeable.sol"; import "@openzeppelin/contracts-upgradeable/proxy/utils/Initializable.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. */ /* solhint-disable */ abstract contract ERC165Upgradeable is Initializable, IERC165Upgradeable { function __ERC165_init() internal onlyInitializing {} function __ERC165_init_unchained() internal onlyInitializing {} /** * @dev See {IERC165-supportsInterface}. */ function supportsInterface(bytes4 interfaceId) public view virtual override returns (bool) { return interfaceId == type(IERC165Upgradeable).interfaceId; } /** * @dev This empty reserved space is put in place to allow future versions to add new * variables without shifting down storage in the inheritance chain. * See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps */ uint256[50] private __gap; }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.9.4) (utils/Context.sol) pragma solidity ^0.8.0; import {Initializable} from "../proxy/utils/Initializable.sol"; /** * @dev Provides information about the current execution context, including the * sender of the transaction and its data. While these are generally available * via msg.sender and msg.data, they should not be accessed in such a direct * manner, since when dealing with 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 ContextUpgradeable is Initializable { function __Context_init() internal onlyInitializing { } function __Context_init_unchained() internal onlyInitializing { } function _msgSender() internal view virtual returns (address) { return msg.sender; } function _msgData() internal view virtual returns (bytes calldata) { return msg.data; } function _contextSuffixLength() internal view virtual returns (uint256) { return 0; } /** * @dev This empty reserved space is put in place to allow future versions to add new * variables without shifting down storage in the inheritance chain. * See https://docs.openzeppelin.com/contracts/4.x/upgradeable#storage_gaps */ uint256[50] private __gap; }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.9.0) (utils/Address.sol) pragma solidity ^0.8.1; /** * @dev Collection of functions related to the address type */ library AddressUpgradeable { /** * @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 * * Furthermore, `isContract` will also return true if the target contract within * the same transaction is already scheduled for destruction by `SELFDESTRUCT`, * which only has an effect at the end of a transaction. * ==== * * [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://consensys.net/diligence/blog/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.8.0/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 functionCallWithValue(target, data, 0, "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"); (bool success, bytes memory returndata) = target.call{value: value}(data); return verifyCallResultFromTarget(target, 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) { (bool success, bytes memory returndata) = target.staticcall(data); return verifyCallResultFromTarget(target, 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) { (bool success, bytes memory returndata) = target.delegatecall(data); return verifyCallResultFromTarget(target, success, returndata, errorMessage); } /** * @dev Tool to verify that a low level call to smart-contract was successful, and revert (either by bubbling * the revert reason or using the provided one) in case of unsuccessful call or if target was not a contract. * * _Available since v4.8._ */ function verifyCallResultFromTarget( address target, bool success, bytes memory returndata, string memory errorMessage ) internal view returns (bytes memory) { if (success) { if (returndata.length == 0) { // only check isContract if the call was successful and the return data is empty // otherwise we already know that it was a contract require(isContract(target), "Address: call to non-contract"); } return returndata; } else { _revert(returndata, errorMessage); } } /** * @dev Tool to verify that a low level call was successful, and revert if it wasn't, either by bubbling the * revert reason or 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 { _revert(returndata, errorMessage); } } function _revert(bytes memory returndata, string memory errorMessage) private pure { // 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.5.0) (interfaces/draft-IERC1822.sol) pragma solidity ^0.8.0; /** * @dev ERC1822: Universal Upgradeable Proxy Standard (UUPS) documents a method for upgradeability through a simplified * proxy whose upgrades are fully controlled by the current implementation. */ interface IERC1822Proxiable { /** * @dev Returns the storage slot that the proxiable contract assumes is being used to store the implementation * address. * * IMPORTANT: A proxy pointing at a proxiable contract should not be considered proxiable itself, because this risks * bricking a proxy that upgrades to it, by delegating to itself until out of gas. Thus it is critical that this * function revert if invoked through a proxy. */ function proxiableUUID() external view returns (bytes32); }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.9.0) (proxy/ERC1967/ERC1967Upgrade.sol) pragma solidity ^0.8.2; import "../beacon/IBeacon.sol"; import "../../interfaces/IERC1967.sol"; import "../../interfaces/draft-IERC1822.sol"; import "../../utils/Address.sol"; import "../../utils/StorageSlot.sol"; /** * @dev This abstract contract provides getters and event emitting update functions for * https://eips.ethereum.org/EIPS/eip-1967[EIP1967] slots. * * _Available since v4.1._ */ abstract contract ERC1967Upgrade is IERC1967 { // This is the keccak-256 hash of "eip1967.proxy.rollback" subtracted by 1 bytes32 private constant _ROLLBACK_SLOT = 0x4910fdfa16fed3260ed0e7147f7cc6da11a60208b5b9406d12a635614ffd9143; /** * @dev Storage slot with the address of the current implementation. * This is the keccak-256 hash of "eip1967.proxy.implementation" subtracted by 1, and is * validated in the constructor. */ bytes32 internal constant _IMPLEMENTATION_SLOT = 0x360894a13ba1a3210667c828492db98dca3e2076cc3735a920a3ca505d382bbc; /** * @dev Returns the current implementation address. */ function _getImplementation() internal view returns (address) { return StorageSlot.getAddressSlot(_IMPLEMENTATION_SLOT).value; } /** * @dev Stores a new address in the EIP1967 implementation slot. */ function _setImplementation(address newImplementation) private { require(Address.isContract(newImplementation), "ERC1967: new implementation is not a contract"); StorageSlot.getAddressSlot(_IMPLEMENTATION_SLOT).value = newImplementation; } /** * @dev Perform implementation upgrade * * Emits an {Upgraded} event. */ function _upgradeTo(address newImplementation) internal { _setImplementation(newImplementation); emit Upgraded(newImplementation); } /** * @dev Perform implementation upgrade with additional setup call. * * Emits an {Upgraded} event. */ function _upgradeToAndCall(address newImplementation, bytes memory data, bool forceCall) internal { _upgradeTo(newImplementation); if (data.length > 0 || forceCall) { Address.functionDelegateCall(newImplementation, data); } } /** * @dev Perform implementation upgrade with security checks for UUPS proxies, and additional setup call. * * Emits an {Upgraded} event. */ function _upgradeToAndCallUUPS(address newImplementation, bytes memory data, bool forceCall) internal { // Upgrades from old implementations will perform a rollback test. This test requires the new // implementation to upgrade back to the old, non-ERC1822 compliant, implementation. Removing // this special case will break upgrade paths from old UUPS implementation to new ones. if (StorageSlot.getBooleanSlot(_ROLLBACK_SLOT).value) { _setImplementation(newImplementation); } else { try IERC1822Proxiable(newImplementation).proxiableUUID() returns (bytes32 slot) { require(slot == _IMPLEMENTATION_SLOT, "ERC1967Upgrade: unsupported proxiableUUID"); } catch { revert("ERC1967Upgrade: new implementation is not UUPS"); } _upgradeToAndCall(newImplementation, data, forceCall); } } /** * @dev Storage slot with the admin of the contract. * This is the keccak-256 hash of "eip1967.proxy.admin" subtracted by 1, and is * validated in the constructor. */ bytes32 internal constant _ADMIN_SLOT = 0xb53127684a568b3173ae13b9f8a6016e243e63b6e8ee1178d6a717850b5d6103; /** * @dev Returns the current admin. */ function _getAdmin() internal view returns (address) { return StorageSlot.getAddressSlot(_ADMIN_SLOT).value; } /** * @dev Stores a new address in the EIP1967 admin slot. */ function _setAdmin(address newAdmin) private { require(newAdmin != address(0), "ERC1967: new admin is the zero address"); StorageSlot.getAddressSlot(_ADMIN_SLOT).value = newAdmin; } /** * @dev Changes the admin of the proxy. * * Emits an {AdminChanged} event. */ function _changeAdmin(address newAdmin) internal { emit AdminChanged(_getAdmin(), newAdmin); _setAdmin(newAdmin); } /** * @dev The storage slot of the UpgradeableBeacon contract which defines the implementation for this proxy. * This is bytes32(uint256(keccak256('eip1967.proxy.beacon')) - 1)) and is validated in the constructor. */ bytes32 internal constant _BEACON_SLOT = 0xa3f0ad74e5423aebfd80d3ef4346578335a9a72aeaee59ff6cb3582b35133d50; /** * @dev Returns the current beacon. */ function _getBeacon() internal view returns (address) { return StorageSlot.getAddressSlot(_BEACON_SLOT).value; } /** * @dev Stores a new beacon in the EIP1967 beacon slot. */ function _setBeacon(address newBeacon) private { require(Address.isContract(newBeacon), "ERC1967: new beacon is not a contract"); require( Address.isContract(IBeacon(newBeacon).implementation()), "ERC1967: beacon implementation is not a contract" ); StorageSlot.getAddressSlot(_BEACON_SLOT).value = newBeacon; } /** * @dev Perform beacon upgrade with additional setup call. Note: This upgrades the address of the beacon, it does * not upgrade the implementation contained in the beacon (see {UpgradeableBeacon-_setImplementation} for that). * * Emits a {BeaconUpgraded} event. */ function _upgradeBeaconToAndCall(address newBeacon, bytes memory data, bool forceCall) internal { _setBeacon(newBeacon); emit BeaconUpgraded(newBeacon); if (data.length > 0 || forceCall) { Address.functionDelegateCall(IBeacon(newBeacon).implementation(), data); } } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.9.0) (utils/math/Math.sol) pragma solidity ^0.8.0; /** * @dev Standard math utilities missing in the Solidity language. */ library MathUpgradeable { enum Rounding { Down, // Toward negative infinity Up, // Toward infinity Zero // Toward zero } /** * @dev Returns the largest of two numbers. */ function max(uint256 a, uint256 b) internal pure returns (uint256) { return a > b ? a : b; } /** * @dev Returns the smallest of two numbers. */ function min(uint256 a, uint256 b) internal pure returns (uint256) { return a < b ? a : b; } /** * @dev Returns the average of two numbers. The result is rounded towards * zero. */ function average(uint256 a, uint256 b) internal pure returns (uint256) { // (a + b) / 2 can overflow. return (a & b) + (a ^ b) / 2; } /** * @dev Returns the ceiling of the division of two numbers. * * This differs from standard division with `/` in that it rounds up instead * of rounding down. */ function ceilDiv(uint256 a, uint256 b) internal pure returns (uint256) { // (a + b - 1) / b can overflow on addition, so we distribute. return a == 0 ? 0 : (a - 1) / b + 1; } /** * @notice Calculates floor(x * y / denominator) with full precision. Throws if result overflows a uint256 or denominator == 0 * @dev Original credit to Remco Bloemen under MIT license (https://xn--2-umb.com/21/muldiv) * with further edits by Uniswap Labs also under MIT license. */ function mulDiv(uint256 x, uint256 y, uint256 denominator) internal pure returns (uint256 result) { unchecked { // 512-bit multiply [prod1 prod0] = x * y. Compute the product mod 2^256 and mod 2^256 - 1, then use // use the Chinese Remainder Theorem to reconstruct the 512 bit result. The result is stored in two 256 // variables such that product = prod1 * 2^256 + prod0. uint256 prod0; // Least significant 256 bits of the product uint256 prod1; // Most significant 256 bits of the product assembly { let mm := mulmod(x, y, not(0)) prod0 := mul(x, y) prod1 := sub(sub(mm, prod0), lt(mm, prod0)) } // Handle non-overflow cases, 256 by 256 division. if (prod1 == 0) { // Solidity will revert if denominator == 0, unlike the div opcode on its own. // The surrounding unchecked block does not change this fact. // See https://docs.soliditylang.org/en/latest/control-structures.html#checked-or-unchecked-arithmetic. return prod0 / denominator; } // Make sure the result is less than 2^256. Also prevents denominator == 0. require(denominator > prod1, "Math: mulDiv overflow"); /////////////////////////////////////////////// // 512 by 256 division. /////////////////////////////////////////////// // Make division exact by subtracting the remainder from [prod1 prod0]. uint256 remainder; assembly { // Compute remainder using mulmod. remainder := mulmod(x, y, denominator) // Subtract 256 bit number from 512 bit number. prod1 := sub(prod1, gt(remainder, prod0)) prod0 := sub(prod0, remainder) } // Factor powers of two out of denominator and compute largest power of two divisor of denominator. Always >= 1. // See https://cs.stackexchange.com/q/138556/92363. // Does not overflow because the denominator cannot be zero at this stage in the function. uint256 twos = denominator & (~denominator + 1); assembly { // Divide denominator by twos. denominator := div(denominator, twos) // Divide [prod1 prod0] by twos. prod0 := div(prod0, twos) // Flip twos such that it is 2^256 / twos. If twos is zero, then it becomes one. twos := add(div(sub(0, twos), twos), 1) } // Shift in bits from prod1 into prod0. prod0 |= prod1 * twos; // Invert denominator mod 2^256. Now that denominator is an odd number, it has an inverse modulo 2^256 such // that denominator * inv = 1 mod 2^256. Compute the inverse by starting with a seed that is correct for // four bits. That is, denominator * inv = 1 mod 2^4. uint256 inverse = (3 * denominator) ^ 2; // Use the Newton-Raphson iteration to improve the precision. Thanks to Hensel's lifting lemma, this also works // in modular arithmetic, doubling the correct bits in each step. inverse *= 2 - denominator * inverse; // inverse mod 2^8 inverse *= 2 - denominator * inverse; // inverse mod 2^16 inverse *= 2 - denominator * inverse; // inverse mod 2^32 inverse *= 2 - denominator * inverse; // inverse mod 2^64 inverse *= 2 - denominator * inverse; // inverse mod 2^128 inverse *= 2 - denominator * inverse; // inverse mod 2^256 // Because the division is now exact we can divide by multiplying with the modular inverse of denominator. // This will give us the correct result modulo 2^256. Since the preconditions guarantee that the outcome is // less than 2^256, this is the final result. We don't need to compute the high bits of the result and prod1 // is no longer required. result = prod0 * inverse; return result; } } /** * @notice Calculates x * y / denominator with full precision, following the selected rounding direction. */ function mulDiv(uint256 x, uint256 y, uint256 denominator, Rounding rounding) internal pure returns (uint256) { uint256 result = mulDiv(x, y, denominator); if (rounding == Rounding.Up && mulmod(x, y, denominator) > 0) { result += 1; } return result; } /** * @dev Returns the square root of a number. If the number is not a perfect square, the value is rounded down. * * Inspired by Henry S. Warren, Jr.'s "Hacker's Delight" (Chapter 11). */ function sqrt(uint256 a) internal pure returns (uint256) { if (a == 0) { return 0; } // For our first guess, we get the biggest power of 2 which is smaller than the square root of the target. // // We know that the "msb" (most significant bit) of our target number `a` is a power of 2 such that we have // `msb(a) <= a < 2*msb(a)`. This value can be written `msb(a)=2**k` with `k=log2(a)`. // // This can be rewritten `2**log2(a) <= a < 2**(log2(a) + 1)` // → `sqrt(2**k) <= sqrt(a) < sqrt(2**(k+1))` // → `2**(k/2) <= sqrt(a) < 2**((k+1)/2) <= 2**(k/2 + 1)` // // Consequently, `2**(log2(a) / 2)` is a good first approximation of `sqrt(a)` with at least 1 correct bit. uint256 result = 1 << (log2(a) >> 1); // At this point `result` is an estimation with one bit of precision. We know the true value is a uint128, // since it is the square root of a uint256. Newton's method converges quadratically (precision doubles at // every iteration). We thus need at most 7 iteration to turn our partial result with one bit of precision // into the expected uint128 result. unchecked { result = (result + a / result) >> 1; result = (result + a / result) >> 1; result = (result + a / result) >> 1; result = (result + a / result) >> 1; result = (result + a / result) >> 1; result = (result + a / result) >> 1; result = (result + a / result) >> 1; return min(result, a / result); } } /** * @notice Calculates sqrt(a), following the selected rounding direction. */ function sqrt(uint256 a, Rounding rounding) internal pure returns (uint256) { unchecked { uint256 result = sqrt(a); return result + (rounding == Rounding.Up && result * result < a ? 1 : 0); } } /** * @dev Return the log in base 2, rounded down, of a positive value. * Returns 0 if given 0. */ function log2(uint256 value) internal pure returns (uint256) { uint256 result = 0; unchecked { if (value >> 128 > 0) { value >>= 128; result += 128; } if (value >> 64 > 0) { value >>= 64; result += 64; } if (value >> 32 > 0) { value >>= 32; result += 32; } if (value >> 16 > 0) { value >>= 16; result += 16; } if (value >> 8 > 0) { value >>= 8; result += 8; } if (value >> 4 > 0) { value >>= 4; result += 4; } if (value >> 2 > 0) { value >>= 2; result += 2; } if (value >> 1 > 0) { result += 1; } } return result; } /** * @dev Return the log in base 2, following the selected rounding direction, of a positive value. * Returns 0 if given 0. */ function log2(uint256 value, Rounding rounding) internal pure returns (uint256) { unchecked { uint256 result = log2(value); return result + (rounding == Rounding.Up && 1 << result < value ? 1 : 0); } } /** * @dev Return the log in base 10, rounded down, of a positive value. * Returns 0 if given 0. */ function log10(uint256 value) internal pure returns (uint256) { uint256 result = 0; unchecked { if (value >= 10 ** 64) { value /= 10 ** 64; result += 64; } if (value >= 10 ** 32) { value /= 10 ** 32; result += 32; } if (value >= 10 ** 16) { value /= 10 ** 16; result += 16; } if (value >= 10 ** 8) { value /= 10 ** 8; result += 8; } if (value >= 10 ** 4) { value /= 10 ** 4; result += 4; } if (value >= 10 ** 2) { value /= 10 ** 2; result += 2; } if (value >= 10 ** 1) { result += 1; } } return result; } /** * @dev Return the log in base 10, following the selected rounding direction, of a positive value. * Returns 0 if given 0. */ function log10(uint256 value, Rounding rounding) internal pure returns (uint256) { unchecked { uint256 result = log10(value); return result + (rounding == Rounding.Up && 10 ** result < value ? 1 : 0); } } /** * @dev Return the log in base 256, rounded down, of a positive value. * Returns 0 if given 0. * * Adding one to the result gives the number of pairs of hex symbols needed to represent `value` as a hex string. */ function log256(uint256 value) internal pure returns (uint256) { uint256 result = 0; unchecked { if (value >> 128 > 0) { value >>= 128; result += 16; } if (value >> 64 > 0) { value >>= 64; result += 8; } if (value >> 32 > 0) { value >>= 32; result += 4; } if (value >> 16 > 0) { value >>= 16; result += 2; } if (value >> 8 > 0) { result += 1; } } return result; } /** * @dev Return the log in base 256, following the selected rounding direction, of a positive value. * Returns 0 if given 0. */ function log256(uint256 value, Rounding rounding) internal pure returns (uint256) { unchecked { uint256 result = log256(value); return result + (rounding == Rounding.Up && 1 << (result << 3) < value ? 1 : 0); } } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.8.0) (utils/math/SignedMath.sol) pragma solidity ^0.8.0; /** * @dev Standard signed math utilities missing in the Solidity language. */ library SignedMathUpgradeable { /** * @dev Returns the largest of two signed numbers. */ function max(int256 a, int256 b) internal pure returns (int256) { return a > b ? a : b; } /** * @dev Returns the smallest of two signed numbers. */ function min(int256 a, int256 b) internal pure returns (int256) { return a < b ? a : b; } /** * @dev Returns the average of two signed numbers without overflow. * The result is rounded towards zero. */ function average(int256 a, int256 b) internal pure returns (int256) { // Formula from the book "Hacker's Delight" int256 x = (a & b) + ((a ^ b) >> 1); return x + (int256(uint256(x) >> 255) & (a ^ b)); } /** * @dev Returns the absolute unsigned value of a signed value. */ function abs(int256 n) internal pure returns (uint256) { unchecked { // must be unchecked in order to support `n = type(int256).min` return uint256(n >= 0 ? n : -n); } } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts v4.4.1 (proxy/beacon/IBeacon.sol) pragma solidity ^0.8.0; /** * @dev This is the interface that {BeaconProxy} expects of its beacon. */ interface IBeacon { /** * @dev Must return an address that can be used as a delegate call target. * * {BeaconProxy} will check that this address is a contract. */ function implementation() external view returns (address); }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.9.0) (interfaces/IERC1967.sol) pragma solidity ^0.8.0; /** * @dev ERC-1967: Proxy Storage Slots. This interface contains the events defined in the ERC. * * _Available since v4.8.3._ */ interface IERC1967 { /** * @dev Emitted when the implementation is upgraded. */ event Upgraded(address indexed implementation); /** * @dev Emitted when the admin account has changed. */ event AdminChanged(address previousAdmin, address newAdmin); /** * @dev Emitted when the beacon is changed. */ event BeaconUpgraded(address indexed beacon); }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v4.9.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 * * Furthermore, `isContract` will also return true if the target contract within * the same transaction is already scheduled for destruction by `SELFDESTRUCT`, * which only has an effect at the end of a transaction. * ==== * * [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://consensys.net/diligence/blog/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.8.0/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 functionCallWithValue(target, data, 0, "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"); (bool success, bytes memory returndata) = target.call{value: value}(data); return verifyCallResultFromTarget(target, 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) { (bool success, bytes memory returndata) = target.staticcall(data); return verifyCallResultFromTarget(target, 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) { (bool success, bytes memory returndata) = target.delegatecall(data); return verifyCallResultFromTarget(target, success, returndata, errorMessage); } /** * @dev Tool to verify that a low level call to smart-contract was successful, and revert (either by bubbling * the revert reason or using the provided one) in case of unsuccessful call or if target was not a contract. * * _Available since v4.8._ */ function verifyCallResultFromTarget( address target, bool success, bytes memory returndata, string memory errorMessage ) internal view returns (bytes memory) { if (success) { if (returndata.length == 0) { // only check isContract if the call was successful and the return data is empty // otherwise we already know that it was a contract require(isContract(target), "Address: call to non-contract"); } return returndata; } else { _revert(returndata, errorMessage); } } /** * @dev Tool to verify that a low level call was successful, and revert if it wasn't, either by bubbling the * revert reason or 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 { _revert(returndata, errorMessage); } } function _revert(bytes memory returndata, string memory errorMessage) private pure { // 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.9.0) (utils/StorageSlot.sol) // This file was procedurally generated from scripts/generate/templates/StorageSlot.js. pragma solidity ^0.8.0; /** * @dev Library for reading and writing primitive types to specific storage slots. * * Storage slots are often used to avoid storage conflict when dealing with upgradeable contracts. * This library helps with reading and writing to such slots without the need for inline assembly. * * The functions in this library return Slot structs that contain a `value` member that can be used to read or write. * * Example usage to set ERC1967 implementation slot: * ```solidity * contract ERC1967 { * bytes32 internal constant _IMPLEMENTATION_SLOT = 0x360894a13ba1a3210667c828492db98dca3e2076cc3735a920a3ca505d382bbc; * * function _getImplementation() internal view returns (address) { * return StorageSlot.getAddressSlot(_IMPLEMENTATION_SLOT).value; * } * * function _setImplementation(address newImplementation) internal { * require(Address.isContract(newImplementation), "ERC1967: new implementation is not a contract"); * StorageSlot.getAddressSlot(_IMPLEMENTATION_SLOT).value = newImplementation; * } * } * ``` * * _Available since v4.1 for `address`, `bool`, `bytes32`, `uint256`._ * _Available since v4.9 for `string`, `bytes`._ */ library StorageSlot { struct AddressSlot { address value; } struct BooleanSlot { bool value; } struct Bytes32Slot { bytes32 value; } struct Uint256Slot { uint256 value; } struct StringSlot { string value; } struct BytesSlot { bytes value; } /** * @dev Returns an `AddressSlot` with member `value` located at `slot`. */ function getAddressSlot(bytes32 slot) internal pure returns (AddressSlot storage r) { /// @solidity memory-safe-assembly assembly { r.slot := slot } } /** * @dev Returns an `BooleanSlot` with member `value` located at `slot`. */ function getBooleanSlot(bytes32 slot) internal pure returns (BooleanSlot storage r) { /// @solidity memory-safe-assembly assembly { r.slot := slot } } /** * @dev Returns an `Bytes32Slot` with member `value` located at `slot`. */ function getBytes32Slot(bytes32 slot) internal pure returns (Bytes32Slot storage r) { /// @solidity memory-safe-assembly assembly { r.slot := slot } } /** * @dev Returns an `Uint256Slot` with member `value` located at `slot`. */ function getUint256Slot(bytes32 slot) internal pure returns (Uint256Slot storage r) { /// @solidity memory-safe-assembly assembly { r.slot := slot } } /** * @dev Returns an `StringSlot` with member `value` located at `slot`. */ function getStringSlot(bytes32 slot) internal pure returns (StringSlot storage r) { /// @solidity memory-safe-assembly assembly { r.slot := slot } } /** * @dev Returns an `StringSlot` representation of the string storage pointer `store`. */ function getStringSlot(string storage store) internal pure returns (StringSlot storage r) { /// @solidity memory-safe-assembly assembly { r.slot := store.slot } } /** * @dev Returns an `BytesSlot` with member `value` located at `slot`. */ function getBytesSlot(bytes32 slot) internal pure returns (BytesSlot storage r) { /// @solidity memory-safe-assembly assembly { r.slot := slot } } /** * @dev Returns an `BytesSlot` representation of the bytes storage pointer `store`. */ function getBytesSlot(bytes storage store) internal pure returns (BytesSlot storage r) { /// @solidity memory-safe-assembly assembly { r.slot := store.slot } } }
{ "optimizer": { "enabled": true, "mode": "z" }, "outputSelection": { "*": { "*": [ "abi" ] } }, "detectMissingLibraries": false, "forceEVMLA": false, "enableEraVMExtensions": false, "libraries": {} }
Contract Security Audit
- No Contract Security Audit Submitted- Submit Audit Here
[{"inputs":[],"name":"EmptyString","type":"error"},{"inputs":[],"name":"InvalidManager","type":"error"},{"inputs":[],"name":"ManagerDoesNotExist","type":"error"},{"inputs":[],"name":"ManagerRemoveBlocked","type":"error"},{"inputs":[],"name":"ManagerSwapBlocked","type":"error"},{"inputs":[],"name":"MintFrozen","type":"error"},{"inputs":[],"name":"MinterRegistrationInvalid","type":"error"},{"inputs":[],"name":"MismatchedArrayLengths","type":"error"},{"inputs":[],"name":"NotAContract","type":"error"},{"inputs":[],"name":"NotMinter","type":"error"},{"inputs":[],"name":"OperatorNotAllowed","type":"error"},{"inputs":[],"name":"OverLimitSupply","type":"error"},{"inputs":[],"name":"RoyaltyBPSInvalid","type":"error"},{"inputs":[],"name":"RoyaltySetBlocked","type":"error"},{"inputs":[],"name":"TokenDoesNotExist","type":"error"},{"inputs":[],"name":"TokenNotInRange","type":"error"},{"inputs":[],"name":"Unauthorized","type":"error"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"previousAdmin","type":"address"},{"indexed":false,"internalType":"address","name":"newAdmin","type":"address"}],"name":"AdminChanged","type":"event"},{"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":"string","name":"oldBaseUri","type":"string"},{"indexed":false,"internalType":"string","name":"newBaseURI","type":"string"}],"name":"BaseURISet","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"beacon","type":"address"}],"name":"BeaconUpgraded","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"recipientAddress","type":"address"},{"indexed":true,"internalType":"uint16","name":"royaltyPercentageBPS","type":"uint16"}],"name":"DefaultRoyaltySet","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"newDefaultTokenManager","type":"address"}],"name":"DefaultTokenManagerChanged","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256[]","name":"ids","type":"uint256[]"},{"components":[{"internalType":"address","name":"recipientAddress","type":"address"},{"internalType":"uint16","name":"royaltyPercentageBPS","type":"uint16"}],"indexed":false,"internalType":"struct IRoyaltyManager.Royalty[]","name":"_newRoyalties","type":"tuple[]"}],"name":"GranularRoyaltiesSet","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256[]","name":"_ids","type":"uint256[]"}],"name":"GranularTokenManagersRemoved","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256[]","name":"_ids","type":"uint256[]"},{"indexed":false,"internalType":"address[]","name":"_tokenManagers","type":"address[]"}],"name":"GranularTokenManagersSet","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint8","name":"version","type":"uint8"}],"name":"Initialized","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"uint256","name":"newLimitSupply","type":"uint256"}],"name":"LimitSupplySet","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"minter","type":"address"},{"indexed":true,"internalType":"bool","name":"registered","type":"bool"}],"name":"MinterRegistrationChanged","type":"event"},{"anonymous":false,"inputs":[],"name":"MintsFrozen","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"previousOwner","type":"address"},{"indexed":true,"internalType":"address","name":"newOwner","type":"address"}],"name":"OwnershipTransferred","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"newRoyaltyManager","type":"address"}],"name":"RoyaltyManagerChanged","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"uint256[]","name":"ids","type":"uint256[]"},{"indexed":false,"internalType":"string[]","name":"uris","type":"string[]"}],"name":"TokenURIsSet","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":true,"internalType":"address","name":"implementation","type":"address"}],"name":"Upgraded","type":"event"},{"inputs":[{"internalType":"address","name":"operator","type":"address"},{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"approve","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"owner","type":"address"}],"name":"balanceOf","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"baseURI","outputs":[{"internalType":"string","name":"","type":"string"}],"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":"defaultManager","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"freezeMints","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":"address","name":"creator","type":"address"},{"internalType":"string","name":"_contractURI","type":"string"},{"components":[{"internalType":"address","name":"recipientAddress","type":"address"},{"internalType":"uint16","name":"royaltyPercentageBPS","type":"uint16"}],"internalType":"struct IRoyaltyManager.Royalty","name":"defaultRoyalty","type":"tuple"},{"internalType":"address","name":"_defaultTokenManager","type":"address"},{"internalType":"string","name":"_name","type":"string"},{"internalType":"string","name":"_symbol","type":"string"},{"internalType":"address","name":"trustedForwarder","type":"address"},{"internalType":"address","name":"initialMinter","type":"address"},{"internalType":"string","name":"newBaseURI","type":"string"},{"internalType":"uint256","name":"_limitSupply","type":"uint256"},{"internalType":"bool","name":"useMarketplaceFiltererRegistry","type":"bool"},{"internalType":"address","name":"_observability","type":"address"}],"name":"initialize","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"creator","type":"address"},{"internalType":"bytes","name":"data","type":"bytes"}],"name":"initialize","outputs":[],"stateMutability":"nonpayable","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":"forwarder","type":"address"}],"name":"isTrustedForwarder","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"limitSupply","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"recipient","type":"address"},{"internalType":"uint256","name":"amount","type":"uint256"}],"name":"mintAmountToOneRecipient","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address[]","name":"recipients","type":"address[]"}],"name":"mintOneToMultipleRecipients","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"recipient","type":"address"}],"name":"mintOneToOneRecipient","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address[]","name":"recipients","type":"address[]"},{"internalType":"uint256","name":"amount","type":"uint256"}],"name":"mintSameAmountToMultipleRecipients","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"recipient","type":"address"},{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"mintSpecificTokenToOneRecipient","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"recipient","type":"address"},{"internalType":"uint256[]","name":"tokenIds","type":"uint256[]"}],"name":"mintSpecificTokensToOneRecipient","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"minters","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":"observability","outputs":[{"internalType":"contract IObservability","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"operatorFiltererRegistry","outputs":[{"internalType":"address","name":"","type":"address"}],"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":"proxiableUUID","outputs":[{"internalType":"bytes32","name":"","type":"bytes32"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"minter","type":"address"}],"name":"registerMinter","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"removeDefaultTokenManager","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256[]","name":"_ids","type":"uint256[]"}],"name":"removeGranularTokenManagers","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"removeRoyaltyManager","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"renounceOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_tokenGroupingId","type":"uint256"},{"internalType":"uint256","name":"_salePrice","type":"uint256"}],"name":"royaltyInfo","outputs":[{"internalType":"address","name":"receiver","type":"address"},{"internalType":"uint256","name":"royaltyAmount","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"royaltyManager","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","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":"newBaseURI","type":"string"}],"name":"setBaseURI","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"string","name":"newName","type":"string"},{"internalType":"string","name":"newSymbol","type":"string"},{"internalType":"string","name":"newContractUri","type":"string"}],"name":"setContractMetadata","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"components":[{"internalType":"address","name":"recipientAddress","type":"address"},{"internalType":"uint16","name":"royaltyPercentageBPS","type":"uint16"}],"internalType":"struct IRoyaltyManager.Royalty","name":"_royalty","type":"tuple"}],"name":"setDefaultRoyalty","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_defaultTokenManager","type":"address"}],"name":"setDefaultTokenManager","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256[]","name":"ids","type":"uint256[]"},{"components":[{"internalType":"address","name":"recipientAddress","type":"address"},{"internalType":"uint16","name":"royaltyPercentageBPS","type":"uint16"}],"internalType":"struct IRoyaltyManager.Royalty[]","name":"_newRoyalties","type":"tuple[]"}],"name":"setGranularRoyalties","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256[]","name":"_ids","type":"uint256[]"},{"internalType":"address[]","name":"_tokenManagers","type":"address[]"}],"name":"setGranularTokenManagers","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_limitSupply","type":"uint256"}],"name":"setLimitSupply","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"newRegistry","type":"address"},{"internalType":"address","name":"subscription","type":"address"}],"name":"setRegistryAndSubscription","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_royaltyManager","type":"address"}],"name":"setRoyaltyManager","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256[]","name":"ids","type":"uint256[]"},{"internalType":"string[]","name":"uris","type":"string[]"}],"name":"setTokenURIs","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"supply","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","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":"id","type":"uint256"}],"name":"tokenManager","outputs":[{"internalType":"address","name":"manager","type":"address"}],"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":"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":[{"internalType":"address","name":"newOwner","type":"address"}],"name":"transferOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"minter","type":"address"}],"name":"unregisterMinter","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"newImplementation","type":"address"}],"name":"upgradeTo","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"newImplementation","type":"address"},{"internalType":"bytes","name":"data","type":"bytes"}],"name":"upgradeToAndCall","outputs":[],"stateMutability":"payable","type":"function"}]
Contract Creation Code
9c4d535b0000000000000000000000000000000000000000000000000000000000000000010007f174c19fce568b0b9be7cb81afb19745794425f8157ad5258c2265c61d00000000000000000000000000000000000000000000000000000000000000600000000000000000000000000000000000000000000000000000000000000000
Deployed Bytecode
0x000400000000000200260000000000020000006003100270000007230830019700030000008103550002000000010355000007230030019d000000010020019000000000030004160000009c0000c13d0000008002000039000000400020043f000000040080008c000010e80000413d000000000201043b000000e006200270000007250060009c000002fc0000613d000007260060009c000000aa0000613d000007270060009c000000cf070000390000030d0000613d000007280060009c0000010302000039000003510000613d000007290060009c0000000402100370000000ae0000613d0000072a0060009c00000024041003700000010509000039000003680000613d0000072b0060009c000003820000613d0000072c0060009c000000aa0000613d0000072d0060009c000003ac0000613d0000072e0060009c000000cb05000039000000b50000613d0000072f0060009c0000004405100370000000b90000613d000007300060009c000003db0000613d000007310060009c000001040b0000390000016c0a000039000003fa0000613d000007320060009c000000d80000613d000007330060009c000000dd0000613d000007340060009c000004500000613d000007350060009c000004580000613d000007360060009c0000049d0000613d000007370060009c000004ad0000613d000007380060009c000004ce0000613d000007390060009c000001240000613d0000073a0060009c000001270000613d0000073b0060009c0000017c0000613d0000073c0060009c000001a20000613d0000073d0060009c000001b40000613d0000073e0060009c000005200000613d0000073f0060009c0000053d0000613d000007400060009c000005480000613d000007410060009c000005730000613d000007420060009c0000057e0000613d000007430060009c0000022b0000613d000007440060009c000005c20000613d000007450060009c000005da0000613d000007460060009c000005ed0000613d000007470060009c000006070000613d000007480060009c000006220000613d000007490060009c000006390000613d0000074a0060009c0000063d0000613d0000074b0060009c0000064e0000613d0000074c0060009c0000067b0000613d0000074d0060009c0000068c0000613d0000074e0060009c000002350000613d0000074f0060009c000006900000613d000007500060009c000002950000613d000007510060009c000006b10000613d000007520060009c000006c30000613d000007530060009c000002ee0000613d000007540060009c000006d30000613d000007550060009c000007d30000613d000007560060009c000007e40000613d000007570060009c000007f70000613d000007580060009c000008000000613d000007590060009c000008040000613d0000075a0060009c000008840000613d0000075b0060009c000008940000613d0000075c0060009c000008b00000613d0000075d0060009c000010e80000c13d000000240080008c000010e80000413d000000000003004b000010e80000c13d000000000102043b000007c800100198000010e80000c13d000007c901100197000007ca0010009c00000000020000390000000102006039000007cb0010009c00000001022061bf000007a90010009c00000001022061bf000000800020043f000007fd0000013d000000a001000039000000400010043f000000000003004b000010e80000c13d0000000001000410000000800010043f000001400000044300000160001004430000002001000039000001000010044300000001010000390000012000100443000007240100004100001c890001042e000000000003004b000010e80000c13d0000016d01000039000000db0000013d000000240080008c000010e80000413d000000000003004b000010e80000c13d000000000102043b1c8818790000040f000002310000013d000000000003004b000010e80000c13d000000000105041a000007fb0000013d000000640080008c000010e80000413d000000000003004b000010e80000c13d000000000202043b000f00000002001d000007600020009c000010e80000213d000000000204043b000e00000002001d000007600020009c000010e80000213d000000000405043b0000006502000039000000000202041a00000760032001970000000002000411000000000032004b000000ce0000c13d000100ce0000003d00001baa0000013d00000000010200190000000002040019000d00000004001d1c8819450000040f1c8818860000040f0000000f010000290000000e020000290000000d030000291c8819a80000040f00000e9d0000013d000000000003004b000010e80000c13d0000016e01000039000000000101041a000007fc0000013d000000440080008c000010e80000413d000000000003004b000010e80000c13d000000000202043b000007670020009c000010e80000213d0000002303200039000000000083004b000010e80000813d0000000403200039000000000331034f000000000303043b000a00000003001d000007670030009c000010e80000213d000900240020003d0000000a0200002900000005022002100000000902200029000000000082004b000010e80000213d000000000204043b000d00000002001d0000006502000039000000000202041a00000760032001970000000002000411000000000032004b000000fd0000c13d000100fd0000003d00001baa0000013d0000076001200197000101000000003d00001b4c0000013d000008c30000613d000101030000003d00001c460000013d0000076101100197000007620010009c00000a4d0000613d0000016d01000039000000000301041a00000000020000190000000a0020006c000011450000813d000b00000002001d0000000501200210000c00090010002d00000000020000190000000d0020006c000001210000813d000f00000002001d000000010100008a000000000013004b0000108b0000613d0000000c010000290000000201100367000000000101043b000007600010009c000010e80000213d0000000102300039000e00000002001d1c8819690000040f0000000e030000290000000f0200002900000001022000390000010f0000013d0000000b020000290000000102200039000001090000013d00000000010800191c8813280000040f1c8813ee0000040f000000000003004b000010e80000c13d00000000010800191c8813410000040f000b00000001001d000c00000002001d0000000001000415000900000001001d000101310000003d00001c780000013d00000760021001970000000001000411000000000021004b000001370000c13d000101370000003d00001b5a0000013d000a07600010019b00000000020000190000000c0020006c00000b910000813d000e00000002001d00000005012002100000000b01100029000d00000001001d0000000201100367000000000101043b000f00000001001d000000000010043f000000cc01000039000101460000003d00001bbb0000013d0000076002100198000006c10000613d000000400500043d00000024015000390000000f0300002900000000003104350000079301000041000000000015043500000004015000390000000a0300002900000000003104350000000001000414000000040020008c000001560000c13d00000001020000310000015f0000013d00000000030500190000004404000039000f00000005001d00000020060000391c8812830000040f0000000102000031000000000001004b00000d460000613d0000000f05000029000101610000003d00001c600000013d000007670010009c000008e40000213d0000000100300190000008e40000c13d000000400010043f000007680020009c000010e80000213d000000200020008c000010e80000413d00000000010504330001016d0000003d00001b550000013d000010e80000c13d000000000001004b00000a6f0000613d0000000d010000290000000201100367000000000101043b000000000010043f000000cc01000039000101770000003d00001bb50000013d0000076902200197000000000021041b0000000e020000290000000102200039000001390000013d000000240080008c000010e80000413d000000000003004b000010e80000c13d000000000102043b000f00000001001d000007600010009c000010e80000213d0000000001000415000e00000001001d1c8814660000040f0000000f010000291c88179d0000040f000000000001004b0000068a0000613d0000000f010000291c8817c50000040f000000000001004b0000068a0000c13d0000002402000039000000400100043d0000002003100039000007a90400004100000000004304350000002404100039000007aa0500004100000000005404350000000000210435000007ab0010009c000008e40000213d0000006002100039000000400020043f0000000f02000029000000040020008c00000e4c0000c13d0000000001030433000000000010043f00000e500000013d000000000003004b000010e80000c13d0000000001000412002600000001001d002500000000003d000080050100003900000044030000390000000004000415000000260440008a0000000504400210000007a5020000411c881b1f0000040f00000760011001970000000002000410000000000012004b000008e80000c13d000007a801000041000007fc0000013d000001a40080008c000010e80000413d000000000003004b000010e80000c13d000000000202043b000f00000002001d000007600020009c000010e80000213d000000000304043b000007670030009c000010e80000213d0000002302300039000000000082004b000010e80000813d0000000402300039000000000121034f000000000201043b0000002401300039000d00000008001d00000000030800191c8812f00000040f000c00000001001d000000400100043d000e00000001001d0000077c0010009c000008e40000213d0000000e010000290000004001100039000000400010043f00000002010003670000004402100370000000000202043b000007600020009c0000000d03000029000010e80000213d0000000e0400002900000000022404360000006404100370000000000404043b0000ffff0040008c000010e80000213d00000000004204350000008402100370000000000202043b000b00000002001d000007600020009c000010e80000213d000000a402100370000000000402043b000007670040009c000010e80000213d0000002302400039000000000032004b000010e80000813d000101ec0000003d00001c040000013d0000000d03000029000a00000001001d0000000202000367000000c401200370000000000101043b000007670010009c000010e80000213d0000002304100039000000000034004b000010e80000813d0000000404100039000000000242034f000000000202043b00000024011000391c8812f00000040f0000000d03000029000900000001001d0000000201000367000000e402100370000000000202043b000800000002001d000007600020009c000010e80000213d0000010402100370000000000202043b000700000002001d000007600020009c000010e80000213d0000012402100370000000000402043b000007670040009c000010e80000213d0000002302400039000000000032004b000010e80000813d000102110000003d00001c040000013d000d00000001001d00000002010003670000014402100370000000000202043b000600000002001d0000016402100370000000000202043b000000000002004b0000000003000039000000010300c039000000000032004b000010e80000c13d0000018401100370000000000101043b000500000001001d000007600010009c000010e80000213d0000000001000415000300000001001d000000000200041a0004ff0000200193000200000002001d0000ffff00200190000012510000c13d00000001010000390000125f0000013d000000240080008c000010e80000413d000000000003004b000010e80000c13d000000000102043b1c8814b30000040f000000400300043d0000000000130435000000200200003900000a5b0000013d000000000003004b000010e80000c13d0000000001000415000f00000001001d0001023b0000003d00001c780000013d00000760021001970000000001000411000000000021004b000002410000c13d000102410000003d00001b5a0000013d000000cb02000039000000000202041a0000076002200198000006c10000613d000000400500043d00000793030000410000000000350435000007600110019700000004035000390000000000130435000000240150003900000000000104350000000001000414000000040020008c000002590000613d000000440400003900000020060000390000000003050019000e00000005001d0000000e050000291c8812830000040f0000000e05000029000000000001004b00000a200000613d0001025b0000003d00001bdd0000013d000007670030009c000008e40000213d0000000100200190000008e40000c13d0000000e02000029000000400020043f000007680010009c000010e80000213d000000200010008c000010e80000413d0000000001050433000102680000003d00001b550000013d000010e80000c13d000000000001004b00000a6f0000613d000000cb02000039000000000102041a0000076901100197000000000012041b0000000e01000029000007230010009c000007230100804100000040011002100000000002000414000007230020009c0000072302008041000000c002200210000000000112019f0000076c011001c70000800d020000390000000203000039000007940400004100000000050000191c881b380000040f0000000100200190000010e80000613d000000d001000039000000000101041a000d00000001001d001607600010019b000102860000003d00001b600000013d000000160440008a00000005044002100000076e020000411c881b1f0000040f000000000001004b00000ab00000613d0000000d01000029000007600210019700000795010000410000000e03000029000000000013043500000004013000390000000000010435000000000100041400000f190000013d000000440080008c000010e80000413d000000000003004b000010e80000c13d000000000202043b000007670020009c000010e80000213d0000002303200039000000000083004b000010e80000813d0000000403200039000000000331034f000000000303043b000d00000003001d000007670030009c000010e80000213d000c00240020003d0000000d0200002900000005022002100000000c02200029000000000082004b000010e80000213d000000000204043b000007670020009c000010e80000213d0000002303200039000000000083004b000010e80000813d0000000403200039000000000131034f000000000101043b000b00000001001d000007670010009c000010e80000213d000a00240020003d0000000b0100002900000006011002100000000a01100029000000000081004b000010e80000213d0000000001000415000500000001001d000102c10000003d00001c780000013d00000760021001970000000001000411000000000021004b000002c70000c13d000102c70000003d00001b5a0000013d000000cf02000039000000000202041a000907600020019c00000ffb0000c13d0000003302000039000000000202041a000000000112013f0000076000100198000004cc0000c13d00000000030000190000000d0030006c0000110e0000813d0000000b0030006c000010500000813d00000006013002100000000a0410002900000020024000390000000201000367000000000221034f000000000202043b0000ffff0020008c000010e80000213d000f00000004001d000027100020008c000008920000213d0000000502300210000e00000003001d0000000c02200029000000000121034f000000000101043b000000000010043f000000ce01000039000102e90000003d00001c0a0000013d0000000f020000291c8814760000040f0000000e030000290000000103300039000002d10000013d000000000003004b000010e80000c13d00000000010800191c8813280000040f000f00000002001d0000000003000415000000000400041a000eff00004001930000ffff00400190000c00000003001d000d00000001001d000008f50000c13d0000000101000039000009050000013d000000000003004b000010e80000c13d000000c902000039000000000102041a000000800010043f000000000020043f0000075e0300004100000080020000390000000004000019000000000014004b000008c50000813d0000002002200039000000000503041a000000000052043500000001044000390000000103300039000003050000013d000000000003004b000010e80000c13d0000000001000415000e00000001001d1c8813b60000040f1c8814660000040f000000cf02000039000000000102041a000007c50110019700000762011001c7000000000012041b000000400100043d000007230010009c000f00000001001d000007230100804100000040011002100000000002000414000007230020009c0000072302008041000000c002200210000000000112019f0000076c011001c70000800d020000390000000103000039000007c6040000411c881b380000040f0000000100200190000010e80000613d000000d001000039000000000101041a000d00000001001d002407600010019b0001032f0000003d00001b600000013d000000240440008a00000005044002100000076e020000411c881b1f0000040f000000000001004b000010e80000613d0000000d010000290000076002100197000007c7010000410000000f0300002900000000001304350000000001000414000000040020008c000003440000613d00000004040000390000000f0500002900000000060000191c8812830000040f0000000f03000029000000000001004b00000a200000613d000103460000003d00001c0f0000013d000007670010009c000008e40000213d0000000100200190000008e40000c13d000000400010043f00000097010000390000000102000039000000000021041b00000000010004150000000e020000290000061f0000013d000000000003004b000010e80000c13d0000000005020019000000000202041a000000010320019000000001012002700000007f0110618f0000001f0010008c00000000040000390000000104002039000000000043004b00000a420000c13d000000800010043f000000000003004b000008dc0000613d000000000050043f000007bb030000410000000002000019000000000012004b000008e10000813d000103670000003d00001baf0000013d000003630000013d000000440080008c000010e80000413d000000000003004b000010e80000c13d000000000102043b000f00000001001d000007600010009c000010e80000213d000000000104043b000e00000001001d0000000f010000291c88171b0000040f0000000e01000029000000000010043f000103780000003d00001b640000013d00000760011001970000000f0010006b00000a710000c13d000000400100043d0000004402100039000007c403000041000000000032043500000024021000390000001403000039000006730000013d000000440080008c000010e80000413d000000000003004b000010e80000c13d000000000202043b000f00000002001d000007600020009c000010e80000213d000000000204043b000e00000002001d0000006502000039000000000202041a00000760032001970000000002000411000000000032004b000003940000c13d000103940000003d00001baa0000013d0000076001200197000103970000003d00001b4c0000013d000008c30000613d0001039a0000003d00001c460000013d0000076101100197000007620010009c00000a4d0000613d0000016d01000039000000000201041a000000010100008a000000000012004b0000108b0000613d000d00000002001d0000016e01000039000000000101041a000000000001004b0000000e0200002900000e950000613d000000000021004b00000e950000813d000007be0100004100000c940000013d000000240080008c000010e80000413d000000000003004b000010e80000c13d000000000202043b000f00000002001d000007600020009c000010e80000213d0000006502000039000000000202041a00000760032001970000000002000411000000000032004b000003bc0000c13d000103bc0000003d00001baa0000013d0000076001200197000103bf0000003d00001b4c0000013d000008c30000613d000103c20000003d00001c460000013d0000076101100197000007620010009c00000a4d0000613d0000016d01000039000000000101041a000000010200008a000000000021004b0000108b0000613d0000016e02000039000000000202041a000000010220008a000000000012004b00000bff0000413d0000000102100039000e00000002001d0000000f010000291c8819690000040f0000016d010000390000000e03000029000000000031041b000103d80000003d00001c370000013d000000400100043d0000000000310435000007fe0000013d000000440080008c000010e80000413d000000000003004b000010e80000c13d000000000104043b000f00000001001d000000000102043b000000000010043f000000ce01000039000103e60000003d00001b8f0000013d000000c003000039000000400030043f000000000201041a0000076001200198000000800010043f000000a0022002700000ffff0420018f000000a00040043f000009910000c13d0000010003000039000000400030043f0000012002000039000000cd01000039000000000401041a0000076001400197000000c00010043f000000a0044002700000ffff0440018f000000e00040043f000009920000013d000000640080008c000010e80000413d000000000003004b000010e80000c13d000000000202043b000007670020009c000010e80000213d0000002303200039000000000083004b000010e80000813d0000000403200039000000000331034f000000000303043b000f00000003001d000007670030009c000010e80000213d0000002403200039000e00000003001d0000000f02300029000000000082004b000010e80000213d000000000204043b000007670020009c000010e80000213d0000002303200039000000000083004b000010e80000813d0000000403200039000000000331034f000000000303043b000d00000003001d000007670030009c000010e80000213d0000002403200039000c00000003001d0000000d02300029000000000082004b000010e80000213d000000000205043b000007670020009c000010e80000213d0000002303200039000000000083004b000010e80000813d0000000403200039000000000131034f000000000101043b000b00000001001d000007670010009c000010e80000213d0000002402200039000a00000002001d0000000b01200029000000000081004b000010e80000213d0000000001000415000900000001001d1c8813b60000040f0000010301000039000104370000003d00001b750000013d00000a420000c13d0000000f020000291c8815090000040f0000000f020000290000001f0020008c000000010120021000000003022002100000114e0000a13d0000010303000039000000000030043f000000200300008a0000000f0530017f000007bb03000041000000020400036700000000060000190000000e08000029000000000056004b0000000007860019000011570000813d000000000774034f000000000707043b000000000073041b00000020066000390000000103300039000004470000013d000000240080008c000010e80000413d000000000003004b000010e80000c13d000000000102043b000007600010009c000010e80000213d1c8813ee0000040f000000240080008c000010e80000413d000000000003004b000010e80000c13d000000000102043b000f00000001001d0000000001000415000d00000001001d1c8813b60000040f1c8814660000040f0000016e010000390000000f05000029000000000051041b000000400100043d000007230010009c000e00000001001d000007230100804100000040011002100000000002000414000007230020009c0000072302008041000000c002200210000000000112019f0000076c011001c70000800d020000390000000203000039000007b9040000411c881b380000040f0000000100200190000010e80000613d000104780000003d00001c5c0000013d002207600010019b0001047b0000003d00001b600000013d000000220440008a00000005044002100000076e020000411c881b1f0000040f000000000001004b000010e80000613d0000000c010000290000076002100197000007ba010000410000000e03000029000000000013043500000004013000390000000f0400002900000000004104350000000001000414000000040020008c000004930000613d00000024040000390000000e0500002900000000060000191c8812830000040f0000000e03000029000000000001004b00000a200000613d000104950000003d00001c0f0000013d000007670010009c000008e40000213d0000000100200190000008e40000c13d000000400010043f0001049c0000003d00001c370000013d0000061d0000013d000000640080008c000010e80000413d000000000003004b000010e80000c13d000000000102043b000007600010009c000010e80000213d000000000204043b000007600020009c000010e80000213d000000000305043b000000a004000039000000400040043f000000800000043f0000008004000039000006af0000013d000000240080008c000010e80000413d000000000003004b000010e80000c13d000000000102043b000f00000001001d0000000001000415000d00000001001d1c8814660000040f0000000f010000291c8814b30000040f000000000501001900000000020004110000006501000039000000000101041a0000076001100197000000000012004b000004c40000c13d000000140100008a00000000011000310000000201100367000000000101043b0000006002100270000000000005004b000e00000002001d0000099f0000c13d00000000010200190000000f020000291c8819450000040f000000000001004b0000107b0000c13d000007b80100004100000c940000013d000000240080008c000010e80000413d000000000003004b000010e80000c13d000000000102043b000f00000001001d000007600010009c000010e80000213d0000000001000415000e00000001001d1c8813b60000040f0000000f010000291c8814210000040f000000000001004b000006370000613d000000400100043d000007230010009c000d00000001001d000007230100804100000040011002100000000002000414000007230020009c0000072302008041000000c002200210000000000112019f0000076c011001c70000800d02000039000000030300003900000001060000390000079c040000410000000f050000291c881b380000040f0000000100200190000010e80000613d000104f20000003d00001c5c0000013d001f07600010019b000104f50000003d00001b600000013d0000001f0440008a00000005044002100000076e020000411c881b1f0000040f000000000001004b000010e80000613d0000000c0100002900000760021001970000000d050000290000002401500039000000010300003900000000003104350000079d01000041000000000015043500000004015000390000000f0300002900000000003104350000000001000414000000040020008c000005110000613d000000440400003900000000030500190000000d0500002900000000060000191c8812830000040f0000000d05000029000000000001004b00000a200000613d00000001010000310000001f02100039000000200300008a000000000332016f0000000002530019000105180000003d00001b710000013d000007670020009c000008e40000213d0000000100300190000008e40000c13d000000400020043f000007680010009c0000034e0000a13d000010e80000013d000000240080008c000010e80000413d000000000003004b000010e80000c13d000000000202043b000007670020009c000010e80000213d0000002303200039000000000083004b000010e80000813d000e00040020003d0000000e01100360000000000101043b000f00000001001d000007670010009c000010e80000213d0000002402200039000d00000002001d0000000f01200029000000000081004b000010e80000213d0000000001000415000c00000001001d1c8814660000040f0000000f06000029000000000006004b00000d540000c13d000007a40100004100000c940000013d000000000003004b000010e80000c13d00000000010800191c8812e50000040f0000006502000039000000000202041a000000000112013f000007600010019800000000020000390000000102006039000007f40000013d000000440080008c000010e80000413d000000000003004b000010e80000c13d000000000202043b000d00000002001d000007600020009c000010e80000213d000000000204043b000c00000002001d0000006502000039000000000202041a00000760032001970000000002000411000000000032004b0000055a0000c13d0001055a0000003d00001baa0000013d00000760012001970001055d0000003d00001b4c0000013d000008c30000613d000105600000003d00001c460000013d0000076101100197000007620010009c00000a4d0000613d0000016d01000039000000000101041a000f00000001001d00000000030000190000000c0030006c00000bf80000813d0000000f02000029000000010220003a0000108b0000613d0000000d01000029000f00000002001d000e00000003001d1c8819690000040f0000000e030000290000000103300039000005670000013d000000240080008c000010e80000413d000000000003004b000010e80000c13d000000000102043b000000000010043f000000200090043f000000400200003900000000010000191c881b0b0000040f000007fa0000013d000000440080008c000010e80000413d000000000003004b000010e80000c13d000000000202043b000d00000002001d000007600020009c000010e80000213d000000000204043b000007670020009c000010e80000213d0000002303200039000000000083004b000010e80000813d0000000403200039000000000331034f000000000303043b000c00000003001d000007670030009c000010e80000213d000b00240020003d0000000c0200002900000005022002100000000b02200029000000000082004b000010e80000213d0000006502000039000000000202041a00000760032001970000000002000411000000000032004b000005a00000c13d000105a00000003d00001baa0000013d0000076001200197000105a30000003d00001b4c0000013d000008c30000613d000105a60000003d00001c460000013d0000076101100197000007620010009c00000a4d0000613d0000016d01000039000000000101041a000f00000001001d0000016e01000039000000000101041a000a00000001001d000000000001004b000010540000c13d00000000020000190000000c0020006c00000e9b0000813d000e00000002001d00000005012002100000000b011000290000000201100367000000000201043b0000000d010000291c8819690000040f0000000f01000029000000010010003a0000108b0000413d000f00010010003d0000000e020000290000000102200039000005b20000013d000000000003004b000010e80000c13d0000013903000039000000000203041a000000010420019000000001012002700000007f0110618f0000001f0010008c00000000050000390000000105002039000000000552013f000000010050019000000a420000c13d000000800010043f000000000004004b000008dc0000613d000000000030043f000007a2030000410000000002000019000000000012004b000008e10000813d000105d90000003d00001baf0000013d000005d50000013d000000240080008c000010e80000413d000000000003004b000010e80000c13d000000000102043b000007600010009c000010e80000213d000000000001004b00000a7d0000c13d0000076501000041000000800010043f0000002001000039000000840010043f0000001501000039000000a40010043f000007a101000041000000c40010043f0000008001000039000006790000013d000000000003004b000010e80000c13d1c8813b60000040f0000003303000039000000000103041a0000076902100197000000000023041b000000400200043d000007230020009c000007230200804100000040022002100000000003000414000007230030009c0000072303008041000000c003300210000000000223019f00000760051001970000076c012001c70000800d020000390000000303000039000007a00400004100000000060000191c881b380000040f0000000100200190000010e80000613d00000e9d0000013d000000440080008c000010e80000413d000000000003004b000010e80000c13d000000000102043b000f00000001001d000007600010009c000010e80000213d000000000104043b000e00000001001d000007600010009c000010e80000213d0000000001000415000d00000001001d1c8813b60000040f0000016b01000039000000000201041a00000769022001970000000f022001af000000000021041b0000000f0000006b00000c010000c13d00000000010004150000000d020000290000000001120049000000000100000200000e9d0000013d000000240080008c000010e80000413d000000000003004b000010e80000c13d000000000102043b000f00000001001d000007600010009c000010e80000213d0000000001000415000e00000001001d1c8813b60000040f0000000f01000029000000000010043f000000ca01000039000000200010043f000000400200003900000000010000191c881b0b0000040f000000000301041a000000000003004b00000a820000c13d000007b40100004100000c940000013d000000000003004b000010e80000c13d0000003301000039000007fa0000013d000000000003004b000010e80000c13d00000000020b041a000106420000003d00001bfb0000013d00000a420000c13d000000800010043f000000000003004b000008dc0000613d0000000000b0043f00000798030000410000000002000019000000000012004b000008e10000813d0001064d0000003d00001baf0000013d000006490000013d000000440080008c000010e80000413d000000000003004b000010e80000c13d000000000102043b000f00000001001d000007600010009c000010e80000213d000000000204043b000000000002004b0000000001000039000000010100c039000e00000002001d000000000012004b000010e80000c13d0000000f010000291c88171b0000040f0000006501000039000000000101041a00000760011001970000000004000411000000000014004b0000066a0000c13d000000140100008a00000000011000310000000201100367000000000101043b000000600410027000000760014001970000000f0010006c00000c0c0000c13d000000400100043d0000004402100039000007970300004100000000003204350000002402100039000000190300003900000000003204350000076502000041000000000021043500000004021000390000002003000039000000000032043500000064020000391c881af90000040f000000240080008c000010e80000413d000000000003004b000010e80000c13d000000000102043b000f00000001001d000007600010009c000010e80000213d0000000001000415000e00000001001d1c8814660000040f0000000f010000291c88176a0000040f000000000001004b00000ab30000c13d000007ae0100004100000c940000013d000000000003004b000010e80000c13d0000016b01000039000007fa0000013d000000840080008c000010e80000413d000000000003004b000010e80000c13d000000000202043b000f00000002001d000007600020009c000010e80000213d000000000204043b000e00000002001d000007600020009c000010e80000213d0000006402100370000000000302043b000007670030009c000010e80000213d0000002302300039000000000082004b000010e80000813d0000000402300039000000000121034f000000000201043b000000000105043b000d00000001001d000000240130003900000000030800191c8812f00000040f00000000040100190000000f010000290000000e020000290000000d030000291c8818920000040f00000e9d0000013d000000000003004b000010e80000c13d0000000001000415000f00000001001d000106b70000003d00001c780000013d00000760021001970000000001000411000000000021004b000006bd0000c13d000106bd0000003d00001b5a0000013d000000cf02000039000000000202041a000007600220019800000a0e0000c13d000007b30100004100000c940000013d000000240080008c000010e80000413d000000000003004b000010e80000c13d000000000102043b000f00000001001d000000000010043f000000200090043f000000400200003900000000010000191c881b0b0000040f000000000101041a000007600010019800000a320000c13d0000078a0100004100000c940000013d000000000003004b000010e80000c13d00000000010800191c88138c0000040f000700000001001d000800000002001d000c00000003001d000f00000004001d0000000001000415000500000001001d1c8814660000040f0000000f02000029000000080020006b00000a460000c13d0000000002000019000000080020006c00000dfc0000813d000b00000002001d00000005042002100000000c014000290000000202000367000000000112034f000000000301043b00000000090000310000000c0590006a0000001f0550008a0000077a065001970000077a07300197000000000867013f000000000067004b00000000060000190000077a06004041000000000053004b00000000050000190000077a050080410000077a0080009c000000000605c0190000000704400029000000000442034f000000000104043b000000000006004b000010e80000c13d0000000c03300029000000000232034f000000000502043b000007670050009c000010e80000213d000000000759004900000020063000390000077a027001970000077a03600197000000000423013f000000000023004b00000000020000190000077a02004041000e00000006001d000000000076004b00000000030000190000077a030020410000077a0040009c000000000203c019000000000002004b000010e80000c13d000f00000005001d000d00000001001d1c8814b30000040f00000000020100190000006501000039000000000101041a00000760011001970000000003000411000000000013004b00000000010300190000071f0000c13d0001071f0000003d00001b5a0000013d000000000002004b000000200300008a0000000f08000029000000000938016f000a001f0080003d000900000009001d0000076c0000613d000000400b00043d0000004403b00039000000600400003900000000004304350000002403b000390000000d0a0000290000000000a304350000077b0300004100000000003b043500000760011001970000000403b0003900000000001304350000006401b0003900000000008104350000008401b0003900000000039100190000000e040000290000000204400367000000000009004b000007400000613d000000000504034f0000000006010019000000005705043c0000000006760436000000000036004b0000073c0000c13d0000001f05800190000007460000613d000000000494034f0000000305500210000107460000003d00001b420000013d000000000181001900000000000104350000000001000414000000040020008c000007570000613d000000200300008a0000000a0330017f000000840430003900000000030b001900000000050b0019000000200600003900060000000b001d1c8812830000040f000000060b0000290000000d0a000029000000000001004b00000a200000613d000107590000003d00001c3b0000013d0000000002b300190001075c0000003d00001b710000013d000007670020009c000008e40000213d0000000100300190000008e40000c13d000000400020043f000007680010009c000010e80000213d000000200010008c000010e80000413d00000000010b0433000107680000003d00001b550000013d000010e80000c13d000000000001004b000007720000c13d000004cc0000013d0000003302000039000000000202041a000000000112013f00000760001001980000000d0a000029000004cc0000c13d0000000000a0043f0000013501000039000107760000003d00001c0a0000013d0000000005010019000000000101041a000000010010019000000001031002700000007f0330618f0000001f0030008c00000000020000390000000102002039000000000121013f000000010010019000000a420000c13d000000200030008c0000000f04000029000d00000005001d0000079a0000413d000000000050043f00000000010000190000002002000039000600000003001d1c881b0b0000040f0000000d050000290000000f040000290000000a020000290000000503200270000000200040008c000000000300401900000006020000290000001f02200039000000050220027000000000022100190000000001310019000000000021004b0000079a0000813d000000000001041b0000000101100039000007950000013d000000200040008c00000001064002100000000307400210000007b20000413d000600000007001d000a00000006001d000000000050043f000000000100001900000020020000391c881b0b0000040f0000000f05000029000000020200036700000000030000190000000e060000290000000907000029000000000073004b0000000004630019000007b80000813d000000000442034f000000000404043b000000000041041b00000020033000390000000101100039000007a90000013d000000000004004b0000000e01000029000007c70000613d0000000201100367000000000101043b000007c80000013d000000000057004b000007c30000813d0000000603000029000000f80330018f000000010500008a000000000335022f000000000353013f000000000242034f000000000202043b000000000232016f000000000021041b0000000a0100002900000001011001bf0000000d05000029000007cd0000013d0000000001000019000000010300008a000000000273022f000000000232013f000000000121016f000000000161019f000000000015041b0000000b02000029000000010020003a00000001022000390000108b0000413d000006e20000013d000000000003004b000010e80000c13d00000000020a041a000107d80000003d00001bfb0000013d00000a420000c13d000000800010043f000000000003004b000008dc0000613d0000000000a0043f00000773030000410000000002000019000000000012004b000008e10000813d000107e30000003d00001baf0000013d000007df0000013d000000000003004b000010e80000c13d00000000010800191c88137d0000040f000f00000002001d0000076001100197000000000010043f0000010801000039000107ee0000003d00001b8f0000013d0000000f020000291c8818640000040f000000000101041a000000ff001001900000000002000039000000010200c039000000400100043d0000000000210435000007fe0000013d000000000003004b000010e80000c13d000000d001000039000000000101041a0000076001100197000000800010043f0000008001000039000000200200003900000e9f0000013d000000000003004b000010e80000c13d000000000107041a000007fb0000013d000000000003004b000010e80000c13d00000000010800191c88138c0000040f000700000001001d000a00000002001d000800000003001d000900000004001d0000000001000415000400000001001d000108100000003d00001c780000013d00000760021001970000000001000411000000000021004b000008160000c13d000108160000003d00001b5a0000013d0000003302000039000000000202041a000000000212013f000607600010019b000507600020019b00000000020000190000000a0020006c00000cf60000813d000000090020006c000010500000813d000e00000002001d0000000502200210000f00000002001d0000000801200029000d00000001001d0000000201100367000000000101043b000007600010009c000010e80000213d1c88176a0000040f000000000001004b0000068a0000613d0000000f020000290000000701200029000f00000001001d0000000201100367000000000101043b1c8814b30000040f000000000001004b0000086f0000613d000c00000001001d0000000f010000290000000201100367000000000101043b000b00000001001d0000000e01000029000000000010043f000000cc010000390001083e0000003d00001bbb0000013d000000400500043d00000024025000390000000b030000290000000000320435000007600110019700000044025000390000000000120435000007720100004100000000001504350000000401500039000000060200002900000000002104350000000c020000290000000001000414000000040020008c000008570000613d00000000030500190000006404000039000c00000005001d0000000c0500002900000020060000391c8812830000040f0000000c05000029000000000001004b00000a200000613d00000001010000310000001f02100039000000200300008a000000000332016f00000000025300190001085e0000003d00001b710000013d000007670020009c000008e40000213d0000000100300190000008e40000c13d000000400020043f000007680010009c0000000f03000029000010e80000213d000000200010008c000010e80000413d00000000010504330001086b0000003d00001b550000013d000010e80000c13d000000000001004b000008720000c13d00000c930000013d000000050000006b0000000f03000029000004cc0000c13d00000002010003670000000d02100360000000000202043b000d00000002001d000007600020009c000010e80000213d000000000131034f000000000101043b000000000010043f000000cc010000390001087e0000003d00001bb50000013d00000769022001970000000d022001af000000000021041b0000000e0200002900000001022000390000081c0000013d000000440080008c000010e80000413d000000000003004b000010e80000c13d0000000001000415000f00000001001d1c8814660000040f00000002010003670000002402100370000000000302043b0000ffff0030008c000010e80000213d000027100030008c00000b070000a13d000007920100004100000c940000013d000000240080008c000010e80000413d000000000003004b000010e80000c13d000000000102043b000f00000001001d000007600010009c000010e80000213d1c8813b60000040f0000000f01000029000000000001004b00000b660000c13d000000400100043d00000064021000390000076303000041000000000032043500000044021000390000076403000041000000000032043500000024021000390000002603000039000000000032043500000765020000410000000000210435000000040210003900000020030000390000000000320435000008f30000013d000000000003004b000010e80000c13d00000000010800191c8813410000040f000c00000001001d000d00000002001d0000006501000039000000000101041a00000760011001970000000002000411000000000012004b0000000001020019000008bf0000c13d000108bf0000003d00001b5a0000013d0000076001100197000108c20000003d00001b4c0000013d00000a480000c13d000007c00100004100000c940000013d000000410120008a000000200200008a000000000321016f0000075f0030009c000008e40000213d0000008001300039000000400010043f00000020020000390000000000210435000000a004300039000000800200043d00000000002404350000008004000039000000c0033000390000000005000019000000000025004b00000a4f0000813d00000020044000390000000006040433000007600660019700000000036304360000000105500039000008d40000013d000001000100008a000000000112016f000000a00010043f000000c00100003900000a550000013d000007740120009a000007750010009c00000a510000813d0000079901000041000000000010043f00000041010000390000108e0000013d0000076501000041000000800010043f0000002001000039000000840010043f0000003801000039000000a40010043f000007a601000041000000c40010043f000007a701000041000000e40010043f000000800100003900000084020000391c881af90000040f0000000001000410001300000001001d00008002010000390000002403000039000b00000004001d0000000004000415000000130440008a00000005044002100000076e020000411c881b1f0000040f0000000b02000029000000ff0220018f000000010220015f00000000001201a0000000000100003900000001010060391c8814c20000040f000001000100008a000000000200041a000000000112016f00000001011001bf0000ffff0200008a000000000221016f00000100022001bf0000000e0000006b000000000201c019000000000020041b0000000f010000290000000001010433000007680010009c000010e80000213d000001800010008c000010e80000413d0000000f020000290000002002200039000b00000002001d0000000004020433000007670040009c000010e80000213d0000000b02100029000a00000002001d0000000b014000291c8814d10000040f0000000f0200002900000040022000390000000a03200069000900000001001d000007680030009c000010e80000213d000000400030008c000010e80000413d000000400100043d000800000001001d0000077c0010009c000008e40000213d00000008010000290000004001100039000000400010043f0000000001020433000007600010009c000010e80000213d000000080200002900000000011204360000000f02000029000000600220003900000000020204330000ffff0020008c000010e80000213d00000000002104350000000f0100002900000080011000390000000001010433000700000001001d000007600010009c000010e80000213d0000000f01000029000000a0011000390000000001010433000007670010009c000010e80000213d0000000b011000290000000a020000291c8814d10000040f000600000001001d0000000f01000029000000c0011000390000000001010433000007670010009c000010e80000213d0000000b011000290000000a020000291c8814d10000040f000500000001001d0000000f01000029000000e0011000390000000001010433000400000001001d000007600010009c000010e80000213d0000000f0100002900000100011000390000000001010433000300000001001d000007600010009c000010e80000213d0000000f0100002900000120011000390000000001010433000007670010009c000010e80000213d0000000b011000290000000a020000291c8814d10000040f00000000090100190000000f01000029000001600110003900000000010104330001096c0000003d00001b550000013d000010e80000c13d0000000f010000290000018001100039000000000b0104330000076000b0009c000010e80000213d0000000f010000290000014001100039000000000a0104330000000d0100002900000009020000290000000803000029000000070400002900000006050000290000000506000029000000040700002900000003080000291c88153f0000040f0000000e0000006b00000db50000c13d000109820000003d00001c700000013d000007230010009c000007230100804100000040011002100000000002000414000007230020009c0000072302008041000000c002200210000000000112019f0000077d011001c70000800d020000390000077e040000411c881b380000040f000000010020019000000db50000c13d000010e80000013d000000e0020000390000000f06000029000000000006004b000009990000613d000000010500008a00000000056500d9000000000054004b0000108b0000213d000000000013043500000000016400a9000027100110011a0000000000120435000000400200003900000a5b0000013d000000400300043d000007a90100004100000000001304350000000401300039000007b50200004100000000002104350000000001000414000000040050008c000a00000005001d000009b20000613d000000240400003900000020060000390000000002050019000c00000003001d0000000c050000291c8812a70000040f0000000c03000029000000000001004b00000a200000613d0000000101000031000c00000001001d0000001f01100039000000200200008a00090000002101730000000902300029000000090020006c00000000010000390000000101004039000b00000002001d000007670020009c000008e40000213d0000000100100190000008e40000c13d0000000b01000029000000400010043f0000000c01000029000007680010009c000010e80000213d0000000c01000029000000200010008c000010e80000413d0000000001030433000109cb0000003d00001b550000013d000010e80000c13d000000000001004b0000000e02000029000004c70000613d0000000f01000029000000000010043f000109d30000003d00001b640000013d000800000001001d0020000a0000002d000109d70000003d00001b600000013d000000200440008a00000005044002100000076e020000411c881b1f0000040f000000000001004b000010e80000613d0000000b0300002900000044013000390000000f0200002900000000002104350000000801000029000007600110019700000024023000390000000000120435000007b50100004100000000001304350000000e0100002900000760011001970000000402300039000000000012043500000000010004140000000a02000029000000040020008c0000106e0000613d00000064040000390000000a020000290000000b03000029000000000503001900000000060000191c8812830000040f000c000100000035000000000001004b0000106a0000c13d0000000303000367000000200100008a0000000c0200002900000000041201700000001f0520018f000000400100043d000000000241001900000a060000613d000000000603034f0000000007010019000000006806043c0000000007870436000000000027004b00000a020000c13d000000000005004b00000a0c0000613d000000000343034f000000030450021000010a0c0000003d00001bd30000013d0000000c020000291c881af90000040f000000400500043d0000078b0300004100000000003504350000076001100197000000040350003900000000001304350000000001000414000000040020008c00000a5d0000613d000000240400003900000020060000390000000003050019000e00000005001d0000000e050000291c8812a70000040f0000000e05000029000000000001004b00000a5d0000c13d00000003040003670000000102000031000000200100008a00010a250000003d00001b8a0000013d00000a2b0000613d000000000704034f000000000801001900010a2a0000003d00001c330000013d00000a280000c13d000000000006004b00000a310000613d000000000454034f000000030560021000010a310000003d00001b420000013d1c881af90000040f0000013501000039000000200010043f00000040020000390000000001000019000e00000002001d1c881b0b0000040f000000000201041a000000010320019000000001052002700000007f0550618f0000001f0050008c00000000040000390000000104002039000000000442013f000000010040019000000b680000613d0000079901000041000000000010043f00000022010000390000108e0000013d000007760100004100000c940000013d00010a4a0000003d00001c460000013d0000076101100197000007620010009c00000b7a0000c13d000007bf0100004100000c940000013d000000000213004900000e9f0000013d0000003f01200039000000200200008a000000000121016f0000008001100039000000400010043f0000008002000039000f00000001001d1c8812df0000040f0000000f030000290000000002310049000000000103001900000e9f0000013d00010a5f0000003d00001bdd0000013d000007670030009c000008e40000213d0000000100200190000008e40000c13d0000000e02000029000000400020043f000007680010009c000010e80000213d000000200010008c000010e80000413d000000000105043300010a6c0000003d00001b550000013d000010e80000c13d000000000001004b00000ef00000c13d000007b20100004100000c940000013d0000006502000039000000000202041a00000760032001970000000002000411000000000032004b00000c2e0000c13d000000140200008a00000000022000310000000202200367000000000202043b000000600220027000000c2f0000013d000000000010043f000001060100003900010a810000003d00001b8f0000013d000000db0000013d000000c904000039000000000204041a000000000002004b0000108b0000613d000000010130008a000000000032004b00000db80000c13d0000079b0230009a000000000002041b000000000014041b0000000f01000029000000000010043f000000ca0100003900010a910000003d00001b8f0000013d000000000001041b000000400100043d000007230010009c000d00000001001d000007230100804100000040011002100000000002000414000007230020009c0000072302008041000000c002200210000000000112019f0000076c011001c70000800d0200003900000003030000390000079c040000410000000f0500002900000000060000191c881b380000040f0000000100200190000010e80000613d00010aa70000003d00001c5c0000013d001807600010019b00010aaa0000003d00001b600000013d000000180440008a00000005044002100000076e020000411c881b1f0000040f000000000001004b00000f390000c13d000000000100001900000000020000191c881af90000040f0000006501000039000000000101041a00000760011001970000000003000411000000000013004b00000abe0000c13d000000140100008a00000000011000310000000201100367000000000101043b0000006003100270000000cb01000039000000000101041a000007600210019800000c640000c13d0000003302000039000000000202041a000000000232013f0000076000200198000004cc0000c13d000000400300043d00000769011001970000000f05000029000000000151019f000000cb02000039000000000012041b000007230030009c000d00000003001d0000072301000041000000000103401900000040011002100000000002000414000007230020009c0000072302008041000000c002200210000000000112019f0000076c011001c70000800d02000039000000020300003900000794040000411c881b380000040f0000000100200190000010e80000613d00010ae00000003d00001c5c0000013d001707600010019b00010ae30000003d00001b600000013d000000170440008a00000005044002100000076e020000411c881b1f0000040f000000000001004b000010e80000613d0000000c01000029000007600210019700000795010000410000000d03000029000000000013043500000004013000390000000f0300002900000000003104350000000001000414000000040020008c00000af90000613d00000024040000390000000d0300002900010af80000003d00001b7f0000013d00000a200000613d00010afb0000003d00001c7c0000013d00010afd0000003d00001b710000013d000007670020009c000008e40000213d0000000100300190000008e40000c13d000000400020043f000007680010009c000010e80000213d00010b060000003d00001c370000013d0000034e0000013d0000006502000039000000000202041a00000760022001970000000004000411000000000024004b00000b120000c13d000000140200008a0000000002200031000000000221034f000000000202043b0000006004200270000000cf02000039000000000202041a000007600220019800000c970000c13d0000003302000039000000000202041a000000000242013f0000076000200198000004cc0000c13d0000000402100370000000000502043b000007600050009c000010e80000213d000000cd02000039000000000302041a0000076904300197000000000454019f000000000042041b0000002401100370000000000601043b0000ffff0060008c000010e80000213d0000076a01300197000000a0036002100000076b03300197000000000131019f000000000151019f000000000012041b000000400100043d000007230010009c000e00000001001d000007230100804100000040011002100000000002000414000007230020009c0000072302008041000000c002200210000000000112019f0000076c011001c70000800d0200003900000003030000390000076d040000411c881b380000040f0000000100200190000010e80000613d000000d001000039000000000101041a00000002020003670000000403200370000000000303043b000d00000003001d000007600030009c000010e80000213d0000002402200370000000000202043b000c00000002001d0000ffff0020008c000010e80000213d000b07600010019b0010000b0000002d00010b500000003d00001b600000013d000000100440008a00000005044002100000076e020000411c881b1f0000040f000000000001004b000010e80000613d0000000e0200002900000024012000390000000c0300002900000000003104350000076f01000041000000000012043500000004012000390000000d02000029000000000021043500000000010004140000000b02000029000000040020008c00000f200000613d00000044040000390000000b0200002900000f1c0000013d1c8813d40000040f00000e9d0000013d000000800050043f000000000003004b00000cc90000613d000000000010043f00000020020000390000000001000019000d00000005001d1c881b0b0000040f0000000d050000290000000002000019000000000052004b00000cce0000813d000000000301041a000000a00420003900000000003404350000002002200039000000010110003900000b720000013d0000016d01000039000000000101041a000f00000001001d00000000020000190000000d0020006c00000bf80000813d000000010100008a0000000f03000029000000000013004b0000108b0000613d0000000501200210000e00000002001d0000000c011000290000000201100367000000000101043b000007600010009c000010e80000213d0000000102300039000f00000002001d1c8819690000040f0000000e02000029000000010220003900000b7e0000013d000000400100043d000000200200003900000000022104360000000c030000290000000000320435000007af0030009c000010e80000213d00000040031000390000000c02000029000f00050020021a0000000f0230002900000ba30000613d0000000b040000290000000204400367000000004504043c0000000003530436000000000023004b00000b9f0000c13d0000000103000039000000000003004b00000000000204350000000002120049000007230020009c00000723020080410000006002200210000007230010009c00000723010080410000004001100210000000000112019f0000000002000414000007230020009c0000072302008041000000c002200210000000000112019f0000076c011001c70000800d02000039000007b0040000411c881b380000040f0000000100200190000010e80000613d000000d001000039000000000101041a000e00000001001d001e07600010019b00010bbf0000003d00001b600000013d0000001e0440008a00000005044002100000076e020000411c881b1f0000040f000000000001004b000010e80000613d000000400300043d00000024013000390000000c020000290000000000210435000007b1010000410000000000130435000000040130003900000020020000390000000000210435000d00000003001d00000044013000390000000f031000290000000f0000006b00000bd90000613d0000000b020000290000000202200367000000002402043c0000000001410436000000000031004b00000bd50000c13d0000000e010000290000076002100197000000010000008b00000000000304350000000001000414000000040020008c00000be70000613d0000000d050000290000000004530049000000000305001900000000060000191c8812830000040f000000000001004b00000a200000613d00010be90000003d00001c7c0000013d00010beb0000003d00001b710000013d000007670020009c000008e40000213d0000000100300190000008e40000c13d000000400020043f000007680010009c000010e80000213d00000097010000390000000102000039000000000021041b000000000100041500000009020000290000061f0000013d0000016e01000039000000000101041a000000000001004b0000000f0200002900000d510000613d000000000021004b00000d510000813d000007bd0100004100000c940000013d001a000f0000002d00010c040000003d00001b600000013d0000001a0440008a00000005044002100000076e020000411c881b1f0000040f000000000001004b00000dcf0000c13d0000079f0100004100000c940000013d000000000010043f0000010801000039000000200010043f00000040020000390000000001000019000d00000004001d1c881b0b0000040f0000000f02000029000000000020043f00010c170000003d00001c0a0000013d000001000200008a000000000301041a000000000223016f0000000e03000029000000000232019f000000000021041b000000400100043d0000000000310435000007230010009c000007230100804100000040011002100000000002000414000007230020009c0000072302008041000000c002200210000000000112019f0000077d011001c70000800d02000039000000030300003900000796040000410000000d050000290000000f06000029000006030000013d0000076002200197000000000012004b00000c460000613d000d00000002001d000000000010043f000001080100003900010c360000003d00001b8f0000013d0000000d02000029000000000020043f00010c3a0000003d00001bbb0000013d000000ff0010019000000c460000c13d000000400100043d0000006402100039000007c10300004100000000003204350000004402100039000007c203000041000000000032043500000024021000390000003d03000039000008a90000013d0000000e01000029000000000010043f000001070100003900010c4b0000003d00001b840000013d00000769022001970000000f022001af000000000021041b0000000e01000029000000000010043f000001050100003900010c530000003d00001bb50000013d000000400100043d000007230010009c000007230100804100000040011002100000000003000414000007230030009c0000072303008041000000c003300210000000000113019f0000076c011001c700000760052001970000800d020000390000000403000039000007c3040000410000000f060000290000000e07000029000006030000013d000000400500043d00000044015000390000000f04000029000000000041043500000772010000410000000000150435000007600130019700000004035000390000000000130435000000240150003900000000000104350000000001000414000000040020008c00000c7b0000613d000000640400003900000020060000390000000003050019000d00000005001d0000000d050000291c8812830000040f0000000d05000029000000000001004b00000a200000613d00000001010000310000001f02100039000000200300008a000000000232016f0000000004520019000000000024004b00000000020000390000000102004039000007670040009c000008e40000213d0000000100200190000008e40000c13d0000000003040019000000400040043f000007680010009c000010e80000213d000000200010008c000010e80000413d000000000105043300010c900000003d00001b550000013d000010e80000c13d000000000001004b000010670000c13d000007ad01000041000000000010043f000007770100004100001c8a00010430000000400600043d0000076605000041000e00000006001d00000000005604350000000401100370000000000101043b000007600010009c000010e80000213d00000760044001970000000e070000290000004405700039000000000045043500000024047000390000000000340435000000040370003900000000001304350000000001000414000000040020008c00000cb20000613d00000064040000390000002006000039000000000307001900000000050700191c8812a70000040f0000000e07000029000000000001004b00000a200000613d00010cb40000003d00001c3b0000013d000000000273001900010cb70000003d00001b710000013d000007670020009c000008e40000213d0000000100300190000008e40000c13d000000400020043f000007680010009c000010e80000213d000000200010008c000010e80000413d0000000e01000029000000000101043300010cc40000003d00001b550000013d000010e80000c13d000000000001004b0000110c0000c13d0000078f0100004100000c940000013d000001000100008a000000000112016f000000a00010043f000000c00300003900000cd60000013d000007740120009a000007750010009c000008e40000413d0000003f01200039000000200200008a000000000121016f000000800500043d0000008003100039000000400030043f000000000005004b00000080020000390000000f0100002900000fdc0000c13d1c881ae80000040f000000400100043d000c00000001001d1c88135a0000040f0000000c03000029000000000131004900010ce30000003d00001c680000013d000d00000002001d000007670020009c000008e40000213d0000000100100190000008e40000c13d00000000070300190000000d02000029000000400020043f0000000001030433000000000001004b00000f330000c13d000007890020009c000008e40000213d0000000d020000290000002001200039000000400010043f0000000000020435000000400300043d00000fdc0000013d000000400100043d000f00000001001d00000007020000290000000a03000029000000080400002900000009050000291c8814430000040f0000000f020000290000000001210049000007230010009c00000723010080410000006001100210000007230020009c00000723020080410000004002200210000000000121019f0000000002000414000007230020009c0000072302008041000000c002200210000000000121019f0000076c011001c70000800d02000039000000010300003900000770040000411c881b380000040f0000000100200190000010e80000613d000000d001000039000000000101041a000f00000001001d001107600010019b00010d180000003d00001b600000013d000000110440008a00000005044002100000076e020000411c881b1f0000040f000000000001004b000010e80000613d0000000f01000029000e07600010019b000000400200043d00000771010000410000000000120435000f00000002001d000000040120003900000007020000290000000a03000029000000080400002900000009050000291c8814430000040f000000000401001900000000010004140000000e03000029000000040030008c00000d350000613d00000000020300190000000f03000029000000000434004900010d340000003d00001b7f0000013d00000a200000613d00010d370000003d00001c820000013d00010d390000003d00001b710000013d000007670020009c000008e40000213d0000000100300190000008e40000c13d000000400020043f000007680010009c000010e80000213d00000097010000390000000102000039000000000021041b000000000100041500000004020000290000061f0000013d0000000304000367000000200100008a00010d4a0000003d00001b8a0000013d00000a2b0000613d000000000704034f000000000801001900010d4f0000003d00001c330000013d00000d4d0000c13d00000a2b0000013d0000016d01000039000000000021041b00000e9b0000013d00000000010004110000006502000039000000000202041a0000076004200197000000cb02000039000000000202041a0000076002200198000b001f00600193000a001f0060003d00000ea10000c13d000000000041004b00000d620000c13d00010d620000003d00001b5a0000013d0000003302000039000000000202041a000000000112013f0000076000100198000004cc0000c13d00000000030000310000000d010000290000000f020000291c8812f00000040f1c8817ee0000040f000000d001000039000000000101041a000d00000001001d001b07600010019b00010d720000003d00001b600000013d0000001b0440008a00000005044002100000076e020000411c881b1f0000040f000000000001004b000010e80000613d000000400300043d000007a301000041000000000013043500000004013000390000002002000039000000000021043500000024013000390000000f020000290000000000210435000000200100008a0000000004120170000900000003001d000000440130003900000000034100190000000e020000290000002002200039000000020520036700000d900000613d000000000205034f0000000006010019000000002702043c0000000006760436000000000036004b00000d8c0000c13d0000000d0200002900000760022001970000000b0000006b00000d990000613d000000000445034f0000000b05000029000000030550021000010d990000003d00001b420000013d0000000f0110002900000000000104350000000001000414000000040020008c00000da50000613d000000200300008a0000000a0330017f0000004404300039000000090300002900010da40000003d00001b7f0000013d00000a200000613d00000001010000310000001f02100039000000200300008a000000000332016f000000090230002900010dac0000003d00001b710000013d000007670020009c000008e40000213d0000000100300190000008e40000c13d000000400020043f000007680010009c000010e80000213d00010db50000003d00001c370000013d00000000010004150000000c020000290000061f0000013d000000000012004b000010500000a13d0000079b0120009a000000000101041a0000079b0230009a000000000012041b000000000010043f000000ca01000039000000200010043f00000040020000390000000001000019000d00000003001d1c881b0b0000040f0000000d02000029000000000021041b000000c901000039000000000101041a000000000001004b00000f570000c13d0000079901000041000000000010043f00000031010000390000108e0000013d0019000f0000002d00010dd20000003d00001b600000013d000000190440008a00000005044002100000076e020000411c881b1f0000040f000000000001004b000010e80000613d000000400300043d00000024013000390000000e0200002900000000002104350000079e01000041000000000013043500000000010004100000076001100197000e00000003001d0000000402300039000000000012043500000000010004140000000f02000029000000040020008c00000ded0000613d00000044040000390000000f020000290000000e0300002900010dec0000003d00001b7f0000013d00000a200000613d00000001010000310000001f02100039000000200300008a000000000332016f0000000e0230002900010df40000003d00001b710000013d000007670020009c000008e40000213d0000000100300190000008e40000c13d000000400020043f000007680010009c0000061d0000a13d000010e80000013d000000400100043d000f00000001001d000000070200002900000008030000290000000c0400002900000000050300191c8816ad0000040f0000000f020000290000000001210049000007230010009c00000723010080410000006001100210000007230020009c00000723020080410000004002200210000000000121019f0000000002000414000007230020009c0000072302008041000000c002200210000000000121019f0000076c011001c70000800d02000039000000010300003900000778040000411c881b380000040f0000000100200190000010e80000613d000000d001000039000000000101041a000f00000001001d001207600010019b00010e1e0000003d00001b600000013d000000120440008a00000005044002100000076e020000411c881b1f0000040f000000000001004b000010e80000613d0000000f01000029000e07600010019b000000400200043d00000779010000410000000000120435000f00000002001d0000000401200039000000070200002900000008030000290000000c0400002900000000050300191c8816ad0000040f000000000401001900000000010004140000000e03000029000000040030008c00000e3b0000613d00000000020300190000000f03000029000000000434004900010e3a0000003d00001b7f0000013d00000a200000613d00010e3d0000003d00001c820000013d00010e3f0000003d00001b710000013d000007670020009c000008e40000213d0000000100300190000008e40000c13d000000400020043f000007680010009c000010e80000213d00000097010000390000000102000039000000000021041b000000000100041500000005020000290000061f0000013d00010e4e0000003d00001b960000013d0000068a0000613d000000000100043d0000000102000031000000200020008c0000068a0000413d000000000001004b0000068a0000613d0000006501000039000000000101041a00000760011001970000000004000411000000000014004b00000e600000c13d000000140100008a00000000011000310000000201100367000000000101043b0000006004100270000000cf01000039000000000101041a000007600310019800000f5b0000c13d0000003302000039000000000202041a000000000242013f0000076000200198000004cc0000c13d000000400200043d000d00000002001d00000769011001970000000f05000029000000000151019f000000cf02000039000000000012041b0000000d01000029000007230010009c000007230100804100000040011002100000000002000414000007230020009c0000072302008041000000c002200210000000000112019f0000076c011001c70000800d0200003900000002030000390000078c040000411c881b380000040f0000000100200190000010e80000613d00010e820000003d00001c5c0000013d001d07600010019b00010e850000003d00001b600000013d0000001d0440008a00000005044002100000076e020000411c881b1f0000040f000000000001004b000010e80000613d0000000c0100002900000760021001970000078d010000410000000d03000029000000000013043500000004013000390000000f030000290000000000310435000000000100041400000af20000013d0000000f010000291c8819690000040f0000000d0100002900000001011000390000016d02000039000000000012041b00010e9d0000003d00001c370000013d000000400100043d000000000200001900000000030000191c881b010000040f0000000203000367000000000041004b00000ea90000c13d000000140100008a0000000001100031000000000113034f000000000101043b0000006001100270000000400700043d0000004404700039000000600500003900000000005404350000077b0400004100000000004704350000076001100197000000040470003900000000001404350000000d043003600000006401700039000000000061043500000024017000390000000000010435000000200100008a0000000005160170000900000007001d0000008401700039000000000351001900000ec30000613d000000000604034f0000000007010019000000006806043c0000000007870436000000000037004b00000ebf0000c13d0000000b0000006b00000eca0000613d000000000454034f0000000b05000029000000030550021000010eca0000003d00001b420000013d0000000f0110002900000000000104350000000001000414000000040020008c00000ed80000613d000000200300008a0000000a0330017f00000084043000390000002006000039000000090300002900000000050300191c8812830000040f000000000001004b00000a200000613d00000001010000310000001f02100039000000200300008a000000000332016f000000090230002900010edf0000003d00001b710000013d000007670020009c000008e40000213d0000000100300190000008e40000c13d000000400020043f000007680010009c000010e80000213d000000200010008c000010e80000413d0000000901000029000000000101043300010eec0000003d00001b550000013d000010e80000c13d000000000001004b000004cc0000613d00000d670000013d000000cf02000039000000000102041a0000076901100197000000000012041b0000000e01000029000007230010009c000007230100804100000040011002100000000002000414000007230020009c0000072302008041000000c002200210000000000112019f0000076c011001c70000800d0200003900000002030000390000078c0400004100000000050000191c881b380000040f0000000100200190000010e80000613d000000d001000039000000000101041a000d00000001001d001407600010019b00010f0b0000003d00001b600000013d000000140440008a00000005044002100000076e020000411c881b1f0000040f000000000001004b000010e80000613d0000000d0100002900000760021001970000078d010000410000000e030000290000000000130435000000040130003900000000000104350000000001000414000000040020008c00000f200000613d00000024040000390000000e0300002900010f1f0000003d00001b7f0000013d00000a200000613d00000001010000310000001f02100039000000200300008a000000000332016f0000000e0230002900010f270000003d00001b710000013d000007670020009c000008e40000213d0000000100300190000008e40000c13d000000400020043f000007680010009c000010e80000213d00010f300000003d00001c370000013d00000000010004150000000f020000290000061f0000013d0000000f010000290000077f0010009c00000f7c0000413d0000000f010000290000077f0110012a00000f880000013d0000000c0100002900000760021001970000079d010000410000000d04000029000000000014043500000004014000390000000f030000290000000000310435000000240140003900000000000104350000000001000414000000040020008c00000f4b0000613d00000044040000390000000d0300002900010f4a0000003d00001b7f0000013d00000a200000613d00010f4d0000003d00001c7c0000013d00010f4f0000003d00001b710000013d000007670020009c000008e40000213d0000000100300190000008e40000c13d000000400020043f000007680010009c0000034e0000a13d00000ab00000013d0000000003010019000000010110008a000000c90400003900000a890000013d0000076001400197000000400500043d00000024045000390000000000140435000007ac010000410000000000150435000c00000005001d00000004015000390000000f0400002900000000004104350000000001000414000000040030008c00000fdf0000613d0000004404000039000000200600003900000000020300190000000c0300002900000000050300191c8812a70000040f0000000102000031000000000001004b00000fdf0000c13d0000000304000367000000200100008a00010f750000003d00001b8a0000013d00000a2b0000613d000000000704034f000000000801001900010f7a0000003d00001c330000013d00000f780000c13d00000a2b0000013d0000000f01000029000007810010009c000007800110212a00000000020000390000002002002039000e00000002001d000007820010009c0000000e0200802900000010022081bf000e00000002801d0000078301108197000007820110812a000007840010009c0000000e020080290000000802208039000e00000002801d0000076701108197000007840110812a000027100010008c0000000e020080290000000402208039000e00000002801d0000072301108197000027100110811a000000640010008c0000000e020080290000000202208039000e00000002801d0000ffff0110818f000000640110811a000000090010008c0000000e020000290000000102202039000000200300008a0000000006020019000000000232016f0000005f01200039000000000131016f0000000d01100029000007670010009c000008e40000213d000000400010043f00000001016000390000000d0300002900000000011304360000002002200039000000200300008a00000000033201700000001f0220018f00000fb50000613d000000000331001900000000040000310000000204400367000000004504043c0000000001510436000000000031004b00000fb10000c13d000000000002004b0000000d0160002900000021011000390000000f04000029000000090040008c0000000a2440011a0000000302200210000000010110008a00000000030104330000078503300197000007860220021f0000078702200197000000000232019f000000000021043500000fb90000213d000000400100043d000f00000001001d000000200210003900000000010700191c88186a0000040f0000078802000041000000000021043500000001021000390000000d010000291c88186a0000040f0000000f030000290000000001310049000000200210008a000000000023043500010fd40000003d00001c680000013d000007670020009c000008e40000213d0000000100100190000008e40000c13d00000000040300190000000003020019000000400020043f0000000002040019000f00000003001d000000000103001900000a580000013d0000001f01200039000000200300008a000000000131016f0000000c03100029000000000013004b00000000010000390000000101004039000d00000003001d000007670030009c000008e40000213d0000000100100190000008e40000c13d0000000d01000029000000400010043f000007680020009c000010e80000213d000000200020008c000010e80000413d0000000c01000029000000000101043300010ff50000003d00001b550000013d000010e80000c13d000000000001004b00000c930000613d000000cf01000039000000000101041a00000e6b0000013d000807600010019b00000000050000190000000d0050006c0000110e0000813d0000000b0050006c000010500000813d00000006015002100000000a0610002900000020016000390000000202000367000000000112034f000000000101043b0000ffff0010008c000010e80000213d000027110010008c000008920000813d00000005035002100000000c07300029000000000372034f000000000303043b000000400800043d0000078e04000041000000000048043500000004048000390000000000340435000000000262034f000000000202043b000007600020009c000010e80000213d000000640380003900000008040000290000000000430435000000440380003900000000001304350000002401800039000000000021043500000000010004140000000902000029000000040020008c000f00000005001d000e00000006001d000010310000613d00000009020000290000000003080019000000840400003900000000050800190000002006000039000700000007001d000600000008001d1c8812a70000040f00000006080000290000000707000029000000000001004b00000a200000613d000110330000003d00001c3b0000013d0000000002830019000110360000003d00001b710000013d000007670020009c000008e40000213d0000000100300190000008e40000c13d000000400020043f000007680010009c000010e80000213d000000200010008c000010e80000413d0000000001080433000110420000003d00001b550000013d000010e80000c13d000000000001004b00000cc70000613d0000000201700367000000000101043b000000000010043f000000ce010000390001104b0000003d00001c0a0000013d0000000e020000291c8814760000040f0000000f05000029000000010550003900000ffd0000013d0000079901000041000000000010043f00000032010000390000108e0000013d00000000020000190000000c0020006c00000e9b0000813d000e00000002001d00000005012002100000000b011000290000000201100367000000000201043b0000000a0020006c000003aa0000213d0000000d010000291c8819690000040f0000000f01000029000000010010003a0000108b0000413d000f00010010003d0000000e020000290000000102200039000010550000013d000000cb01000039000000000101041a00000ac80000013d0000000c010000290000001f01100039000000200200008a000900000021017300000009020000290000000b01200029000000000021004b00000000020000390000000102004039000007670010009c000008e40000213d0000000100200190000008e40000c13d000000400010043f0000000c01000029000007680010009c000010e80000213d0000000f01000029000000000010043f0001107f0000003d00001b640000013d000c00000001001d0000010701000039000110830000003d00001bb50000013d0000076902200197000000000021041b0000000c010000290000076001100197000c00000001001d0001108a0000003d00001c180000013d000010910000c13d0000079901000041000000000010043f0000001101000039000000040010043f0000079a0100004100001c8a00010430000000010220008a000000000021041b0000000f01000029000000000010043f0000010501000039000110980000003d00001b840000013d0000076902200197000000000021041b000000400100043d000007230010009c000007230100804100000040011002100000000002000414000007230020009c0000072302008041000000c002200210000000000112019f0000076c011001c70000800d020000390000000403000039000007b6040000410000000c0500002900000000060000190000000f070000291c881b380000040f0000000100200190000010e80000613d0000000f01000029000000000010043f0000013501000039000000200010043f000000400200003900000000010000191c881b0b0000040f000000000301041a000000010030019000000001043002700000007f0440618f0000001f0040008c00000000020000390000000102002039000000000223013f000000010020019000000a420000c13d000c00000001001d000b00000004001d000000000004004b000010dd0000613d0000000b01000029000000200010008c000010db0000413d0000000c01000029000000000010043f000000200200003900000000010000191c881b0b0000040f0000000b020000290000001f02200039000000050220027000000000022100190000000101100039000000000021004b000010d30000813d000000000001041b000010ce0000013d0000000c01000029000000000010043f000000200200003900000000010000191c881b0b0000040f0000000c02000029000000000002041b000c00000001001d0000000c01000029000000000001041b000110df0000003d00001c5c0000013d002107600010019b000110e20000003d00001b600000013d000000210440008a00000005044002100000076e020000411c881b1f0000040f000000000001004b000010e90000c13d00001b940000013d0000000c010000290000076002100197000000400400043d00000044014000390000000f030000290000000000310435000007b70100004100000000001404350000000e01000029000007600110019700000004034000390000000000130435000f00000004001d000000240140003900000000000104350000000001000414000000040020008c000011000000613d00000064040000390000000f03000029000110ff0000003d00001b7f0000013d00000a200000613d000111020000003d00001c820000013d000111040000003d00001b710000013d000007670020009c000008e40000213d0000000100300190000008e40000c13d000000400020043f000007680010009c0000049a0000a13d000010e80000013d000000020100036700000b1b0000013d000000400100043d000f00000001001d0000000c020000290000000d030000290000000a040000290000000b050000291c88148b0000040f0000000f020000290000000001210049000007230010009c00000723010080410000006001100210000007230020009c00000723020080410000004002200210000000000121019f0000000002000414000007230020009c0000072302008041000000c002200210000000000121019f0000076c011001c70000800d02000039000000010300003900000790040000411c881b380000040f0000000100200190000010e80000613d000000d001000039000000000101041a000f00000001001d001507600010019b000111300000003d00001b600000013d000000150440008a00000005044002100000076e020000411c881b1f0000040f000000000001004b000010e80000613d0000000f01000029000e07600010019b000000400200043d00000791010000410000000000120435000f00000002001d00000004012000390000000c020000290000000d030000290000000a040000290000000b050000291c88148b0000040f0000000004010019000000000100041400000e320000013d0000016e01000039000000000101041a000000000001004b0000114b0000613d000000000031004b00000bff0000413d0000016d01000039000000000031041b00000e9b0000013d0000000f0000006b0000000003000019000011540000613d0000000e030000290000000203300367000000000303043b000111560000003d00001bcd0000013d0000115c0000013d0000000f0050006c0000115b0000813d0001115b0000003d00001c2a0000013d00000001011001bf0000010302000039000000000012041b0000010401000039000111610000003d00001b750000013d00000a420000c13d0000000d020000291c88152d0000040f0000000d02000029000000200020008c000000010120021000000003022002100000117a0000413d0000010403000039000000000030043f000000200300008a0000000d0530017f0000079803000041000000020400036700000000060000190000000c08000029000000000056004b0000000007860019000011830000813d000000000774034f000000000707043b000000000073041b00000020066000390000000103300039000011710000013d0000000d0000006b0000000003000019000011800000613d0000000c030000290000000203300367000000000303043b000111820000003d00001bcd0000013d000011880000013d0000000d0050006c000011870000813d000111870000003d00001c2a0000013d00000001011001bf0000010402000039000000000012041b0000016c010000390001118d0000003d00001b750000013d00000a420000c13d0000000b020000291c88151b0000040f0000000b01000029000000200010008c000011a40000413d0000016c01000039000000000010043f000000200100008a0000000b0310017f0000077301000041000000020200036700000000040000190000000a06000029000000000034004b0000000005640019000011b30000813d000000000552034f000000000505043b000000000051041b000000200440003900000001011000390000119b0000013d0000000b0000006b0000000001000019000011aa0000613d0000000a010000290000000201100367000000000101043b0000000b040000290000000302400210000000010300008a000000000223022f000000000232013f000000000121016f0000000102400210000000000121019f000011c20000013d0000000b0030006c000011bf0000813d0000000b030000290000000303300210000000f80330018f000000010400008a000000000334022f000000000343013f000000000252034f000000000202043b000000000232016f000000000021041b0000000b01000029000000010110021000000001011001bf0000016c02000039000000000012041b000000d001000039000000000101041a000700000001001d002307600010019b000111ca0000003d00001b600000013d000000230440008a00000005044002100000076e020000411c881b1f0000040f000000000001004b000010e80000613d000000400400043d000007bc01000041000000000014043500000064014000390000000f030000290000000000310435000000040140003900000060020000390000000000210435000000200200008a00000000052301700000001f0630018f000800000004001d0000008403400039000000000453001900000002020003670000000e07200360000011e80000613d000000000807034f0000000009030019000000008a08043c0000000009a90436000000000049004b000011e40000c13d000000000006004b000011ee0000613d000000000557034f0000000306600210000111ee0000003d00001bf10000013d0000000f05000029000000000453001900000000000404350000001f04500039000000200600008a000000000464016f000000000343001900000000041300490000000805000029000000240550003900000000004504350000000c052003600000000d04000029000000000343043600000000066401700000001f0740018f0000000004630019000012060000613d000000000805034f0000000009030019000000008a08043c0000000009a90436000000000049004b000012020000c13d000000000007004b0000120c0000613d000000000565034f00000003067002100001120c0000003d00001bf10000013d0000000d05000029000000000453001900000000000404350000001f04500039000000200500008a000000000454016f000000000343001900000000011300490000000804000029000000440440003900000000001404350000000a042003600000000b01000029000000000313043600000000055101700000001f0610018f0000000001530019000012240000613d000000000204034f0000000007030019000000002802043c0000000007870436000000000017004b000012200000c13d00000007020000290000076002200197000000000006004b000012330000613d000000000454034f0000000305600210000000000601043300000000065601cf000000000656022f000000000404043b0000010005500089000000000454022f00000000045401cf000000000464019f00000000004104350000000b0130002900000000000104350000000001000414000000040020008c000012440000613d0000000b040000290000001f04400039000000200500008a000000000454016f000000080500002900000000045400490000000004340019000000000305001900000000060000191c8812830000040f000000000001004b00000a200000613d000112460000003d00001c3b0000013d0000000802300029000112490000003d00001b710000013d000007670020009c000008e40000213d0000000100300190000008e40000c13d000000400020043f000007680010009c000010e80000213d00000bf50000013d0000000001000410001c00000001001d000112550000003d00001b600000013d0000001c0440008a00000005044002100000076e020000411c881b1f0000040f0000000202000029000000ff0220018f000000010220015f00000000001201a0000000000100003900000001010060391c8814c20000040f000001000100008a000000000200041a000000000112016f00000001011001bf000000000010041b000000040000006b0000127e0000c13d0000ffff0200008a000000000121016f00000100011001bf000000000010041b0001126d0000003d00001b9d0000013d0001126f0000003d00001c700000013d000007230010009c000007230100804100000040011002100000000002000414000007230020009c0000072302008041000000c002200210000000000112019f0000077d011001c70000800d020000390000077e040000411c881b380000040f0000000100200190000010e80000613d000012800000013d000112800000003d00001b9d0000013d000000000100041500000003020000290000061f0000013d0003000000000002000300000006001d000200000005001d000007230030009c00000723030080410000004003300210000007230040009c00000723040080410000006004400210000000000334019f000007230010009c0000072301008041000000c001100210000000000113019f1c881b380000040f000000020900002900000060031002700000072303300197000000030030006c000000030400002900000000040340190000001f0540018f000007cc064001980000000004690019000012a10000613d000000000701034f000000007807043c0000000009890436000000000049004b0000129d0000c13d000000010220018f000000000005004b000012a60000613d000112a60000003d00001bc10000013d00001c4a0000013d0003000000000002000300000006001d000200000005001d000007230030009c00000723030080410000004003300210000007230040009c00000723040080410000006004400210000000000334019f000007230010009c0000072301008041000000c001100210000000000113019f1c881b3d0000040f000000020900002900000060031002700000072303300197000000030030006c000000030400002900000000040340190000001f0540018f000007cc064001980000000004690019000012c50000613d000000000701034f000000007807043c0000000009890436000000000049004b000012c10000c13d000000010220018f000000000005004b000012ca0000613d000112ca0000003d00001bc10000013d00001c4a0000013d0000002004100039000000000301043300000000013204360000000002000019000000000032004b000012d70000813d00000000051200190000000006240019000000000606043300000000006504350000002002200039000012cf0000013d000012da0000a13d000000000213001900000000000204350000001f02300039000000200300008a000000000232016f0000000001210019000000000001042d00000000030200190000002002000039000000000221043600000000010300191c8812cb0000040f000000000001042d000007680010009c000012ef0000213d000000230010008c000012ef0000a13d00000004010000390000000201100367000000000101043b000007600010009c000012ef0000213d000000000001042d00001b940000013d000007cd0020009c000013210000813d00000000040100190000001f01200039000000200600008a000000000161016f0000003f01100039000000000561016f000000400100043d0000000005510019000000000015004b00000000070000390000000107004039000007670050009c000013210000213d0000000100700190000013210000c13d000000400050043f00000000052104360000000007420019000000000037004b000013270000213d00000000066201700000001f0720018f00000002044003670000000003650019000013110000613d000000000804034f0000000009050019000000008a08043c0000000009a90436000000000039004b0000130d0000c13d000000000007004b0000131e0000613d000000000464034f0000000306700210000000000703043300000000076701cf000000000767022f000000000404043b0000010006600089000000000464022f00000000046401cf000000000474019f000000000043043500000000022500190000000000020435000000000001042d0000079901000041000000000010043f0000004101000039000000040010043f0000079a0100004100001c8a0001043000001b940000013d0002000000000002000007680010009c000013400000213d0000000003010019000000430010008c000013400000a13d00000002010003670000000402100370000000000202043b000200000002001d000007600020009c000013400000213d0000002402100370000000000402043b000007670040009c000013400000213d0000002302400039000000000032004b000013400000813d0001133d0000003d00001c040000013d00000000020100190000000201000029000000000001042d00001b940000013d000007680010009c000013590000213d000000230010008c000013590000a13d00000002020003670000000403200370000000000303043b000007670030009c000013590000213d0000002304300039000000000014004b000013590000813d0000000404300039000000000242034f000000000202043b000007670020009c000013590000213d000000240330003900000005042002100000000004340019000000000014004b000013590000213d0000000001030019000000000001042d00001b940000013d0000013905000039000000000405041a000000010640019000000001024002700000007f0220618f0000001f0020008c00000000030000390000000103002039000000000036004b000013770000c13d0000000003210436000000000006004b000013720000613d000000000050043f000007a20400004100000000050000190000000001530019000000000025004b000013760000813d000000000604041a0000000000610435000000200550003900000001044000390000136a0000013d000001000200008a000000000224016f00000000002304350000004001100039000000000001042d0000079901000041000000000010043f0000002201000039000000040010043f0000079a0100004100001c8a00010430000007680010009c0000138b0000213d000000430010008c0000138b0000a13d00000002020003670000000401200370000000000101043b000007600010009c0000138b0000213d0000002402200370000000000202043b000007600020009c0000138b0000213d000000000001042d00001b940000013d000007680010009c000013b50000213d000000430010008c000013b50000a13d00000002030003670000000402300370000000000402043b000007670040009c000013b50000213d0000002302400039000000000012004b000013b50000813d0000000402400039000000000223034f000000000202043b000007670020009c000013b50000213d000000240540003900000005042002100000000004540019000000000014004b000013b50000213d0000002404300370000000000604043b000007670060009c000013b50000213d0000002304600039000000000014004b000013b50000813d0000000404600039000000000343034f000000000403043b000007670040009c000013b50000213d000000240360003900000005064002100000000006360019000000000016004b000013b50000213d0000000001050019000000000001042d00001b940000013d0000003301000039000000000101041a0000006502000039000000000202041a00000760032001970000000002000411000000000032004b000013c30000c13d000000140200008a00000000022000310000000202200367000000000202043b0000006002200270000000000112013f0000076000100198000013c70000c13d000000000001042d000000400100043d0000004402100039000007ce030000410000000000320435000007650200004100000000002104350000002402100039000000200300003900000000003204350000000402100039000000000032043500000064020000391c881af90000040f000000000601001900000760011001970000003302000039000000000302041a0000076904300197000000000114019f000000000012041b000000400100043d000007230010009c000007230100804100000040011002100000000002000414000007230020009c0000072302008041000000c002200210000000000112019f00000760053001970000076c011001c70000800d020000390000000303000039000007a0040000411c881b380000040f0000000100200190000013ed0000613d000000000001042d00001b940000013d00020000000000020000000001000412000200000001001d000100000000003d000080050100003900000044030000390000000004000415000000020440008a0000000504400210000007a5020000411c881b1f0000040f0000076502000041000000400300043d00000000002304350000000402300039000000200400003900000000004204350000004402300039000000240430003900000760011001970000000005000410000000000015004b0000140c0000c13d0000002c010000390000000000140435000007cf0100004100000000001204350000006401300039000007d2020000410000141d0000013d000007a805000041000000000505041a0000076005500197000000000015004b000014170000c13d0000000f010000390000000000140435000007d101000041000000000012043500000064020000390000141f0000013d0000002c010000390000000000140435000007cf0100004100000000001204350000006401300039000007d0020000410000000000210435000000840200003900000000010300191c881af90000040f0003000000000002000300000001001d000114250000003d00001b4c0000013d0000000001000019000014280000613d000000000001042d000000ca050000390000000304000029000000c901000039000000000201041a000007670020009c0000143d0000213d0000000103200039000000000031041b000007d30220009a000000000042041b000000000101041a000200000001001d000000000040043f000000200050043f000000400200003900000000010000191c881b0b0000040f0000000202000029000000000021041b0000000101000039000000000001042d0000079901000041000000000010043f0000004101000039000000040010043f0000079a0100004100001c8a0001043000000000060100190000004001000039000000000816043600000040016000390000000000310435000007d40030009c000014650000813d0000000207000367000000600960003900000005013002120000000001190019000014540000613d000000000227034f000000002302043c0000000009390436000000000019004b000014500000c13d000000010000008b00000000026100490000000000280435000000000051043500000000020000190000002001100039000000000052004b000014640000813d000000000347034f000000000303043b000007600030009c000014650000213d000000000031043500000001022000390000002004400039000014590000013d000000000001042d00001b940000013d0000009701000039000000000201041a000000020020008c0000146d0000613d0000000202000039000000000021041b000000000001042d000000400100043d0000004402100039000007d503000041000000000032043500000024021000390000001f030000390000000000320435000007650200004100001b6b0000013d0000000203000367000000000423034f000000000404043b000007620040009c0000148a0000813d000000000501041a0000076905500197000000000445019f000000000041041b0000002002200039000000000223034f000000000202043b0000ffff0020008c0000148a0000213d000007d603400197000000a0022002100000076b02200197000000000232019f000000000021041b000000000001042d00001b940000013d0000004006000039000000000761043600000040061000390000000000360435000007d40030009c000014b20000813d00000002060003670000006008100039000000050330021200000000033800190000149b0000613d000000000226034f000000002902043c0000000008980436000000000038004b000014970000c13d000000010000008b0000000001130049000000000017043500000000015304360000000002000019000000000052004b000014b10000813d000000000346034f000000000303043b000007600030009c000014b20000213d00000000033104360000002007400039000000000776034f000000000707043b0000ffff0070008c000014b20000213d0000000000730435000000010220003900000040044000390000004001100039000014a00000013d000000000001042d00001b940000013d0002000000000002000000cb02000039000000000202041a000200000002001d000000000010043f000000cc01000039000114bb0000003d00001b8f0000013d00000002020000290000076002200197000000000101041a0000076001100198000000000201c0190000000001020019000000000001042d000000000001004b000014c50000613d000000000001042d000000400100043d0000006402100039000007d70300004100000000003204350000004402100039000007d803000041000000000032043500000024021000390000002e030000390000000000320435000007650200004100001c560000013d0000001f03100039000000000023004b00000000040000190000077a040040410000077a052001970000077a03300197000000000653013f000000000053004b00000000030000190000077a030020410000077a0060009c000000000304c019000000000003004b000015080000613d0000000043010434000007cd0030009c000015020000813d0000001f01300039000000200500008a000000000151016f0000003f01100039000000000551016f000000400100043d0000000005510019000000000015004b00000000060000390000000106004039000007670050009c000015020000213d0000000100600190000015020000c13d000000400050043f00000000053104360000000006340019000000000026004b000015080000213d0000000002000019000000000032004b000014fe0000813d00000000065200190000000007240019000000000707043300000000007604350000002002200039000014f60000013d000015010000a13d00000000023500190000000000020435000000000001042d0000079901000041000000000010043f0000004101000039000000040010043f0000079a0100004100001c8a0001043000001b940000013d000000200010008c0000151a0000413d0000010303000039000000000030043f0000001f032000390000000503300270000007d90330009a000000200020008c000007bb030040410000001f011000390000000501100270000007d90110009a000000000013004b0000151a0000813d000000000003041b0000000103300039000015150000013d000000000001042d000000200010008c0000152c0000413d0000016c03000039000000000030043f0000001f032000390000000503300270000007da0330009a000000200020008c00000773030040410000001f011000390000000501100270000007da0110009a000000000013004b0000152c0000813d000000000003041b0000000103300039000015270000013d000000000001042d000000200010008c0000153e0000413d0000010403000039000000000030043f0000001f032000390000000503300270000007db0330009a000000200020008c00000798030040410000001f011000390000000501100270000007db0110009a000000000013004b0000153e0000813d000000000003041b0000000103300039000015390000013d000000000001042d000f000000000002000c0000000b001d00040000000a001d000300000009001d000500000008001d000800000007001d000a00000006001d000b00000005001d000700000004001d000e00000003001d000900000002001d000600000001001d000000000100041a0000000801100270000000ff0110018f1c88169e0000040f000000000100041a0000000801100270000000ff0110018f000d00000001001d1c88169e0000040f0000000e010000290000002001100039000200000001001d00000000010104330000ffff0110018f000027110010008c000016880000813d0000000d010000291c88169e0000040f0000000d010000291c88169e0000040f0000006501000039000000000101041a00000760021001970000000001000411000000000021004b000015670000c13d000115670000003d00001b5a0000013d1c8813d40000040f000000000100041a0000000801100270000000ff0110018f000d00000001001d1c88169e0000040f0000000d010000291c88169e0000040f000115710000003d00001c370000013d00000006010000291c8813d40000040f0000000e0100002900000000010104330000076001100198000015810000613d000000cd02000039000000000302041a0000076a0330019700000002040000290000000004040433000000a0044002100000076b04400197000000000343019f000000000113019f000000000012041b00000007010000290000076001100198000015890000613d000000cb02000039000000000302041a0000076903300197000000000113019f000000000012041b000000000100041a0000000801100270000000ff0110018f1c88169e0000040f000000080100002900000760011001970000006503000039000000000203041a0000076902200197000000000112019f000000000013041b000000000100041a0000000801100270000000ff0110018f000e00000001001d1c88169e0000040f0000000e010000291c88169e0000040f0000000b010000290000000012010434000007cd0020009c0000167d0000813d000d00000001001d0000010301000039000000000101041a000000010410019000000001011002700000007f0110618f0000001f0010008c00000000030000390000000103002039000000000034004b000016810000c13d000e00000002001d1c8815090000040f0000000e0b0000290000001f00b0008c000000200300008a0000000101b002100000000302b00210000015c00000a13d000001030a0000390000000000a0043f00000000043b016f0000002005000039000007bb0300004100000000060000190000000a080000290000000b09000029000000000046004b0000000007950019000015c60000813d000115bf0000003d00001c400000013d000015ba0000013d00000000000b004b000001030a000039000015cc0000613d0000000d030000290000000003030433000015cd0000013d0000000000b4004b000015ca0000813d000115ca0000003d00001c4e0000013d00000001011001bf000015d00000013d00000000030000190000000a08000029000115d00000003d00001bcd0000013d00000000001a041b0000000012080434000007670020009c0000167d0000213d000d00000001001d0000010401000039000115d80000003d00001be70000013d000016810000c13d000e00000002001d1c88152d0000040f0000000e0a0000290000002000a0008c0000000101a002100000000302a00210000015ee0000413d0000010409000039000000000090043f000000200300008a00000000043a016f0000002005000039000007980300004100000000060000190000000a08000029000000000046004b0000000007850019000015f40000813d000115ed0000003d00001c400000013d000015e80000013d00000000000a004b0000010409000039000015fa0000613d0000000d030000290000000003030433000015fb0000013d0000000000a4004b000015f80000813d000115f80000003d00001c4e0000013d00000001011001bf000015fd0000013d0000000003000019000115fd0000003d00001bcd0000013d000000000019041b000000050100002900000760011001971c8814210000040f00000009010000290000000012010434000007670020009c0000167d0000213d000d00000001001d0000016c01000039000116090000003d00001be70000013d000016810000c13d000e00000002001d1c88151b0000040f0000000e09000029000000200090008c000016210000413d00000020020000390000016c08000039000000000080043f000000200100008a000000000319016f000007730100004100000000040000190000000c060000290000000907000029000000000034004b0000000005720019000016280000813d0000000005050433000000000051041b000000200440003900000020022000390000000101100039000016180000013d000000000009004b0000000c060000290000016c08000039000016350000613d0000000d010000290000000001010433000016360000013d000000000093004b000016320000813d0000000302900210000000f80220018f000000010300008a000000000223022f000000000232013f0000000003050433000000000223016f000000000021041b000000010190021000000001011001bf0000163d0000013d00000000010000190000000302900210000000010300008a000000000223022f000000000232013f000000000121016f0000000102900210000000000121019f000000000018041b000f07600060019b000116410000003d00001b600000013d0000000f0440008a00000005044002100000076e020000411c881b1f0000040f000000000001004b000016870000613d0000000c010000290000076002100197000000400500043d000007dc01000041000000000015043500000000010004100000076001100197000000040350003900000000001304350000000001000414000000040020008c0000165e0000613d00000024040000390000000003050019000d00000002001d000e00000005001d0000000e0500002900000000060000191c8812830000040f0000000e050000290000000d02000029000000000001004b0000168c0000613d00000001010000310000001f04100039000000200300008a000000000334016f0000000004530019000000000034004b00000000030000390000000103004039000007670040009c0000167d0000213d00000001003001900000167d0000c13d000000400040043f000007680010009c000016870000213d000000d001000039000000000301041a0000076903300197000000000223019f000000000021041b00000003010000290000000002010433000000000002004b000016770000613d1c8817ee0000040f0000000402000029000000000002004b0000167c0000613d0000016e01000039000000000021041b000000000001042d0000079901000041000000000010043f0000004101000039000016840000013d0000079901000041000000000010043f0000002201000039000000040010043f0000079a0100004100001c8a0001043000001b940000013d0000079201000041000000000010043f000007770100004100001c8a0001043000000003040003670000000102000031000000200100008a000116910000003d00001b8a0000013d000016970000613d000000000704034f0000000008010019000116960000003d00001c330000013d000016940000c13d000000000006004b0000169d0000613d000000000454034f00000003056002100001169d0000003d00001b420000013d1c881af90000040f000000000001004b000016a10000613d000000000001042d000000400100043d0000006402100039000007dd0300004100000000003204350000004402100039000007de03000041000000000032043500000024021000390000002b030000390000000000320435000007650200004100001c560000013d0003000000000002000300000005001d0000004005000039000000000751043600000040051000390000000000350435000007d40030009c0000171a0000813d0000000206000367000000600510003900000005033002120000000003350019000016bf0000613d000000000226034f000000002802043c0000000005850436000000000035004b000016bb0000c13d000000010000008b0000000001130049000000000017043500000003010000290000000000130435000000050110021000000000011300190000002001100039000100000000003500000000024000790000001f0220008a000200000002001d0000077a08200197000000000a030019000000000b040019000000000c0000190000000300c0006c000017190000813d000000200aa000390000000002310049000000200220008a00000000002a04350000000002b6034f000000000902043b0000077a02900197000000000582013f000000000082004b00000000020000190000077a02002041000000020090006c000000000d0000190000077a0d0040410000077a0050009c00000000020dc019000000000002004b0000171a0000613d0000000009490019000000000296034f000000000d02043b0000076700d0009c0000171a0000213d0000000102d00069000000000042004b00000000050000190000077a050040410000077a0e4001970000077a02200197000000000fe2013f0000000000e2004b00000000020000190000077a020020410000077a00f0009c000000000205c019000000000002004b0000171a0000c13d0000002002900039000000000526034f0000000002d10436000000200100008a00000000091d0170000000000e920019000017030000613d000000000f05034f000000000102001900000000f70f043c00000000017104360000000000e1004b000016ff0000c13d0000001f01d00190000017100000613d000000000595034f000000030110021000000000070e043300000000071701cf000000000717022f000000000505043b0000010001100089000000000515022f00000000011501cf000000000171019f00000000001e04350000001f01d00039000000200500008a000000000151016f00000000052d001900000000000504350000000001210019000000010cc00039000000200bb00039000016cf0000013d000000000001042d00001b940000013d00020000000000020000016b02000039000000000202041a00000760022001980000174c0000613d0000076001100197000000400300043d00000024043000390000000000140435000007df01000041000000000013043500000000010004100000076001100197000000040430003900000000001404350000000001000414000000040020008c000017350000613d00000044040000390000002006000039000200000003001d00000002050000291c8812a70000040f0000000203000029000000000001004b000017580000613d00000001010000310000001f02100039000000200400008a000000000442016f0000000002340019000000000042004b00000000040000390000000104004039000007670020009c0000174e0000213d00000001004001900000174e0000c13d000000400020043f000007680010009c0000174d0000213d0000001f0010008c0000174d0000a13d0000000001030433000117490000003d00001b550000013d0000174d0000c13d000000000001004b000017540000613d000000000001042d00001b940000013d0000079901000041000000000010043f0000004101000039000000040010043f0000079a0100004100001c8a00010430000007e001000041000000000010043f000007770100004100001c8a000104300000000304000367000000200100008a00000001020000310001175d0000003d00001b8a0000013d000017630000613d000000000704034f0000000008010019000117620000003d00001c330000013d000017600000c13d000000000006004b000017690000613d000000000454034f0000000305600210000117690000003d00001b420000013d1c881af90000040f0002000000000002000200000001001d1c88179d0000040f000000000001004b000017940000613d00000002010000291c8817c50000040f000000000001004b0000000002000019000017950000c13d0000000205000029000000400100043d0000002003100039000007a90200004100000000002304350000002402100039000007e104000041000000000042043500000024020000390000000000210435000007e20010009c000017970000813d0000006002100039000000400020043f000000040050008c000017870000c13d0000000001030433000000000010043f0000178c0000013d00000000020500190001178a0000003d00001b960000013d000017940000613d000000000100043d0000000102000031000000200020008c0000000002000019000017950000413d000000000001004b0000000002000039000000010200c039000017950000013d0000000002000019000000010120018f000000000001042d0000079901000041000000000010043f0000004101000039000000040010043f0000079a0100004100001c8a0001043000010000000000020000000002010019000000400100043d0000002003100039000007a90400004100000000004304350000002405100039000000000045043500000024040000390000000000410435000007e20010009c000017bf0000813d0000006004100039000000400040043f000000040020008c000017b00000c13d0000000001030433000000000010043f000017b40000013d000117b20000003d00001b960000013d000017bc0000613d000000000100043d0000000102000031000000200020008c0000000002000019000017bd0000413d000000000001004b0000000002000039000000010200c039000017bd0000013d0000000002000019000000010120018f000000000001042d0000079901000041000000000010043f0000004101000039000000040010043f0000079a0100004100001c8a0001043000010000000000020000000002010019000000400100043d0000002003100039000007a90400004100000000004304350000002404100039000007c905000041000000000054043500000024040000390000000000410435000007e20010009c000017e80000813d0000006004100039000000400040043f000000040020008c000017d90000c13d0000000001030433000000000010043f000017dd0000013d000117db0000003d00001b960000013d000017e50000613d000000000100043d0000000102000031000000200020008c0000000002000019000017e60000413d000000000001004b0000000002000039000000010200c039000017e60000013d0000000002000019000000010120018f000000000001042d0000079901000041000000000010043f0000004101000039000000040010043f0000079a0100004100001c8a000104300004000000000002000300000001001d000000400200043d000400000002001d00000040010000390000000001120436000200000001001d00000040012000391c88135a0000040f0000000002010019000000040120006a0000000203000029000000000013043500000003010000291c8812cb0000040f00000004020000290000000001210049000007230020009c00000723020080410000004002200210000007230010009c00000723010080410000006001100210000000000121019f0000000002000414000007230020009c0000072302008041000000c002200210000000000121019f0000076c011001c70000800d020000390000000103000039000007e3040000411c881b380000040f0000000100200190000018590000613d00000003080000290000000032080434000007cd0020009c0000185a0000813d0000013901000039000000000401041a000000010540019000000001044002700000007f0440618f0000001f0040008c00000000060000390000000106002039000000000065004b0000185e0000c13d000000200040008c000018300000413d000000000010043f0000001f052000390000000505500270000007e40550009a000000200020008c000007a2050040410000001f044000390000000504400270000007e40440009a000000000045004b000018300000813d000000000005041b00000001055000390000182b0000013d0000001f0020008c0000183e0000a13d000000000010043f000000200300008a000000000432016f0000002005000039000007a2030000410000000006000019000000000046004b0000000007850019000018420000813d0001183d0000003d00001c400000013d000018380000013d000000000002004b0000184f0000613d0000000003030433000018500000013d000000000024004b0000184c0000813d0000000304200210000000f80440018f000000010500008a000000000445022f000000000454013f0000000005070433000000000445016f000000000043041b000000010220021000000001022001bf000018570000013d00000000030000190000000304200210000000010500008a000000000445022f000000000454013f000000000343016f0000000102200210000000000223019f000000000021041b000000000001042d00001b940000013d0000079901000041000000000010043f0000004101000039000018610000013d0000079901000041000000000010043f0000002201000039000000040010043f0000079a0100004100001c8a0001043000010000000000020000076002200197000000000020043f000118690000003d00001b8f0000013d000000000001042d000000200310003900000000010104330000000004000019000000000014004b000018750000813d000000000524001900000000064300190000000006060433000000000065043500000020044000390000186d0000013d0000000001210019000018780000a13d0000000000010435000000000001042d0001000000000002000100000001001d1c881ae80000040f0000000101000029000000000010043f0000010701000039000000200010043f000000400200003900000000010000191c881b0b0000040f000000000101041a0000076001100197000000000001042d000000000001004b000018890000613d000000000001042d000000400100043d0000004402100039000007e503000041000000000032043500000024021000390000001b030000390000000000320435000007650200004100001b6b0000013d0007000000000002000300000004001d000500000003001d000600000002001d000400000001001d0000006501000039000000000101041a00000760011001970000000002000411000000000012004b000000140300008a0000000001020019000018a30000c13d00000000013000310000000201100367000000000101043b000000600110027000000005020000291c8819450000040f1c8818860000040f0000000401000029000000060200002900000005030000291c8819a80000040f000700060000002d000118ad0000003d00001b600000013d000000070440008a00000005044002100000076e020000411c881b1f0000040f000000000001004b000018f80000613d0000006501000039000000000101041a00000760011001970000000003000411000000000013004b000018be0000c13d000000140100008a00000000011000310000000201100367000000000101043b0000006003100270000000400400043d00000064014000390000008002000039000200000002001d00000000002104350000004401400039000000050200002900000000002104350000000401000029000007600110019700000024024000390000000000120435000007e6010000410000000000140435000007600130019700000004024000390000000000120435000500000004001d000000840240003900000003010000291c8812cb0000040f0000000003010019000000000100041400000006020000290000076002200197000000040020008c000018e00000613d00000005050000290000000004530049000000200600003900000000030500191c8812830000040f000000000001004b000019020000613d00000001010000310000001f02100039000000200300008a000000000232016f00000005030000290000000004320019000000000024004b00000000020000390000000102004039000007670040009c000019330000213d0000000100200190000019330000c13d000000400040043f000007680010009c000018f90000213d0000001f0010008c000018f90000a13d0000000001030433000007c800100198000018f90000c13d000007c901100197000007e60010009c000018fa0000c13d000000000001042d00001b940000013d000007650100004100000000001404350000000401400039000600000004001d1c88193b0000040f000000060210006a00000006010000291c881af90000040f00000001020000320000190e0000c13d00000060010000390000000002010433000000000002004b000019390000c13d000000400200043d000600000002001d000007650100004100000000001204350000000401200039000018fe0000013d000007670020009c000019330000213d0000001f01200039000000200300008a000000000131016f0000003f01100039000000000431016f000000400100043d0000000004410019000000000014004b00000000050000390000000105004039000007670040009c000019330000213d0000000100500190000019330000c13d000000400040043f000000000521043600000000033201700000001f0420018f000200000005001d000000000235001900000003050003670000192c0000613d000000000605034f0000000207000029000000006806043c0000000007870436000000000027004b000019280000c13d000000000004004b000019050000613d000000000335034f0000000304400210000119320000003d00001bd30000013d000019050000013d0000079901000041000000000010043f0000004101000039000000040010043f0000079a0100004100001c8a0001043000000002010000291c881af90000040f0000004002100039000007e7030000410000000000320435000000200210003900000018030000390000000000320435000000200200003900000000002104350000006001100039000000000001042d0003000000000002000300000001001d000200000002001d000000000020043f0000010501000039000000200010043f000000400200003900000000010000191c881b0b0000040f0000000302000029000307600020019b000000000101041a0000076001100197000000030010006b000019560000c13d0000000101000039000000000001042d000000000010043f0000010801000039000000200010043f000000400200003900000000010000191c881b0b0000040f0000000302000029000000000020043f000119600000003d00001bbb0000013d000000ff01100190000019630000613d000000000001042d00000002010000291c8818790000040f000000030010006c00000000010000390000000101006039000000000001042d0004000000000002000200000001001d000407600010019c000019940000613d000300000002001d000000000020043f000119710000003d00001b640000013d0000076000100198000019980000c13d0000000401000029000119760000003d00001c210000013d000019a10000613d000000000021041b0000000301000029000000000010043f00000105010000390001197d0000003d00001b840000013d000007690220019700000004022001af000000000021041b000000400100043d000007230010009c000007230100804100000040011002100000000002000414000007230020009c0000072302008041000000c002200210000000000112019f0000076c011001c70000800d020000390000000403000039000007b6040000410000000005000019000000020600002900000003070000291c881b380000040f0000000100200190000019a70000613d000000000001042d000000400100043d0000004402100039000007e9030000410000199b0000013d000000400100043d0000004402100039000007e8030000410000000000320435000000240210003900000014030000390000000000320435000007650200004100001b6b0000013d0000079901000041000000000010043f0000001101000039000000040010043f0000079a0100004100001c8a0001043000001b940000013d000b000000000002000600000002001d000800000001001d000900000003001d000000000030043f000119af0000003d00001b640000013d000000080110014f000007600010019800001a9f0000c13d0000000601000029000707600010019c00001aa60000613d0000010701000039000119b80000003d00001b840000013d0000076902200197000000000021041b00000008010000290000076001100197000500000001001d000119bf0000003d00001c180000013d00001a990000613d000000010220008a000000000021041b0000000701000029000119c50000003d00001c210000013d00001a990000613d000000000021041b0000000901000029000000000010043f0000010501000039000119cc0000003d00001b840000013d000007690220019700000007022001af000000000021041b000000400100043d000007230010009c000007230100804100000040011002100000000002000414000007230020009c0000072302008041000000c002200210000000000112019f0000076c011001c70000800d020000390000000403000039000007b6040000410000000805000029000000060600002900000009070000291c881b380000040f000000010020019000001a940000613d0000006501000039000000000101041a00000760011001970000000002000411000000000012004b000019ed0000c13d000000140100008a00000000011000310000000201100367000000000101043b0000006002100270000000080120014f0000076000100198000600000002001d000019f30000613d00000006010000291c88171b0000040f00000009010000291c8814b30000040f000000000001004b00001a620000613d0000000007010019000000400300043d000007a90100004100000000001304350000000401300039000007eb0200004100000000002104350000000001000414000000040070008c000800000007001d00001a0c0000613d0000002404000039000000200600003900000000020700190000000005030019000400000003001d1c8812a70000040f00000004030000290000000807000029000000000001004b00001aba0000613d00000001050000310000001f01500039000000200200008a000000000421016f0000000006340019000000000046004b00000000010000390000000101004039000007670060009c00001a950000213d000000010010019000001a950000c13d000000400060043f000007680050009c00001a940000213d000000200050008c00001a940000413d000000000103043300011a200000003d00001b550000013d00001a940000c13d000000000001004b00001a620000613d000b00000007001d00008002010000390000002403000039000200000004001d00000000040004150000000b0440008a00000005044002100000076e02000041000300000005001d000400000006001d1c881b1f0000040f0000000204000029000000040300002900000008060000290000000305000029000000000001004b00001a940000613d0000008401300039000000a0020000390000000000210435000000640130003900000009020000290000000000210435000000440130003900000007020000290000000000210435000000240130003900000005020000290000000000210435000007ec0100004100000000001304350000000601000029000007600110019700000004023000390000000000120435000000a40130003900000000000104350000000001000414000000040060008c00001a570000613d000000c4040000390000000002060019000000000503001900000000060000191c8812830000040f0000000105000031000000000001004b00001acc0000613d0000001f01500039000000200200008a000000000421016f00000004030000290000000001340019000000000041004b00000000020000390000000102004039000007670010009c00001a950000213d000000010020019000001a950000c13d000000400010043f000007680050009c00001a940000213d000000d001000039000000000101041a000800000001001d000a07600010019b00011a680000003d00001b600000013d0000000a0440008a00000005044002100000076e020000411c881b1f0000040f000000000001004b00001a940000613d00000008010000290000076002100197000000400500043d000000440150003900000009030000290000000000310435000000240150003900000007030000290000000000310435000007b70100004100000000001504350000000401500039000000050300002900000000003104350000000001000414000000040020008c00001a810000c13d000000010200003100001a8a0000013d00000064040000390000000003050019000900000005001d00000000060000191c8812830000040f0000000102000031000000000001004b00001aaf0000613d000000090500002900011a8c0000003d00001c600000013d000007670010009c00001a950000213d000000010030019000001a950000c13d000000400010043f000007680020009c00001a940000213d000000000001042d00001b940000013d0000079901000041000000000010043f000000410100003900001a9c0000013d0000079901000041000000000010043f0000001101000039000000040010043f0000079a0100004100001c8a00010430000000400100043d0000004402100039000007ea0300004100000000003204350000002402100039000000160300003900001aac0000013d000000400100043d0000004402100039000007ed03000041000000000032043500000024021000390000001b030000390000000000320435000007650200004100001b6b0000013d0000000304000367000000200100008a00011ab30000003d00001b8a0000013d00001ac50000613d000000000704034f000000000801001900011ab80000003d00001c330000013d00001ab60000c13d00001ac50000013d0000000304000367000000200100008a000000010200003100011abf0000003d00001b8a0000013d00001ac50000613d000000000704034f000000000801001900011ac40000003d00001c330000013d00001ac20000c13d000000000006004b00001ae70000613d000000000454034f000000030560021000011acb0000003d00001b420000013d1c881af90000040f0000000303000367000000200100008a000000000415017000000000020500190000001f0550018f000000400100043d000000000941001900001ada0000613d000000000603034f0000000007010019000000006806043c0000000007870436000000000097004b00001ad60000c13d000000000005004b00001ae70000613d000000000343034f0000000304500210000000000509043300000000054501cf000000000545022f000000000303043b0000010004400089000000000343022f00000000034301cf000000000353019f00000000003904351c881af90000040f0001000000000002000000000010043f00011aec0000003d00001b640000013d000007600010019800001aef0000613d000000000001042d000000400100043d0000004402100039000007ee030000410000000000320435000000240210003900000018030000390000000000320435000007650200004100001b6b0000013d000000000001042f000007230010009c00000723010080410000004001100210000007230020009c00000723020080410000006002200210000000000112019f00001c8a00010430000007230010009c00000723010080410000004001100210000007230020009c00000723020080410000006002200210000000000112019f000000e002300210000000000121019f00001c890001042e000007230010009c00000723010080410000004001100210000007230020009c00000723020080410000006002200210000000000112019f0000000002000414000007230020009c0000072302008041000000c002200210000000000112019f0000076c011001c700008010020000391c881b3d0000040f000000010020019000001b1e0000613d000000000101043b000000000001042d00001b940000013d00000000050100190000000000200443000000040100003900000005024002700000000002020031000000000121043a0000002004400039000000000031004b00001b220000413d000007230030009c000007230300804100000060013002100000000002000414000007230020009c0000072302008041000000c002200210000000000112019f000007ef011001c700000000020500191c881b3d0000040f000000010020019000001b370000613d000000000101043b000000000001042d000000000001042f00001b3b002104210000000102000039000000000001042d0000000002000019000000000001042d00001b40002104230000000102000039000000000001042d0000000002000019000000000001042d000000000603043300000000065601cf000000000656022f000000000404043b0000010005500089000000000454022f00000000045401cf000000000464019f0000000000430435000000010000013b000000000010043f000000ca01000039000000200010043f000000400200003900000000010000191c881b0b0000040f000000000101041a000000000001004b000000010000013b000000000001004b0000000002000039000000010200c039000000000021004b000000010000013b000000140100008a00000000011000310000000201100367000000000101043b0000006001100270000000010000013b000080020100003900000024030000390000000004000415000000010000013b0000010501000039000000200010043f000000400200003900000000010000191c881b0b0000040f000000000101041a000000010000013b000000000021043500000004021000390000002003000039000000000032043500000064020000391c881af90000040f000000000032004b00000000030000390000000103004039000000010000013b000000000201041a000000010020019000000001012002700000007f0110618f0000001f0010008c00000000030000390000000103002039000000000232013f0000000100200190000000010000013b000000000503001900000000060000191c8812830000040f000000000001004b000000010000013b000000200010043f000000400200003900000000010000191c881b0b0000040f000000000201041a000000010000013b00000000051201700000001f0620018f000000400100043d0000000003510019000000010000013b000000200010043f000000400200003900000000010000191c881b0b0000040f000000010000013b000000000100001900001c8a0001043000000000040104330000753001000039000000200600003900000000050000191c8812a70000040f000000000001004b000000010000013b0000000f010000290000000c020000290000000e030000290000000b040000290000000a050000290000000906000029000000080700002900000007080000290000000d09000029000000060a000029000000050b0000291c88153f0000040f000000010000013b000000140280008a000000000121034f000000000101043b0000006002100270000000010000013b000000000403041a000000a005200039000000000045043500000020022000390000000103300039000000010000013b000000200010043f000000000100001900000040020000391c881b0b0000040f000000000201041a000000010000013b000000200010043f000000000100001900000040020000391c881b0b0000040f000000000101041a000000010000013b000000000661034f0000000305500210000000000704043300000000075701cf000000000757022f000000000606043b0000010005500089000000000656022f00000000055601cf000000000575019f0000000000540435000000010000013b000000010400008a000000000224022f000000000242013f000000000223016f000000000112019f000000010000013b000000000502043300000000054501cf000000000545022f000000000303043b0000010004400089000000000343022f00000000034301cf000000000353019f0000000000320435000000010000013b00000001010000310000001f02100039000000200300008a000000000232016f0000000003520019000000000023004b00000000020000390000000102004039000e00000003001d000000010000013b000000000401041a000000010040019000000001014002700000007f0110618f0000001f0010008c00000000030000390000000103002039000000000334013f0000000100300190000000010000013b000000000704043300000000076701cf000000000767022f000000000505043b0000010006600089000000000565022f00000000056501cf000000000575019f0000000000540435000000010000013b000000010320019000000001012002700000007f0110618f0000001f0010008c00000000040000390000000104002039000000000442013f0000000100400190000000010000013b0000000402400039000000000121034f000000000201043b00000024014000391c8812f00000040f000000010000013b000000200010043f000000000100001900000040020000391c881b0b0000040f000000010000013b0000001f010000390000000101100031000000200200008a000000000221016f0000000001320019000000000021004b00000000020000390000000102004039000000010000013b000000000010043f0000010601000039000000200010043f000000000100001900000040020000391c881b0b0000040f000000000201041a000000000002004b000000010000013b000000000010043f0000010601000039000000200010043f000000400200003900000000010000191c881b0b0000040f000000000201041a000000010220003a000000010000013b000000f80220018f000000010500008a000000000225022f000000000252013f000000000474034f000000000404043b000000000224016f000000000023041b000000010000013b000000007907043c0000000008980436000000000038004b000000010000013b00000001010000390000009702000039000000000012041b000000010000013b00000001010000310000001f02100039000000200300008a000000000332016f000000010000013b0000000007070433000000000073041b000000200660003900000020055000390000000103300039000000010000013b1c8814660000040f000000cf01000039000000000101041a000000010000013b000100000003001f00030000000103550000000001020019000000000001042d000000f80220018f000000010400008a000000000224022f000000000242013f0000000004070433000000000224016f000000000023041b000000010000013b000000000021043500000004021000390000002003000039000000000032043500000084020000391c881af90000040f000000d001000039000000000101041a000c00000001001d000000010000013b0000001f01200039000000200300008a000000000331016f0000000001530019000000000031004b00000000030000390000000103004039000000010000013b0000001f01100039000000200200008a000000000121016f0000000002310019000000000012004b00000000010000390000000101004039000000010000013b0000ff010100008a000000000200041a000000000112016f000000000010041b0000000103000039000000400100043d0000000000310435000000010000013b1c8814660000040f0000006501000039000000000101041a000000010000013b00000001010000310000001f02100039000000200300008a000000000332016f0000000d02300029000000010000013b00000001010000310000001f02100039000000200300008a000000000332016f0000000f02300029000000010000013b00001c880000043200001c890001042e00001c8a00010430000000000000000000000000000000000000000000000000000000000000000000000000ffffffff000000020000000000000000000000000000008000000100000000000000000000000000000000000000000000000000000000000000000000000000f97b57ec00000000000000000000000000000000000000000000000000000000047fc9aa00000000000000000000000000000000000000000000000000000000063a75020000000000000000000000000000000000000000000000000000000006fdde0300000000000000000000000000000000000000000000000000000000081812fc00000000000000000000000000000000000000000000000000000000095ea7b3000000000000000000000000000000000000000000000000000000000d640e810000000000000000000000000000000000000000000000000000000018160ddd00000000000000000000000000000000000000000000000000000000184a94d50000000000000000000000000000000000000000000000000000000018bae6c80000000000000000000000000000000000000000000000000000000023b872dd000000000000000000000000000000000000000000000000000000002a55205a000000000000000000000000000000000000000000000000000000002a5ac764000000000000000000000000000000000000000000000000000000002ddcb21f000000000000000000000000000000000000000000000000000000002fc3a8d9000000000000000000000000000000000000000000000000000000003659cfe6000000000000000000000000000000000000000000000000000000003713c1a10000000000000000000000000000000000000000000000000000000042842e0e0000000000000000000000000000000000000000000000000000000042966c680000000000000000000000000000000000000000000000000000000047230dcb000000000000000000000000000000000000000000000000000000004f1ef286000000000000000000000000000000000000000000000000000000004f58122a0000000000000000000000000000000000000000000000000000000050cf5a310000000000000000000000000000000000000000000000000000000052d1902d00000000000000000000000000000000000000000000000000000000548205410000000000000000000000000000000000000000000000000000000055f804b300000000000000000000000000000000000000000000000000000000572b6c05000000000000000000000000000000000000000000000000000000005be95448000000000000000000000000000000000000000000000000000000006352211e000000000000000000000000000000000000000000000000000000006371fbe7000000000000000000000000000000000000000000000000000000006502abea000000000000000000000000000000000000000000000000000000006c0360eb0000000000000000000000000000000000000000000000000000000070a0823100000000000000000000000000000000000000000000000000000000715018a600000000000000000000000000000000000000000000000000000000752d4cad00000000000000000000000000000000000000000000000000000000834a20df000000000000000000000000000000000000000000000000000000008da5cb5b0000000000000000000000000000000000000000000000000000000095d89b4100000000000000000000000000000000000000000000000000000000a22cb46500000000000000000000000000000000000000000000000000000000a305f5b200000000000000000000000000000000000000000000000000000000b0e5afe300000000000000000000000000000000000000000000000000000000b5c5801e00000000000000000000000000000000000000000000000000000000b88d4fde00000000000000000000000000000000000000000000000000000000bfb2995c00000000000000000000000000000000000000000000000000000000c2f5070100000000000000000000000000000000000000000000000000000000c87b56dd00000000000000000000000000000000000000000000000000000000d1f5789400000000000000000000000000000000000000000000000000000000db3e4c8400000000000000000000000000000000000000000000000000000000e8a3d48500000000000000000000000000000000000000000000000000000000e985e9c500000000000000000000000000000000000000000000000000000000eb87215000000000000000000000000000000000000000000000000000000000ee295d6200000000000000000000000000000000000000000000000000000000eeab058900000000000000000000000000000000000000000000000000000000ef60ceaf00000000000000000000000000000000000000000000000000000000f2fde38b00000000000000000000000000000000000000000000000000000000f7441c3a0000000000000000000000000000000000000000000000000000000001ffc9a766be4f155c5ef2ebd3772b228f2f00681e4ed5826cdb3b1943cc11ad15ad1d28000000000000000000000000000000000000000000000000ffffffffffffff7f000000000000000000000000ffffffffffffffffffffffffffffffffffffffff0000000000000000000000ff0000000000000000000000000000000000000000000000000000000000000001000000000000000000000000000000000000000064647265737300000000000000000000000000000000000000000000000000004f776e61626c653a206e6577206f776e657220697320746865207a65726f206108c379a00000000000000000000000000000000000000000000000000000000049a5c0f000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000ffffffffffffffff7fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff0000000000000000000000000000000000000000ffffffffffffffffffff0000000000000000000000000000000000000000000000000000000000000000ffff000000000000000000000000000000000000000002000000000000000000000000000000000000000000000000000000000000002c5ea6e4103e78cb101e796fb2dace540362fc542cbff5145eaa24af7dd8fe411806aa1896bbf26568e884a7374b41e002500962caba6a15023a8d90e8508b83d10072c000000000000000000000000000000000000000000000000000000000acc0f6922d8ccd133dbdaee1908d4d734f25a540223d55145e9fad2898471a4e15bd85bf0000000000000000000000000000000000000000000000000000000085e335b600000000000000000000000000000000000000000000000000000000e5af487827b121aa2456a019311df153887da6c98285803eea72252e4f09489b000000000000000000000000000000000000000000000000ffffffffffffff41ffffffffffffffffffffffffffffffffffffffffffffffff0000000000000080568efce20000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000040000000000000000000000002df7dd608aac5806395769062aca937bf08722cb45659d38c0cce82f5cd0cde374c1ace10000000000000000000000000000000000000000000000000000000080000000000000000000000000000000000000000000000000000000000000003d820a4d00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000ffffffffffffffbf02000000000000000000000000000000000000200000000000000000000000007f26b83ff96e1f2b6a682f133852f6798a09c465da95921460cefb38474024980000000000184f03e93ff9f4daa797ed6e38ed64bf6a1f01000000000000000000000000000000000000000000000000000004ee2d6d415b85acef810000000000000000000000000000000000000000000004ee2d6d415b85acef80ffffffff000000000000000000000000000000000000000000000000002386f26fc1000000000000000000000000000000000000ffffffffffffffffffffffffffffffff0000000000000000000000000000000000000000000000000000000005f5e10000ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff30313233343536373839616263646566000000000000000000000000000000007f000000000000000000000000000000000000000000000000000000000000002f00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000ffffffffffffffdfceea21b600000000000000000000000000000000000000000000000000000000d018c0fa00000000000000000000000000000000000000000000000000000000e236d251e7da4ac8a3c275f80aebc9b74cbf356425fa04864da5043511a81219eabab42200000000000000000000000000000000000000000000000000000000ad6e40c7000000000000000000000000000000000000000000000000000000009c39c41c00000000000000000000000000000000000000000000000000000000eaf422d1ab2d8f38856ab6be8378c08d2886463a33667e896ba3a6ebf2c6260c57c31fde00000000000000000000000000000000000000000000000000000000f6bdd07c00000000000000000000000000000000000000000000000000000000ce2003a500000000000000000000000000000000000000000000000000000000c407dfe559ca4a63d7cf8439b9fce2965d7a4c8c92eb5d5047f53bb9eec295fd16365cdd0000000000000000000000000000000000000000000000000000000017307eab39ab6107e8899845ad3d59bd9653f200f220920489ca2b5937696c314552433732313a20617070726f766520746f2063616c6c6572000000000000004c0be60200faa20559308cb7b5a1bb3255c16cb1cab91f525b5ae7a03d02fabe4e487b710000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000240000000000000000000000009941b0eaa3a10d142c88d4dd70d0ff97e1b12a7d9324c4e6bc33ee52ea52e2d973e05f4c0a27cbf60841afa4a2f8577f5e99db3e3e9c293800d6e6b1f7b8a7aba9292a6f000000000000000000000000000000000000000000000000000000007d3e3dbe0000000000000000000000000000000000000000000000000000000009ee12d5000000000000000000000000000000000000000000000000000000008be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e04552433732313a20496e76616c6964206f776e6572000000000000000000000022d66071756d4d57920b8322c1a903a27bd75b44065e925d5f9f1c08908d28f659faca9600000000000000000000000000000000000000000000000000000000ecd7b0d100000000000000000000000000000000000000000000000000000000310ab089e4439a4c15d089f94afb7896ff553aecb10793d0ab882de59d99a32e555550535570677261646561626c653a206d757374206e6f742062652063616c6c6564207468726f7567682064656c656761746563616c6c0000000000000000360894a13ba1a3210667c828492db98dca3e2076cc3735a920a3ca505d382bbc01ffc9a7000000000000000000000000000000000000000000000000000000008808898a00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000ffffffffffffff9fbcdbc94700000000000000000000000000000000000000000000000000000000e10a05050000000000000000000000000000000000000000000000000000000034e70f7a0000000000000000000000000000000000000000000000000000000007ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff4d30c2588339a383e2c3092b5383924cf202ae9c377b76c5fe5208cd41d8d2aa80d93734000000000000000000000000000000000000000000000000000000004a3891c200000000000000000000000000000000000000000000000000000000bf317c2a00000000000000000000000000000000000000000000000000000000ea6eb5c300000000000000000000000000000000000000000000000000000000129dae8b00000000000000000000000000000000000000000000000000000000ddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef23de66510000000000000000000000000000000000000000000000000000000082b4290000000000000000000000000000000000000000000000000000000000ad7735a1decc3db45c1eab757cc4a1cb4bff8e332877b55f6db4ea11d82320c62504956c0000000000000000000000000000000000000000000000000000000002c297ab74aad0aede3a1895c857b1f2c71e6a203feb727bec95ac752998cb785bf57bc300000000000000000000000000000000000000000000000000000000be1a657d00000000000000000000000000000000000000000000000000000000dd03602c0000000000000000000000000000000000000000000000000000000045a9661700000000000000000000000000000000000000000000000000000000f8d2906c000000000000000000000000000000000000000000000000000000006b656e206f776e6572206f7220617070726f76656420666f7220616c6c0000004552433732313a20617070726f76652063616c6c6572206973206e6f7420746f8c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b9254552433732313a20616c7265616479206f776e73000000000000000000000000ffffffffffffffffffffff00ffffffffffffffffffffffffffffffffffffffff9d4b38394dff663b46f53f52ced584161b5021180321b381f6104325eecebf49fd120bd20000000000000000000000000000000000000000000000000000000000000000ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff000000000000000000000000000000000000000000000000000000005b5e139f0000000000000000000000000000000000000000000000000000000080ac58cd0000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000ffffffe000000000000000000000000000000000000000000000000100000000000000004f776e61626c653a2063616c6c6572206973206e6f7420746865206f776e657246756e6374696f6e206d7573742062652063616c6c6564207468726f756768206163746976652070726f787900000000000000000000000000000000000000004e6f74207570677261646561626c65000000000000000000000000000000000064656c656761746563616c6c00000000000000000000000000000000000000009941b0eaa3a10d142c88d4dd70d0ff97e1b12a7d9324c4e6bc33ee52ea52e2d808000000000000000000000000000000000000000000000000000000000000005265656e7472616e637947756172643a207265656e7472616e742063616c6c00ffffffffffffffffffff0000ffffffffffffffffffffffffffffffffffffffff647920696e697469616c697a6564000000000000000000000000000000000000496e697469616c697a61626c653a20636f6e747261637420697320616c726561fd3d68548b552f5121c5e76a37a84e0d38e195dfc0148d84136a538ad66734881a50b787d84ede55dba95fe6cee20eac778259367d7a7fc1158ddad1b0f6b765b3f419fdff055dfaa6cf73484a5e44cdaa3e934e3546e0ada4a5185fc2fd05422445a9b6000000000000000000000000000000000000000000000000000000006e697469616c697a696e67000000000000000000000000000000000000000000496e697469616c697a61626c653a20636f6e7472616374206973206e6f742069c6171134000000000000000000000000000000000000000000000000000000008a1091930000000000000000000000000000000000000000000000000000000076413c5e00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000ffffffffffffffa02e0a5b969d96a99aee0b35787d9a60516a02ca6f528a5f66d3f936468d8f0382dd299f8e8a92b2a86df47cdd3e56fc5d8428a4bbf9a16da2a060e3f76f72d70a4552433732313a2063616c6c657220756e617574686f72697a65640000000000150b7a02000000000000000000000000000000000000000000000000000000004552433732313a20696e76616c696420726563656976657200000000000000004552433732313a20746f6b656e206d696e7465640000000000000000000000004552433732313a207a65726f20616464726573730000000000000000000000004552433732313a2066726f6d206e6f74206f776e657200000000000000000000f5e8d69e000000000000000000000000000000000000000000000000000000008af6791b000000000000000000000000000000000000000000000000000000004552433732313a20746f20746865207a65726f206164647265737300000000004552433732313a20696e76616c696420746f6b656e204944000000000000000002000002000000000000000000000000000000000000000000000000000000008f319d6fbe60af38c6be010c4582ffb17be560597cbfe065201925228b4053d9
Loading...
Loading
Loading...
Loading
Multichain Portfolio | 31 Chains
Chain | Token | Portfolio % | Price | Amount | Value |
---|
[ Download: CSV Export ]
A contract address hosts a smart contract, which is a set of code stored on the blockchain that runs when predetermined conditions are met. Learn more about addresses in our Knowledge Base.