ERC-721
Overview
Max Total Supply
10,000 abszkm
Holders
1,671
Market
Onchain Market Cap
$0.00
Circulating Supply Market Cap
-
Other Info
Token Contract
Balance
1 abszkmLoading...
Loading
Loading...
Loading
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 Source Code Verified (Exact Match)
Contract Name:
ERC721Merkle
Compiler Version
v0.8.24+commit.e11b9ed9
ZkSolc Version
v1.5.9
Optimization Enabled:
Yes with Mode 3
Other Settings:
paris EvmVersion
Contract Source Code (Solidity Standard Json-Input format)
//SPDX-License-Identifier: MIT pragma solidity ^0.8.23; import "@openzeppelin/contracts/utils/cryptography/MerkleProof.sol"; import "@openzeppelin/contracts/access/Ownable.sol"; import "./ERC721Template.sol"; contract ERC721Merkle is ERC721Template { using SafeERC20 for IERC20; struct Tier { string title; bytes32 merkleRoot; uint256 price; uint256 erc20Price; uint128 maxMintAmount; uint128 saleStartTime; mapping(address => uint256) mints; } mapping(uint256 => Tier) public tiers; uint256[] public tierIds; constructor( string memory _name, string memory _symbol, string memory _contractURI, uint256 _maxSupply, uint256 _publicPrice, string memory _defaultBaseURI, string memory _notRevealedURI, address payable _withdrawalRecipientAddress, address payable _commissionRecipientAddress, uint256 _fixedCommisionThreshold, uint256 _commissionPercentageIn10000, address payable _defaultRoyaltyRecipient, // separate from withdrawal recipient to enhance security uint256 _defaultRoyaltyPercentageIn10000 ) ERC721Template( _name, _symbol, _contractURI, _maxSupply, _publicPrice, _defaultBaseURI, _notRevealedURI, _withdrawalRecipientAddress, _commissionRecipientAddress, _fixedCommisionThreshold, _commissionPercentageIn10000, _defaultRoyaltyRecipient, _defaultRoyaltyPercentageIn10000 ) { // add code here if you want to do something specific during contract deployment } /** * @notice Get how many more the user is eligible to mint * @param tierId Id of the tier minting from * @param user Address of the user * @param proof Merkle proof * @return Amount left to mint */ function getMintEligibility(uint256 tierId, address user, bytes32[] calldata proof) external view returns (uint256) { //require(MerkleProof.verify(proof, tier.merkleRoot, keccak256(abi.encodePacked(msg.sender))), "Not in presale list for this tier"); // return 0 if user is not in the merkleRoot if (!MerkleProof.verify(proof, tiers[tierId].merkleRoot, keccak256(abi.encodePacked(user)))) { return 0; } if (tiers[tierId].mints[user] >= tiers[tierId].maxMintAmount) { return 0; } return tiers[tierId].maxMintAmount - tiers[tierId].mints[user]; } /** * @notice Get the mint's tier details * @param tierId Id of the tier to get the details for * @return merkleRoot Merkle root * @return price Price in ETH * @return maxMintAmount Max amount mintable * @return saleStartTime Mint's start timestamp * @return title Tier's title * @return ERC20Price Price in ERC20 */ function getTierDetails(uint256 tierId) external view returns (bytes32 merkleRoot, uint256 price, uint256 maxMintAmount, uint256 saleStartTime, string memory title, uint256 ERC20Price) { Tier storage tier = tiers[tierId]; uint256 requiredERC20Tokens = 0; if (ethPriceFeedAddress != address(0) && ERC20PriceFeedAddress != address(0)) { requiredERC20Tokens = getRequiredERC20TokensChainlink(publicPrice); } else { requiredERC20Tokens = tier.erc20Price; } return (tier.merkleRoot, tier.price, tier.maxMintAmount, tier.saleStartTime, tier.title, requiredERC20Tokens); } /** * @notice Get all the tier ids */ function getTierIds() external view returns (uint256[] memory) { return tierIds; } /** * @notice Mint tokens in a specific tier using ETH * @param tierId Id of the tier * @param amount Amount of tokens * @param proof Merkle proof */ function whitelistMint(uint256 tierId, uint256 amount, bytes32[] calldata proof) external payable { Tier storage tier = tiers[tierId]; checkWhitelistMintRequirements(amount, tier, proof); require(msg.value >= amount * tier.price, "Insufficient funds for mint"); tier.mints[msg.sender] += amount; _safeMint(msg.sender, amount); } /** * @notice Mint tokens in a specific tier using ERC20 and Chainlink * @param tierId Id of the tier * @param amount Amount of tokens * @param proof Merkle proof */ function whitelistMintWithERC20ChainlinkPrice(uint256 tierId, uint256 amount, bytes32[] calldata proof) external { Tier storage tier = tiers[tierId]; checkWhitelistMintRequirements(amount, tier, proof); // Let's make sure price feed contract address exists require(ERC20TokenAddress != address(0), "Payment token address not set"); // Calculate the cost in ERC20 tokens uint256 requiredTokenAmount = getRequiredERC20TokensChainlink(tier.price * amount); tier.mints[msg.sender] += amount; IERC20(ERC20TokenAddress).safeTransferFrom(msg.sender, address(this), requiredTokenAmount); _safeMint(msg.sender, amount); } /** * @notice Mint tokens in a specific tier using ERC20 and fixed price * @param tierId Id of the tier * @param amount Amount of tokens * @param proof Merkle proof */ function whitelistMintWithFixedERC20Price(uint256 tierId, uint256 amount, bytes32[] calldata proof) external { Tier storage tier = tiers[tierId]; checkWhitelistMintRequirements(amount, tier, proof); uint256 erc20Price = tier.erc20Price; require(ERC20TokenAddress != address(0), "Payment token address not set"); require(erc20Price > 0, "Price per token not set"); // Calculate the cost in ERC20 tokens uint256 requiredTokenAmount = erc20Price * amount; tier.mints[msg.sender] += amount; IERC20(ERC20TokenAddress).safeTransferFrom(msg.sender, address(this), requiredTokenAmount); _safeMint(msg.sender, amount); } /** * @notice Add or Set existing tier details * @param tierId Id of the tier * @param title Title * @param merkleRoot Merkle root * @param price Price in ETH * @param erc20Price Price in ERC20 * @param maxMintAmount Max amount mintable * @param saleStartTime Mint's start timestamp */ function setTier(uint256 tierId, string calldata title, bytes32 merkleRoot, uint256 price, uint256 erc20Price, uint256 maxMintAmount, uint256 saleStartTime) external onlyOwner { Tier storage tier = tiers[tierId]; tier.merkleRoot = merkleRoot; tier.title = title; tier.price = price; tier.erc20Price = erc20Price; tier.maxMintAmount = uint128(maxMintAmount); tier.saleStartTime = uint128(saleStartTime); // type(uint256).max; is used to disable the tier // check if tierId is already in the array bool isNewTierId = true; for (uint256 i = 0; i < tierIds.length; i++) { if (tierIds[i] == tierId) { isNewTierId = false; break; } } if (isNewTierId) { tierIds.push(tierId); } } /** * @notice Enable a tier by setting its mint's start timestamp to 0 * @param tierId Id of the tier */ function enableTier(uint256 tierId) external onlyOwner { tiers[tierId].saleStartTime = 0; } /** * @notice Disable a tier by setting its mint's start timestamp to uint128.max * @param tierId Id of the tier */ function disableTier(uint256 tierId) external onlyOwner { tiers[tierId].saleStartTime = type(uint128).max; } /** * @notice Set mint's start timestamp for a tier * @param tierId Id of the tier * @param saleStartTime Mint's start timestamp */ function setTierSaleStartTime(uint256 tierId, uint256 saleStartTime) external onlyOwner { require(tiers[tierId].merkleRoot != bytes32(0), "Tier does not exist"); tiers[tierId].saleStartTime = uint128(saleStartTime); } /** * @notice Set the prices for a tier * @param tierId Id of the tier * @param price Price in ETH * @param erc20Price Price in ERC20 */ function setTierPrice(uint256 tierId, uint256 price, uint256 erc20Price) external onlyOwner { require(tiers[tierId].merkleRoot != bytes32(0), "Tier does not exist"); tiers[tierId].price = price; tiers[tierId].erc20Price = erc20Price; } /** * @notice Set tier's max mintable amount * @param tierId Id of the tier * @param maxMintAmount Max mintable amount */ function setTierMaxMintAmount(uint256 tierId, uint256 maxMintAmount) external onlyOwner { require(tiers[tierId].merkleRoot != bytes32(0), "Tier does not exist"); tiers[tierId].maxMintAmount = uint128(maxMintAmount); } /** * @notice Set the merkle root for a tier * @param tierId Id of the tier * @param merkleRoot Merkle root */ function setTierMerkleRoot(uint256 tierId, bytes32 merkleRoot) external onlyOwner { require(tiers[tierId].merkleRoot != bytes32(0), "Tier does not exist"); tiers[tierId].merkleRoot = merkleRoot; } /** * @notice Enforce tier's mint requirements * @param _mintAmount Amount of tokens to mint * @param tier Id of the tier * @param _proof Merkle proof */ function checkWhitelistMintRequirements(uint256 _mintAmount, Tier storage tier, bytes32[] calldata _proof) internal view { bytes32 merkleRoot = tier.merkleRoot; require(merkleRoot != bytes32(0), "Tier does not exist"); require(block.timestamp >= tier.saleStartTime, "Tier sale not started"); require(MerkleProof.verify(_proof, merkleRoot, keccak256(abi.encodePacked(msg.sender))), "Not in presale list for this tier"); require(_mintAmount <= tier.maxMintAmount - tier.mints[msg.sender], "Exceeds tier max mint amount"); require(totalSupply() + _mintAmount <= maxSupply, "Exceeds max supply"); } }
//SPDX-License-Identifier: Unlicense pragma solidity ^0.8.23; import "@openzeppelin/contracts/access/Ownable.sol"; import "@openzeppelin/contracts/interfaces/IERC2981.sol"; import "@openzeppelin/contracts/utils/Strings.sol"; import "./ERC721A.sol"; import "@openzeppelin/contracts/token/ERC20/ERC20.sol"; import "@openzeppelin/contracts/token/ERC20/utils/SafeERC20.sol"; import {AggregatorV3Interface} from "@chainlink/contracts/src/v0.8/shared/interfaces/AggregatorV3Interface.sol"; contract ERC721Template is IERC2981, Ownable, ERC721A { using Strings for uint256; using SafeERC20 for ERC20; string private baseURI; string public notRevealedURI; uint256 public maxSupply; uint256 public publicPrice; uint256 public publicMaxMintAmount; uint256 public publicSaleStartTime = type(uint256).max; bool public isRevealed; address payable public immutable withdrawalRecipientAddress; // address that will receive revenue address payable public immutable commissionRecipientAddress;// address that will receive a part of revenue on withdrawal uint256 public immutable commissionPercentageIn10000; // percentage of revenue to be sent to commissionRecipientAddress uint256 private immutable fixedCommissionThreshold; uint256 private totalCommissionWithdrawn; uint256 private commissionToWithdraw; uint256 private ownerToWithdraw; uint256 immutable deployTimestamp = block.timestamp; string public contractURI; //presale price is set after // Default royalty info address payable public defaultRoyaltyRecipient; uint256 public defaultRoyaltyPercentageIn10000; // Add new state variables to handle ERC20 payments address public ERC20TokenAddress; uint256 public ERC20FixedPricePerToken; // Fixed price per token in ERC20 uint256 public ERC20DiscountIn10000; // Discount percentage for ERC20 payments address public ERC20PriceFeedAddress; uint32 private ERC20PriceFeedStaleness; uint32 private ERC20PriceFeedDecimals; address public ethPriceFeedAddress; uint32 private ethPriceFeedStaleness; uint32 private ethPriceFeedDecimals; // Per-token royalty info mapping(uint256 => address payable) public tokenRoyaltyRecipient; mapping(uint256 => uint256) public tokenRoyaltyPercentage; event ContractURIUpdated(); bool public tradingEnabled = true; mapping(address => bool) public blacklist; constructor( string memory _name, string memory _symbol, string memory _contractURI, uint256 _maxSupply, uint256 _publicPrice, string memory _defaultBaseURI, string memory _notRevealedURI, address payable _withdrawalRecipientAddress, address payable _commissionRecipientAddress, uint256 _fixedCommisionThreshold, uint256 _commissionPercentageIn10000, address payable _defaultRoyaltyRecipient, // separate from withdrawal recipient to enhance security uint256 _defaultRoyaltyPercentageIn10000 // set max mint amount after deployment ) ERC721A(_name, _symbol) Ownable(msg.sender) { maxSupply = _maxSupply; publicPrice = _publicPrice; contractURI = _contractURI; // no need to emit event here, as it's set in the constructor baseURI = _defaultBaseURI; notRevealedURI =_notRevealedURI; publicMaxMintAmount = 10000; withdrawalRecipientAddress = _withdrawalRecipientAddress; commissionRecipientAddress = _commissionRecipientAddress; fixedCommissionThreshold = _fixedCommisionThreshold; // Ensure commission percentage is between 0 and 10000 (0-100%) require(_commissionPercentageIn10000 <= 10000, "Invalid commission percentage"); commissionPercentageIn10000 = _commissionPercentageIn10000; defaultRoyaltyRecipient = _defaultRoyaltyRecipient; defaultRoyaltyPercentageIn10000 = _defaultRoyaltyPercentageIn10000; isRevealed = bytes(_notRevealedURI).length == 0; } /** * @notice Implement EIP * @param interfaceId bytes to check if EIP compatible */ function supportsInterface(bytes4 interfaceId) public view virtual override(ERC721A, IERC165) returns (bool) { return interfaceId == type(IERC2981).interfaceId || ERC721A.supportsInterface(interfaceId) || super.supportsInterface(interfaceId); } /** * @notice implement ERC2981 * @param _tokenId Id of the token * @param _salePrice Price of the token * @return recipient address * @return royalty amount */ function royaltyInfo(uint256 _tokenId, uint256 _salePrice) external view override returns (address, uint256) { uint256 royaltyPercentage = tokenRoyaltyPercentage[_tokenId] != 0 ? tokenRoyaltyPercentage[_tokenId] : defaultRoyaltyPercentageIn10000; address royaltyRecipient = tokenRoyaltyRecipient[_tokenId] != address(0) ? tokenRoyaltyRecipient[_tokenId] : defaultRoyaltyRecipient; return (royaltyRecipient, (_salePrice * royaltyPercentage) / 10000); } /** * @notice Returns tokenURI, override to enable reveal/notRevealed * @param tokenId Id of the token to get URI for * @return URI */ function tokenURI( uint256 tokenId ) public view virtual override returns (string memory) { require( ownerOf(tokenId) != address(0), "ERC721Metadata: URI query for nonexistent token" ); if (isRevealed == false) { return notRevealedURI; } string memory identifier = tokenId.toString(); return bytes(baseURI).length != 0 ? string(abi.encodePacked(baseURI, identifier, ".json")) : ""; } /** * @notice Get the amount of mint available for msg.sender * @return Amount of mint available */ function getPublicMintEligibility() public view returns (uint256) { uint256 balance = balanceOf(msg.sender); uint256 maxMint = publicMaxMintAmount; if (balance >= maxMint) { return 0; } return maxMint - balance; } /** * @notice Get the mint details when minting with ETH * @return Mint's max supply * @return Mint's Public price * @return Current total supply * @return Mint's start timestamp */ function getLaunchpadDetails() external view returns (uint256, uint256, uint256, uint256) { return (maxSupply, publicPrice, totalSupply(), publicSaleStartTime); } /** * @notice Get the mint details when minting with ERC20 * @return Mint's ERC20 address * @return Mint's ERC20 price * @return Mint's ERC20 price if using Chainlink */ function getLaunchpadDetailsERC20() external view returns (address, uint256, uint256) { uint256 requiredTokens = 0; if (ethPriceFeedAddress != address(0) && ERC20PriceFeedAddress != address(0)) { requiredTokens = getRequiredERC20TokensChainlink(publicPrice); } return (ERC20TokenAddress, ERC20FixedPricePerToken, requiredTokens); } /** * @notice Get the amount of ERC20 needed to mint at ETH price using Chainlink * @param ethPrice Total ETH needed for the mint * @return Amount of ERC20 needed */ function getRequiredERC20TokensChainlink(uint256 ethPrice) public view returns (uint256) { address ethPriceFeed = ethPriceFeedAddress; address ERC20PriceFeed = ERC20PriceFeedAddress; require(ethPriceFeed != address(0) && ERC20PriceFeed != address(0), "Price feed addresses not set"); // Get the latest prices from Chainlink uint256 ethPriceInUsd = getLatestPrice(ethPriceFeed, ethPriceFeedStaleness); uint256 ERC20PriceInUsd = getLatestPrice(ERC20PriceFeed, ERC20PriceFeedStaleness); // Prices from Chainlink are usually returned with 8 decimals uint256 ethPriceInUsdScaled = ethPriceInUsd * 10**(18 - ethPriceFeedDecimals); // Scale to 18 decimals uint256 ERC20PriceInUsdScaled = ERC20PriceInUsd * 10**(18 - ERC20PriceFeedDecimals); // Scale to 18 decimals // Calculate the equivalent cost in ERC20 tokens uint256 decimalsDiff = 10 ** (18 - ERC20(ERC20TokenAddress).decimals()); //most tokens don't go over 18 decimals uint256 totalERC20Cost = (ethPrice * ethPriceInUsdScaled) / ERC20PriceInUsdScaled / decimalsDiff; // Apply discount if set uint256 ERC20Discount = ERC20DiscountIn10000; if (ERC20Discount > 0) { totalERC20Cost = (totalERC20Cost * (10000 - ERC20Discount)) / 10000; } return totalERC20Cost; } /** * @notice Mint tokens with ETH for the msg.sender * @param _mintAmount Amount of token to mint */ function mint(uint256 _mintAmount) external payable { checkMintRequirements(_mintAmount); require(msg.value >= publicPrice * _mintAmount, "Cost is higher than the amount sent"); _safeMint(msg.sender, _mintAmount); } /** * @notice Mint tokens using ERC20 and Chainlink * @param _mintAmount Amount of tokens to mint */ function mintWithERC20ChainlinkPrice(uint256 _mintAmount) external { checkMintRequirements(_mintAmount); // Let's make sure price feed contract address exists address ERC20Token = ERC20TokenAddress; require(ERC20Token != address(0), "Payment token address not set"); // Calculate the cost in ERC20 tokens uint256 requiredTokenAmount = getRequiredERC20TokensChainlink(publicPrice * _mintAmount); ERC20(ERC20Token).safeTransferFrom(msg.sender, address(this), requiredTokenAmount); _safeMint(msg.sender, _mintAmount); } /** * @notice Mint tokens using ERC20 and a fixed price set by the owner * @param _mintAmount Amount of tokens to mint */ function mintWithFixedERC20Price(uint256 _mintAmount) external { checkMintRequirements(_mintAmount); address ERC20Token = ERC20TokenAddress; uint256 ERC20FixedPrice = ERC20FixedPricePerToken; require(ERC20Token != address(0), "Payment token address not set"); require(ERC20FixedPrice > 0, "Price per token not set"); // Calculate the cost in ERC20 tokens uint256 requiredTokenAmount = ERC20FixedPrice * _mintAmount; ERC20(ERC20Token).safeTransferFrom(msg.sender, address(this), requiredTokenAmount); _safeMint(msg.sender, _mintAmount); } /** * @notice Mint tokens for free as admin to one address * @param _to Address receiving the tokens * @param _mintAmount Amount of tokens to mint */ function adminMint(address _to, uint256 _mintAmount) public onlyOwner { require(totalSupply() + _mintAmount <= maxSupply, "Total supply exceeded"); _safeMint(_to, _mintAmount); } /** * @notice Mint tokens for free as admin to multiple addresses * @param recipients Array of addressed receiving the tokens * @param amounts Array of amount of tokens to mint for each address */ function batchAdminMint(address[] calldata recipients, uint256[] calldata amounts) external onlyOwner { for (uint256 i = 0; i < recipients.length; i++) { adminMint(recipients[i], amounts[i]); } } /** * @notice Set the public price in ETH * @param _newPrice Price in ETH */ function setPublicPrice(uint256 _newPrice) external onlyOwner { publicPrice = _newPrice; } /** * @notice Set the base URI * @param _newBaseURI Base URI */ function setBaseURI(string memory _newBaseURI) external onlyOwner { baseURI = _newBaseURI; } /** * @notice Set the not revealed URI * @param _notRevealedURI Not revealed URI */ function setNotRevealedURI(string memory _notRevealedURI) external onlyOwner { notRevealedURI = _notRevealedURI; } /** * @notice Set the Max supply possible * @param _newMaxSupply Max supply amount */ function setMaxSupply(uint256 _newMaxSupply) external onlyOwner { maxSupply = _newMaxSupply; } /** * @notice Sets the start time of the public sale to a specific timestamp * @param _publicSaleStartTime Start timestamp */ function setPublicSaleStartTime(uint256 _publicSaleStartTime) external onlyOwner { publicSaleStartTime = _publicSaleStartTime; } /** * @notice Set the max amount mintable per address * @param _publicMaxMintAmount Amount mintable */ function setPublicMaxMintAmount(uint256 _publicMaxMintAmount) external onlyOwner { publicMaxMintAmount = _publicMaxMintAmount; } /** * @notice Set the default royalty recipient and BPS * @param _defaultRoyaltyRecipient Address of the recipient * @param _defaultRoyaltyPercentageIn10000 Amount of royalty in BPS */ function setDefaultRoyaltyInfo(address payable _defaultRoyaltyRecipient, uint256 _defaultRoyaltyPercentageIn10000) external onlyOwner { defaultRoyaltyRecipient = _defaultRoyaltyRecipient; defaultRoyaltyPercentageIn10000 = _defaultRoyaltyPercentageIn10000; } /** * @notice Set the royalty info for a specific id * @param _tokenId Id of the token * @param _royaltyRecipient Address of the recipient * @param _royaltyPercentage Amount of royalty in BPS */ function setTokenRoyaltyInfo(uint256 _tokenId, address payable _royaltyRecipient, uint256 _royaltyPercentage) external onlyOwner { require(ownerOf(_tokenId) != address(0), "Token does not exist"); tokenRoyaltyRecipient[_tokenId] = _royaltyRecipient; tokenRoyaltyPercentage[_tokenId] = _royaltyPercentage; } /** * @notice Set the contract URI * @param newURI Contract URI */ function setContractURI(string memory newURI) external onlyOwner { contractURI = newURI; emit ContractURIUpdated(); } /** * @notice Set the ERC20 that can be used to mint * @param _ERC20TokenAddress Address of the ERC20 */ function setERC20TokenAddress(address _ERC20TokenAddress) external onlyOwner { ERC20TokenAddress = _ERC20TokenAddress; } /** * @notice Set the fixed price to mint with ERC20 * @param _erc20FixedPricePerToken Price in ERC20 */ function setErc20FixedPricePerToken(uint256 _erc20FixedPricePerToken) external onlyOwner { ERC20FixedPricePerToken = _erc20FixedPricePerToken; } /** * @notice Set the ERC20 priceFeed details * @param _ERC20PriceFeedAddress Address of the pricefeed * @param _maxStaleness Max staleness */ function setERC20PriceFeedAddress(address _ERC20PriceFeedAddress, uint256 _maxStaleness) external onlyOwner { ERC20PriceFeedAddress = _ERC20PriceFeedAddress; ERC20PriceFeedDecimals = uint32(AggregatorV3Interface(_ERC20PriceFeedAddress).decimals()); ERC20PriceFeedStaleness = uint32(_maxStaleness); } /** * @notice Set the ERC20 priceFeed details * @param _ethPriceFeedAddress Address of the pricefeed * @param _maxStaleness Max staleness */ function setETHPriceFeedAddress(address _ethPriceFeedAddress, uint256 _maxStaleness) external onlyOwner { ethPriceFeedAddress = _ethPriceFeedAddress; ethPriceFeedDecimals = uint32(AggregatorV3Interface(_ethPriceFeedAddress).decimals()); ethPriceFeedStaleness = uint32(_maxStaleness); } /** * @notice Set the discount when minting with ERC20 * @param _erc20DiscountIn10000 Discount in BPS */ function setERC20DiscountIn10000(uint256 _erc20DiscountIn10000) external onlyOwner { require(_erc20DiscountIn10000 <= 10000, "Invalid discount percentage"); ERC20DiscountIn10000 = _erc20DiscountIn10000; } /** * Set trading enabled for the token * @param _tradingEnabled Boolean to enable trading or not */ function setTradingEnabled(bool _tradingEnabled) external onlyOwner { tradingEnabled = _tradingEnabled; } /** * @notice Add an address to the blacklist, not allowing them to transfer tokens anymore * @param _address Address to add to the blacklist */ function addToBlacklist(address _address) external onlyOwner { blacklist[_address] = true; } /** * Remove an address from the blacklist * @param _address Address to remove from the blacklist */ function removeFromBlacklist(address _address) external onlyOwner { blacklist[_address] = false; } /** * @notice Toggle the sale on or off by modifying the publicSaleStartTime variable */ function togglePublicSaleActive() external onlyOwner { if (block.timestamp < publicSaleStartTime) { publicSaleStartTime = block.timestamp; } else { // This effectively disables the public sale by setting the start time to a far future publicSaleStartTime = type(uint256).max; } } /** * @notice Toggle reveal on or off */ function toggleReveal() external onlyOwner { isRevealed = !isRevealed; } /** * @notice Withdraw the fixed commission for the commissionRecipientAddress */ function withdrawFixedCommission() external { require( msg.sender == owner() || msg.sender == commissionRecipientAddress, "Only owner or commission recipient can withdraw" ); uint256 withdrawn = totalCommissionWithdrawn; uint256 remainingCommission = fixedCommissionThreshold - withdrawn; uint256 amount = remainingCommission > address(this).balance ? address(this).balance : remainingCommission; // Updating the total withdrawn by A before making the transfer totalCommissionWithdrawn += amount; (bool success, ) = commissionRecipientAddress.call{value: amount}(""); require(success, "Transfer failed"); } /** * @notice Withdraw ETH for the actor calling and saves the other actors due in storage to limit DOS from one actor */ function withdraw() external virtual { require( msg.sender == owner() || msg.sender == commissionRecipientAddress || msg.sender == withdrawalRecipientAddress, "Only owner or commission recipient can withdraw" ); uint256 _commissionToWithdraw = commissionToWithdraw; uint256 _ownerToWithdraw = ownerToWithdraw; //This is ok if fixedCommissionThreshold makes it underflow, we don't allow eth withdraws until fixedCommissionThreshold can be paid fully uint256 available = address(this).balance - (fixedCommissionThreshold - totalCommissionWithdrawn) - _commissionToWithdraw - _ownerToWithdraw; uint256 newCommission = available * commissionPercentageIn10000 / 10000; uint256 newOwnerAmount = available - newCommission; if (msg.sender == commissionRecipientAddress) { ownerToWithdraw += newOwnerAmount; _commissionToWithdraw += newCommission; commissionToWithdraw = 0; (bool success, ) = commissionRecipientAddress.call{value: _commissionToWithdraw}(""); require(success); } else if (msg.sender == withdrawalRecipientAddress || msg.sender == owner()) { commissionToWithdraw += newCommission; _ownerToWithdraw += newOwnerAmount; ownerToWithdraw = 0; (bool success, ) = withdrawalRecipientAddress.call{value: _ownerToWithdraw}(""); require(success); } } /** * @notice Withdraw ERC20 for every actors * @param erc20Token Address of ther ERC20 to withdraw */ function withdrawERC20(ERC20 erc20Token) external { require( msg.sender == owner() || msg.sender == commissionRecipientAddress || msg.sender == withdrawalRecipientAddress, "Only owner or commission recipient can withdraw" ); uint256 erc20Balance = erc20Token.balanceOf(address(this)); uint256 commission = (erc20Balance * commissionPercentageIn10000) / 10000; uint256 withdrawalAddressAmount = erc20Balance - commission; //withdrawalRecipientAddress if(commission > 0) { erc20Token.safeTransfer(commissionRecipientAddress, commission); } if(withdrawalAddressAmount > 0) { erc20Token.safeTransfer(withdrawalRecipientAddress, withdrawalAddressAmount); } } /** * @notice Allows the owner to withdraw all ETH 28 weeks after deploy time */ function emergencyWithdraw() external onlyOwner { require(block.timestamp > deployTimestamp + 28 weeks, "Too early to emergency withdraw"); uint256 balance = address(this).balance; (bool success,) = payable(owner()).call{value: balance}(""); require(success); } /** * @notice Allows the owner to withdraw all ERC20 28 weeks after deploy time */ function emergencyWithdrawERC20(ERC20 erc20Token) external onlyOwner { require(block.timestamp > deployTimestamp + 28 weeks, "Too early to emergency withdraw"); uint256 balance = erc20Token.balanceOf(address(this)); erc20Token.safeTransfer(owner(), balance); } /** * @notice Get the price to mint with an ERC20 using Chainlink * @param priceFeedAddress Address of the pricefeed * @param maxStaleness Max staleness accepted for the pricefeed * @return Price received from Chainlink */ function getLatestPrice(address priceFeedAddress, uint256 maxStaleness) internal view returns (uint256) { AggregatorV3Interface priceFeed = AggregatorV3Interface(priceFeedAddress); (, int256 price,,uint256 updatedAt,) = priceFeed.latestRoundData(); require(price > 0, "Invalid price from Chainlink"); require(updatedAt > block.timestamp - maxStaleness, "Invalid price from Chainlink"); return uint256(price); } /** * @notice Enforce mint requirements * @param _mintAmount Amount of token to mint */ function checkMintRequirements(uint256 _mintAmount) internal view { require(totalSupply() + _mintAmount <= maxSupply, "Total supply exceeded"); require(block.timestamp >= publicSaleStartTime, "Public sale not active"); require(getPublicMintEligibility() >= _mintAmount, "Invalid amount to be minted"); } /** * @notice Override to start from 1 instead of 0 */ function _startTokenId() internal view virtual override returns (uint256) { return 1; } /** * @notice Override to enable blacklist and pause */ function _beforeTokenTransfers( address from, address to, uint256 tokenId, uint256 batchSize ) internal override { if (from != address(0)) { require(tradingEnabled, "Trading is disabled"); require(!blacklist[msg.sender], "Blacklisted entities cannot execute trades"); } super._beforeTokenTransfers(from, to, tokenId, batchSize); } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v5.0.0) (access/Ownable.sol) pragma solidity ^0.8.20; import {Context} from "../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. * * The initial owner is set to the address provided by the deployer. This can * later be changed with {transferOwnership}. * * This module is used through inheritance. It will make available the modifier * `onlyOwner`, which can be applied to your functions to restrict their use to * the owner. */ abstract contract Ownable is Context { address private _owner; /** * @dev The caller account is not authorized to perform an operation. */ error OwnableUnauthorizedAccount(address account); /** * @dev The owner is not a valid owner account. (eg. `address(0)`) */ error OwnableInvalidOwner(address owner); event OwnershipTransferred(address indexed previousOwner, address indexed newOwner); /** * @dev Initializes the contract setting the address provided by the deployer as the initial owner. */ constructor(address initialOwner) { if (initialOwner == address(0)) { revert OwnableInvalidOwner(address(0)); } _transferOwnership(initialOwner); } /** * @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 { if (owner() != _msgSender()) { revert OwnableUnauthorizedAccount(_msgSender()); } } /** * @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 { if (newOwner == address(0)) { revert OwnableInvalidOwner(address(0)); } _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: MIT // ERC721A Contracts v4.3.0 // Creator: Chiru Labs pragma solidity ^0.8.4; import './IERC721A.sol'; /** * @dev Interface of ERC721 token receiver. */ interface ERC721A__IERC721Receiver { function onERC721Received( address operator, address from, uint256 tokenId, bytes calldata data ) external returns (bytes4); } /** * @title ERC721A * * @dev Implementation of the [ERC721](https://eips.ethereum.org/EIPS/eip-721) * Non-Fungible Token Standard, including the Metadata extension. * Optimized for lower gas during batch mints. * * Token IDs are minted in sequential order (e.g. 0, 1, 2, 3, ...) * starting from `_startTokenId()`. * * The `_sequentialUpTo()` function can be overriden to enable spot mints * (i.e. non-consecutive mints) for `tokenId`s greater than `_sequentialUpTo()`. * * Assumptions: * * - An owner cannot have more than 2**64 - 1 (max value of uint64) of supply. * - The maximum token ID cannot exceed 2**256 - 1 (max value of uint256). */ contract ERC721A is IERC721A { // Bypass for a `--via-ir` bug (https://github.com/chiru-labs/ERC721A/pull/364). struct TokenApprovalRef { address value; } // ============================================================= // CONSTANTS // ============================================================= // Mask of an entry in packed address data. uint256 private constant _BITMASK_ADDRESS_DATA_ENTRY = (1 << 64) - 1; // The bit position of `numberMinted` in packed address data. uint256 private constant _BITPOS_NUMBER_MINTED = 64; // The bit position of `numberBurned` in packed address data. uint256 private constant _BITPOS_NUMBER_BURNED = 128; // The bit position of `aux` in packed address data. uint256 private constant _BITPOS_AUX = 192; // Mask of all 256 bits in packed address data except the 64 bits for `aux`. uint256 private constant _BITMASK_AUX_COMPLEMENT = (1 << 192) - 1; // The bit position of `startTimestamp` in packed ownership. uint256 private constant _BITPOS_START_TIMESTAMP = 160; // The bit mask of the `burned` bit in packed ownership. uint256 private constant _BITMASK_BURNED = 1 << 224; // The bit position of the `nextInitialized` bit in packed ownership. uint256 private constant _BITPOS_NEXT_INITIALIZED = 225; // The bit mask of the `nextInitialized` bit in packed ownership. uint256 private constant _BITMASK_NEXT_INITIALIZED = 1 << 225; // The bit position of `extraData` in packed ownership. uint256 private constant _BITPOS_EXTRA_DATA = 232; // Mask of all 256 bits in a packed ownership except the 24 bits for `extraData`. uint256 private constant _BITMASK_EXTRA_DATA_COMPLEMENT = (1 << 232) - 1; // The mask of the lower 160 bits for addresses. uint256 private constant _BITMASK_ADDRESS = (1 << 160) - 1; // The maximum `quantity` that can be minted with {_mintERC2309}. // This limit is to prevent overflows on the address data entries. // For a limit of 5000, a total of 3.689e15 calls to {_mintERC2309} // is required to cause an overflow, which is unrealistic. uint256 private constant _MAX_MINT_ERC2309_QUANTITY_LIMIT = 5000; // The `Transfer` event signature is given by: // `keccak256(bytes("Transfer(address,address,uint256)"))`. bytes32 private constant _TRANSFER_EVENT_SIGNATURE = 0xddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef; // ============================================================= // STORAGE // ============================================================= // The next token ID to be minted. uint256 private _currentIndex; // The number of tokens burned. uint256 private _burnCounter; // Token name string private _name; // Token symbol string private _symbol; // Mapping from token ID to ownership details // An empty struct value does not necessarily mean the token is unowned. // See {_packedOwnershipOf} implementation for details. // // Bits Layout: // - [0..159] `addr` // - [160..223] `startTimestamp` // - [224] `burned` // - [225] `nextInitialized` // - [232..255] `extraData` mapping(uint256 => uint256) private _packedOwnerships; // Mapping owner address to address data. // // Bits Layout: // - [0..63] `balance` // - [64..127] `numberMinted` // - [128..191] `numberBurned` // - [192..255] `aux` mapping(address => uint256) private _packedAddressData; // Mapping from token ID to approved address. mapping(uint256 => TokenApprovalRef) private _tokenApprovals; // Mapping from owner to operator approvals mapping(address => mapping(address => bool)) private _operatorApprovals; // The amount of tokens minted above `_sequentialUpTo()`. // We call these spot mints (i.e. non-sequential mints). uint256 private _spotMinted; // ============================================================= // CONSTRUCTOR // ============================================================= constructor(string memory name_, string memory symbol_) { _name = name_; _symbol = symbol_; _currentIndex = _startTokenId(); if (_sequentialUpTo() < _startTokenId()) _revert(SequentialUpToTooSmall.selector); } // ============================================================= // TOKEN COUNTING OPERATIONS // ============================================================= /** * @dev Returns the starting token ID for sequential mints. * * Override this function to change the starting token ID for sequential mints. * * Note: The value returned must never change after any tokens have been minted. */ function _startTokenId() internal view virtual returns (uint256) { return 0; } /** * @dev Returns the maximum token ID (inclusive) for sequential mints. * * Override this function to return a value less than 2**256 - 1, * but greater than `_startTokenId()`, to enable spot (non-sequential) mints. * * Note: The value returned must never change after any tokens have been minted. */ function _sequentialUpTo() internal view virtual returns (uint256) { return type(uint256).max; } /** * @dev Returns the next token ID to be minted. */ function _nextTokenId() internal view virtual returns (uint256) { return _currentIndex; } /** * @dev Returns the total number of tokens in existence. * Burned tokens will reduce the count. * To get the total number of tokens minted, please see {_totalMinted}. */ function totalSupply() public view virtual override returns (uint256 result) { // Counter underflow is impossible as `_burnCounter` cannot be incremented // more than `_currentIndex + _spotMinted - _startTokenId()` times. unchecked { // With spot minting, the intermediate `result` can be temporarily negative, // and the computation must be unchecked. result = _currentIndex - _burnCounter - _startTokenId(); if (_sequentialUpTo() != type(uint256).max) result += _spotMinted; } } /** * @dev Returns the total amount of tokens minted in the contract. */ function _totalMinted() internal view virtual returns (uint256 result) { // Counter underflow is impossible as `_currentIndex` does not decrement, // and it is initialized to `_startTokenId()`. unchecked { result = _currentIndex - _startTokenId(); if (_sequentialUpTo() != type(uint256).max) result += _spotMinted; } } /** * @dev Returns the total number of tokens burned. */ function _totalBurned() internal view virtual returns (uint256) { return _burnCounter; } /** * @dev Returns the total number of tokens that are spot-minted. */ function _totalSpotMinted() internal view virtual returns (uint256) { return _spotMinted; } // ============================================================= // ADDRESS DATA OPERATIONS // ============================================================= /** * @dev Returns the number of tokens in `owner`'s account. */ function balanceOf(address owner) public view virtual override returns (uint256) { if (owner == address(0)) _revert(BalanceQueryForZeroAddress.selector); return _packedAddressData[owner] & _BITMASK_ADDRESS_DATA_ENTRY; } /** * Returns the number of tokens minted by `owner`. */ function _numberMinted(address owner) internal view returns (uint256) { return (_packedAddressData[owner] >> _BITPOS_NUMBER_MINTED) & _BITMASK_ADDRESS_DATA_ENTRY; } /** * Returns the number of tokens burned by or on behalf of `owner`. */ function _numberBurned(address owner) internal view returns (uint256) { return (_packedAddressData[owner] >> _BITPOS_NUMBER_BURNED) & _BITMASK_ADDRESS_DATA_ENTRY; } /** * Returns the auxiliary data for `owner`. (e.g. number of whitelist mint slots used). */ function _getAux(address owner) internal view returns (uint64) { return uint64(_packedAddressData[owner] >> _BITPOS_AUX); } /** * Sets the auxiliary data for `owner`. (e.g. number of whitelist mint slots used). * If there are multiple variables, please pack them into a uint64. */ function _setAux(address owner, uint64 aux) internal virtual { uint256 packed = _packedAddressData[owner]; uint256 auxCasted; // Cast `aux` with assembly to avoid redundant masking. assembly { auxCasted := aux } packed = (packed & _BITMASK_AUX_COMPLEMENT) | (auxCasted << _BITPOS_AUX); _packedAddressData[owner] = packed; } // ============================================================= // IERC165 // ============================================================= /** * @dev Returns true if this contract implements the interface defined by * `interfaceId`. See the corresponding * [EIP section](https://eips.ethereum.org/EIPS/eip-165#how-interfaces-are-identified) * to learn more about how these ids are created. * * This function call must use less than 30000 gas. */ function supportsInterface(bytes4 interfaceId) public view virtual override returns (bool) { // The interface IDs are constants representing the first 4 bytes // of the XOR of all function selectors in the interface. // See: [ERC165](https://eips.ethereum.org/EIPS/eip-165) // (e.g. `bytes4(i.functionA.selector ^ i.functionB.selector ^ ...)`) return interfaceId == 0x01ffc9a7 || // ERC165 interface ID for ERC165. interfaceId == 0x80ac58cd || // ERC165 interface ID for ERC721. interfaceId == 0x5b5e139f; // ERC165 interface ID for ERC721Metadata. } // ============================================================= // IERC721Metadata // ============================================================= /** * @dev Returns the token collection name. */ function name() public view virtual override returns (string memory) { return _name; } /** * @dev Returns the token collection symbol. */ function symbol() public view virtual override returns (string memory) { return _symbol; } /** * @dev Returns the Uniform Resource Identifier (URI) for `tokenId` token. */ function tokenURI(uint256 tokenId) public view virtual override returns (string memory) { if (!_exists(tokenId)) _revert(URIQueryForNonexistentToken.selector); string memory baseURI = _baseURI(); return bytes(baseURI).length != 0 ? string(abi.encodePacked(baseURI, _toString(tokenId))) : ''; } /** * @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, it can be overridden in child contracts. */ function _baseURI() internal view virtual returns (string memory) { return ''; } // ============================================================= // OWNERSHIPS OPERATIONS // ============================================================= /** * @dev Returns the owner of the `tokenId` token. * * Requirements: * * - `tokenId` must exist. */ function ownerOf(uint256 tokenId) public view virtual override returns (address) { return address(uint160(_packedOwnershipOf(tokenId))); } /** * @dev Gas spent here starts off proportional to the maximum mint batch size. * It gradually moves to O(1) as tokens get transferred around over time. */ function _ownershipOf(uint256 tokenId) internal view virtual returns (TokenOwnership memory) { return _unpackedOwnership(_packedOwnershipOf(tokenId)); } /** * @dev Returns the unpacked `TokenOwnership` struct at `index`. */ function _ownershipAt(uint256 index) internal view virtual returns (TokenOwnership memory) { return _unpackedOwnership(_packedOwnerships[index]); } /** * @dev Returns whether the ownership slot at `index` is initialized. * An uninitialized slot does not necessarily mean that the slot has no owner. */ function _ownershipIsInitialized(uint256 index) internal view virtual returns (bool) { return _packedOwnerships[index] != 0; } /** * @dev Initializes the ownership slot minted at `index` for efficiency purposes. */ function _initializeOwnershipAt(uint256 index) internal virtual { if (_packedOwnerships[index] == 0) { _packedOwnerships[index] = _packedOwnershipOf(index); } } /** * @dev Returns the packed ownership data of `tokenId`. */ function _packedOwnershipOf(uint256 tokenId) private view returns (uint256 packed) { if (_startTokenId() <= tokenId) { packed = _packedOwnerships[tokenId]; if (tokenId > _sequentialUpTo()) { if (_packedOwnershipExists(packed)) return packed; _revert(OwnerQueryForNonexistentToken.selector); } // If the data at the starting slot does not exist, start the scan. if (packed == 0) { if (tokenId >= _currentIndex) _revert(OwnerQueryForNonexistentToken.selector); // Invariant: // There will always be an initialized ownership slot // (i.e. `ownership.addr != address(0) && ownership.burned == false`) // before an unintialized ownership slot // (i.e. `ownership.addr == address(0) && ownership.burned == false`) // Hence, `tokenId` will not underflow. // // We can directly compare the packed value. // If the address is zero, packed will be zero. for (;;) { unchecked { packed = _packedOwnerships[--tokenId]; } if (packed == 0) continue; if (packed & _BITMASK_BURNED == 0) return packed; // Otherwise, the token is burned, and we must revert. // This handles the case of batch burned tokens, where only the burned bit // of the starting slot is set, and remaining slots are left uninitialized. _revert(OwnerQueryForNonexistentToken.selector); } } // Otherwise, the data exists and we can skip the scan. // This is possible because we have already achieved the target condition. // This saves 2143 gas on transfers of initialized tokens. // If the token is not burned, return `packed`. Otherwise, revert. if (packed & _BITMASK_BURNED == 0) return packed; } _revert(OwnerQueryForNonexistentToken.selector); } /** * @dev Returns the unpacked `TokenOwnership` struct from `packed`. */ function _unpackedOwnership(uint256 packed) private pure returns (TokenOwnership memory ownership) { ownership.addr = address(uint160(packed)); ownership.startTimestamp = uint64(packed >> _BITPOS_START_TIMESTAMP); ownership.burned = packed & _BITMASK_BURNED != 0; ownership.extraData = uint24(packed >> _BITPOS_EXTRA_DATA); } /** * @dev Packs ownership data into a single uint256. */ function _packOwnershipData(address owner, uint256 flags) private view returns (uint256 result) { assembly { // Mask `owner` to the lower 160 bits, in case the upper bits somehow aren't clean. owner := and(owner, _BITMASK_ADDRESS) // `owner | (block.timestamp << _BITPOS_START_TIMESTAMP) | flags`. result := or(owner, or(shl(_BITPOS_START_TIMESTAMP, timestamp()), flags)) } } /** * @dev Returns the `nextInitialized` flag set if `quantity` equals 1. */ function _nextInitializedFlag(uint256 quantity) private pure returns (uint256 result) { // For branchless setting of the `nextInitialized` flag. assembly { // `(quantity == 1) << _BITPOS_NEXT_INITIALIZED`. result := shl(_BITPOS_NEXT_INITIALIZED, eq(quantity, 1)) } } // ============================================================= // APPROVAL OPERATIONS // ============================================================= /** * @dev Gives permission to `to` to transfer `tokenId` token to another account. See {ERC721A-_approve}. * * Requirements: * * - The caller must own the token or be an approved operator. */ function approve(address to, uint256 tokenId) public payable virtual override { _approve(to, tokenId, true); } /** * @dev Returns the account approved for `tokenId` token. * * Requirements: * * - `tokenId` must exist. */ function getApproved(uint256 tokenId) public view virtual override returns (address) { if (!_exists(tokenId)) _revert(ApprovalQueryForNonexistentToken.selector); return _tokenApprovals[tokenId].value; } /** * @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) public virtual override { _operatorApprovals[_msgSenderERC721A()][operator] = approved; emit ApprovalForAll(_msgSenderERC721A(), operator, approved); } /** * @dev Returns if the `operator` is allowed to manage all of the assets of `owner`. * * See {setApprovalForAll}. */ function isApprovedForAll(address owner, address operator) public view virtual override returns (bool) { return _operatorApprovals[owner][operator]; } /** * @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. See {_mint}. */ function _exists(uint256 tokenId) internal view virtual returns (bool result) { if (_startTokenId() <= tokenId) { if (tokenId > _sequentialUpTo()) return _packedOwnershipExists(_packedOwnerships[tokenId]); if (tokenId < _currentIndex) { uint256 packed; while ((packed = _packedOwnerships[tokenId]) == 0) --tokenId; result = packed & _BITMASK_BURNED == 0; } } } /** * @dev Returns whether `packed` represents a token that exists. */ function _packedOwnershipExists(uint256 packed) private pure returns (bool result) { assembly { // The following is equivalent to `owner != address(0) && burned == false`. // Symbolically tested. result := gt(and(packed, _BITMASK_ADDRESS), and(packed, _BITMASK_BURNED)) } } /** * @dev Returns whether `msgSender` is equal to `approvedAddress` or `owner`. */ function _isSenderApprovedOrOwner( address approvedAddress, address owner, address msgSender ) private pure returns (bool result) { assembly { // Mask `owner` to the lower 160 bits, in case the upper bits somehow aren't clean. owner := and(owner, _BITMASK_ADDRESS) // Mask `msgSender` to the lower 160 bits, in case the upper bits somehow aren't clean. msgSender := and(msgSender, _BITMASK_ADDRESS) // `msgSender == owner || msgSender == approvedAddress`. result := or(eq(msgSender, owner), eq(msgSender, approvedAddress)) } } /** * @dev Returns the storage slot and value for the approved address of `tokenId`. */ function _getApprovedSlotAndAddress(uint256 tokenId) private view returns (uint256 approvedAddressSlot, address approvedAddress) { TokenApprovalRef storage tokenApproval = _tokenApprovals[tokenId]; // The following is equivalent to `approvedAddress = _tokenApprovals[tokenId].value`. assembly { approvedAddressSlot := tokenApproval.slot approvedAddress := sload(approvedAddressSlot) } } // ============================================================= // TRANSFER OPERATIONS // ============================================================= /** * @dev Transfers `tokenId` from `from` to `to`. * * 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 ) public payable virtual override { uint256 prevOwnershipPacked = _packedOwnershipOf(tokenId); // Mask `from` to the lower 160 bits, in case the upper bits somehow aren't clean. from = address(uint160(uint256(uint160(from)) & _BITMASK_ADDRESS)); if (address(uint160(prevOwnershipPacked)) != from) _revert(TransferFromIncorrectOwner.selector); (uint256 approvedAddressSlot, address approvedAddress) = _getApprovedSlotAndAddress(tokenId); // The nested ifs save around 20+ gas over a compound boolean condition. if (!_isSenderApprovedOrOwner(approvedAddress, from, _msgSenderERC721A())) if (!isApprovedForAll(from, _msgSenderERC721A())) _revert(TransferCallerNotOwnerNorApproved.selector); _beforeTokenTransfers(from, to, tokenId, 1); // Clear approvals from the previous owner. assembly { if approvedAddress { // This is equivalent to `delete _tokenApprovals[tokenId]`. sstore(approvedAddressSlot, 0) } } // Underflow of the sender's balance is impossible because we check for // ownership above and the recipient's balance can't realistically overflow. // Counter overflow is incredibly unrealistic as `tokenId` would have to be 2**256. unchecked { // We can directly increment and decrement the balances. --_packedAddressData[from]; // Updates: `balance -= 1`. ++_packedAddressData[to]; // Updates: `balance += 1`. // Updates: // - `address` to the next owner. // - `startTimestamp` to the timestamp of transfering. // - `burned` to `false`. // - `nextInitialized` to `true`. _packedOwnerships[tokenId] = _packOwnershipData( to, _BITMASK_NEXT_INITIALIZED | _nextExtraData(from, to, prevOwnershipPacked) ); // If the next slot may not have been initialized (i.e. `nextInitialized == false`) . if (prevOwnershipPacked & _BITMASK_NEXT_INITIALIZED == 0) { uint256 nextTokenId = tokenId + 1; // If the next slot's address is zero and not burned (i.e. packed value is zero). if (_packedOwnerships[nextTokenId] == 0) { // If the next slot is within bounds. if (nextTokenId != _currentIndex) { // Initialize the next slot to maintain correctness for `ownerOf(tokenId + 1)`. _packedOwnerships[nextTokenId] = prevOwnershipPacked; } } } } // Mask `to` to the lower 160 bits, in case the upper bits somehow aren't clean. uint256 toMasked = uint256(uint160(to)) & _BITMASK_ADDRESS; assembly { // Emit the `Transfer` event. log4( 0, // Start of data (0, since no data). 0, // End of data (0, since no data). _TRANSFER_EVENT_SIGNATURE, // Signature. from, // `from`. toMasked, // `to`. tokenId // `tokenId`. ) } if (toMasked == 0) _revert(TransferToZeroAddress.selector); _afterTokenTransfers(from, to, tokenId, 1); } /** * @dev Equivalent to `safeTransferFrom(from, to, tokenId, '')`. */ function safeTransferFrom( address from, address to, uint256 tokenId ) public payable virtual override { safeTransferFrom(from, to, tokenId, ''); } /** * @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 memory _data ) public payable virtual override { transferFrom(from, to, tokenId); if (to.code.length != 0) if (!_checkContractOnERC721Received(from, to, tokenId, _data)) { _revert(TransferToNonERC721ReceiverImplementer.selector); } } /** * @dev Hook that is called before a set of serially-ordered token IDs * are about to be transferred. This includes minting. * And also called before burning one token. * * `startTokenId` - the first token ID to be transferred. * `quantity` - the amount to be transferred. * * Calling conditions: * * - When `from` and `to` are both non-zero, `from`'s `tokenId` will be * transferred to `to`. * - When `from` is zero, `tokenId` will be minted for `to`. * - When `to` is zero, `tokenId` will be burned by `from`. * - `from` and `to` are never both zero. */ function _beforeTokenTransfers( address from, address to, uint256 startTokenId, uint256 quantity ) internal virtual {} /** * @dev Hook that is called after a set of serially-ordered token IDs * have been transferred. This includes minting. * And also called after one token has been burned. * * `startTokenId` - the first token ID to be transferred. * `quantity` - the amount to be transferred. * * Calling conditions: * * - When `from` and `to` are both non-zero, `from`'s `tokenId` has been * transferred to `to`. * - When `from` is zero, `tokenId` has been minted for `to`. * - When `to` is zero, `tokenId` has been burned by `from`. * - `from` and `to` are never both zero. */ function _afterTokenTransfers( address from, address to, uint256 startTokenId, uint256 quantity ) internal virtual {} /** * @dev Private function to invoke {IERC721Receiver-onERC721Received} on a target contract. * * `from` - Previous owner of the given token ID. * `to` - Target address that will receive the token. * `tokenId` - Token ID to be transferred. * `_data` - Optional data to send along with the call. * * Returns whether the call correctly returned the expected magic value. */ function _checkContractOnERC721Received( address from, address to, uint256 tokenId, bytes memory _data ) private returns (bool) { try ERC721A__IERC721Receiver(to).onERC721Received(_msgSenderERC721A(), from, tokenId, _data) returns ( bytes4 retval ) { return retval == ERC721A__IERC721Receiver(to).onERC721Received.selector; } catch (bytes memory reason) { if (reason.length == 0) { _revert(TransferToNonERC721ReceiverImplementer.selector); } assembly { revert(add(32, reason), mload(reason)) } } } // ============================================================= // MINT OPERATIONS // ============================================================= /** * @dev Mints `quantity` tokens and transfers them to `to`. * * Requirements: * * - `to` cannot be the zero address. * - `quantity` must be greater than 0. * * Emits a {Transfer} event for each mint. */ function _mint(address to, uint256 quantity) internal virtual { uint256 startTokenId = _currentIndex; if (quantity == 0) _revert(MintZeroQuantity.selector); _beforeTokenTransfers(address(0), to, startTokenId, quantity); // Overflows are incredibly unrealistic. // `balance` and `numberMinted` have a maximum limit of 2**64. // `tokenId` has a maximum limit of 2**256. unchecked { // Updates: // - `address` to the owner. // - `startTimestamp` to the timestamp of minting. // - `burned` to `false`. // - `nextInitialized` to `quantity == 1`. _packedOwnerships[startTokenId] = _packOwnershipData( to, _nextInitializedFlag(quantity) | _nextExtraData(address(0), to, 0) ); // Updates: // - `balance += quantity`. // - `numberMinted += quantity`. // // We can directly add to the `balance` and `numberMinted`. _packedAddressData[to] += quantity * ((1 << _BITPOS_NUMBER_MINTED) | 1); // Mask `to` to the lower 160 bits, in case the upper bits somehow aren't clean. uint256 toMasked = uint256(uint160(to)) & _BITMASK_ADDRESS; if (toMasked == 0) _revert(MintToZeroAddress.selector); uint256 end = startTokenId + quantity; uint256 tokenId = startTokenId; if (end - 1 > _sequentialUpTo()) _revert(SequentialMintExceedsLimit.selector); do { assembly { // Emit the `Transfer` event. log4( 0, // Start of data (0, since no data). 0, // End of data (0, since no data). _TRANSFER_EVENT_SIGNATURE, // Signature. 0, // `address(0)`. toMasked, // `to`. tokenId // `tokenId`. ) } // The `!=` check ensures that large values of `quantity` // that overflows uint256 will make the loop run out of gas. } while (++tokenId != end); _currentIndex = end; } _afterTokenTransfers(address(0), to, startTokenId, quantity); } /** * @dev Mints `quantity` tokens and transfers them to `to`. * * This function is intended for efficient minting only during contract creation. * * It emits only one {ConsecutiveTransfer} as defined in * [ERC2309](https://eips.ethereum.org/EIPS/eip-2309), * instead of a sequence of {Transfer} event(s). * * Calling this function outside of contract creation WILL make your contract * non-compliant with the ERC721 standard. * For full ERC721 compliance, substituting ERC721 {Transfer} event(s) with the ERC2309 * {ConsecutiveTransfer} event is only permissible during contract creation. * * Requirements: * * - `to` cannot be the zero address. * - `quantity` must be greater than 0. * * Emits a {ConsecutiveTransfer} event. */ function _mintERC2309(address to, uint256 quantity) internal virtual { uint256 startTokenId = _currentIndex; if (to == address(0)) _revert(MintToZeroAddress.selector); if (quantity == 0) _revert(MintZeroQuantity.selector); if (quantity > _MAX_MINT_ERC2309_QUANTITY_LIMIT) _revert(MintERC2309QuantityExceedsLimit.selector); _beforeTokenTransfers(address(0), to, startTokenId, quantity); // Overflows are unrealistic due to the above check for `quantity` to be below the limit. unchecked { // Updates: // - `balance += quantity`. // - `numberMinted += quantity`. // // We can directly add to the `balance` and `numberMinted`. _packedAddressData[to] += quantity * ((1 << _BITPOS_NUMBER_MINTED) | 1); // Updates: // - `address` to the owner. // - `startTimestamp` to the timestamp of minting. // - `burned` to `false`. // - `nextInitialized` to `quantity == 1`. _packedOwnerships[startTokenId] = _packOwnershipData( to, _nextInitializedFlag(quantity) | _nextExtraData(address(0), to, 0) ); if (startTokenId + quantity - 1 > _sequentialUpTo()) _revert(SequentialMintExceedsLimit.selector); emit ConsecutiveTransfer(startTokenId, startTokenId + quantity - 1, address(0), to); _currentIndex = startTokenId + quantity; } _afterTokenTransfers(address(0), to, startTokenId, quantity); } /** * @dev Safely mints `quantity` tokens and transfers them to `to`. * * Requirements: * * - If `to` refers to a smart contract, it must implement * {IERC721Receiver-onERC721Received}, which is called for each safe transfer. * - `quantity` must be greater than 0. * * See {_mint}. * * Emits a {Transfer} event for each mint. */ function _safeMint( address to, uint256 quantity, bytes memory _data ) internal virtual { _mint(to, quantity); unchecked { if (to.code.length != 0) { uint256 end = _currentIndex; uint256 index = end - quantity; do { if (!_checkContractOnERC721Received(address(0), to, index++, _data)) { _revert(TransferToNonERC721ReceiverImplementer.selector); } } while (index < end); // This prevents reentrancy to `_safeMint`. // It does not prevent reentrancy to `_safeMintSpot`. if (_currentIndex != end) revert(); } } } /** * @dev Equivalent to `_safeMint(to, quantity, '')`. */ function _safeMint(address to, uint256 quantity) internal virtual { _safeMint(to, quantity, ''); } /** * @dev Mints a single token at `tokenId`. * * Note: A spot-minted `tokenId` that has been burned can be re-minted again. * * Requirements: * * - `to` cannot be the zero address. * - `tokenId` must be greater than `_sequentialUpTo()`. * - `tokenId` must not exist. * * Emits a {Transfer} event for each mint. */ function _mintSpot(address to, uint256 tokenId) internal virtual { if (tokenId <= _sequentialUpTo()) _revert(SpotMintTokenIdTooSmall.selector); uint256 prevOwnershipPacked = _packedOwnerships[tokenId]; if (_packedOwnershipExists(prevOwnershipPacked)) _revert(TokenAlreadyExists.selector); _beforeTokenTransfers(address(0), to, tokenId, 1); // Overflows are incredibly unrealistic. // The `numberMinted` for `to` is incremented by 1, and has a max limit of 2**64 - 1. // `_spotMinted` is incremented by 1, and has a max limit of 2**256 - 1. unchecked { // Updates: // - `address` to the owner. // - `startTimestamp` to the timestamp of minting. // - `burned` to `false`. // - `nextInitialized` to `true` (as `quantity == 1`). _packedOwnerships[tokenId] = _packOwnershipData( to, _nextInitializedFlag(1) | _nextExtraData(address(0), to, prevOwnershipPacked) ); // Updates: // - `balance += 1`. // - `numberMinted += 1`. // // We can directly add to the `balance` and `numberMinted`. _packedAddressData[to] += (1 << _BITPOS_NUMBER_MINTED) | 1; // Mask `to` to the lower 160 bits, in case the upper bits somehow aren't clean. uint256 toMasked = uint256(uint160(to)) & _BITMASK_ADDRESS; if (toMasked == 0) _revert(MintToZeroAddress.selector); assembly { // Emit the `Transfer` event. log4( 0, // Start of data (0, since no data). 0, // End of data (0, since no data). _TRANSFER_EVENT_SIGNATURE, // Signature. 0, // `address(0)`. toMasked, // `to`. tokenId // `tokenId`. ) } ++_spotMinted; } _afterTokenTransfers(address(0), to, tokenId, 1); } /** * @dev Safely mints a single token at `tokenId`. * * Note: A spot-minted `tokenId` that has been burned can be re-minted again. * * Requirements: * * - If `to` refers to a smart contract, it must implement {IERC721Receiver-onERC721Received}. * - `tokenId` must be greater than `_sequentialUpTo()`. * - `tokenId` must not exist. * * See {_mintSpot}. * * Emits a {Transfer} event. */ function _safeMintSpot( address to, uint256 tokenId, bytes memory _data ) internal virtual { _mintSpot(to, tokenId); unchecked { if (to.code.length != 0) { uint256 currentSpotMinted = _spotMinted; if (!_checkContractOnERC721Received(address(0), to, tokenId, _data)) { _revert(TransferToNonERC721ReceiverImplementer.selector); } // This prevents reentrancy to `_safeMintSpot`. // It does not prevent reentrancy to `_safeMint`. if (_spotMinted != currentSpotMinted) revert(); } } } /** * @dev Equivalent to `_safeMintSpot(to, tokenId, '')`. */ function _safeMintSpot(address to, uint256 tokenId) internal virtual { _safeMintSpot(to, tokenId, ''); } // ============================================================= // APPROVAL OPERATIONS // ============================================================= /** * @dev Equivalent to `_approve(to, tokenId, false)`. */ function _approve(address to, uint256 tokenId) internal virtual { _approve(to, tokenId, false); } /** * @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: * * - `tokenId` must exist. * * Emits an {Approval} event. */ function _approve( address to, uint256 tokenId, bool approvalCheck ) internal virtual { address owner = ownerOf(tokenId); if (approvalCheck && _msgSenderERC721A() != owner) if (!isApprovedForAll(owner, _msgSenderERC721A())) { _revert(ApprovalCallerNotOwnerNorApproved.selector); } _tokenApprovals[tokenId].value = to; emit Approval(owner, to, tokenId); } // ============================================================= // BURN OPERATIONS // ============================================================= /** * @dev Equivalent to `_burn(tokenId, false)`. */ function _burn(uint256 tokenId) internal virtual { _burn(tokenId, false); } /** * @dev Destroys `tokenId`. * The approval is cleared when the token is burned. * * Requirements: * * - `tokenId` must exist. * * Emits a {Transfer} event. */ function _burn(uint256 tokenId, bool approvalCheck) internal virtual { uint256 prevOwnershipPacked = _packedOwnershipOf(tokenId); address from = address(uint160(prevOwnershipPacked)); (uint256 approvedAddressSlot, address approvedAddress) = _getApprovedSlotAndAddress(tokenId); if (approvalCheck) { // The nested ifs save around 20+ gas over a compound boolean condition. if (!_isSenderApprovedOrOwner(approvedAddress, from, _msgSenderERC721A())) if (!isApprovedForAll(from, _msgSenderERC721A())) _revert(TransferCallerNotOwnerNorApproved.selector); } _beforeTokenTransfers(from, address(0), tokenId, 1); // Clear approvals from the previous owner. assembly { if approvedAddress { // This is equivalent to `delete _tokenApprovals[tokenId]`. sstore(approvedAddressSlot, 0) } } // Underflow of the sender's balance is impossible because we check for // ownership above and the recipient's balance can't realistically overflow. // Counter overflow is incredibly unrealistic as `tokenId` would have to be 2**256. unchecked { // Updates: // - `balance -= 1`. // - `numberBurned += 1`. // // We can directly decrement the balance, and increment the number burned. // This is equivalent to `packed -= 1; packed += 1 << _BITPOS_NUMBER_BURNED;`. _packedAddressData[from] += (1 << _BITPOS_NUMBER_BURNED) - 1; // Updates: // - `address` to the last owner. // - `startTimestamp` to the timestamp of burning. // - `burned` to `true`. // - `nextInitialized` to `true`. _packedOwnerships[tokenId] = _packOwnershipData( from, (_BITMASK_BURNED | _BITMASK_NEXT_INITIALIZED) | _nextExtraData(from, address(0), prevOwnershipPacked) ); // If the next slot may not have been initialized (i.e. `nextInitialized == false`) . if (prevOwnershipPacked & _BITMASK_NEXT_INITIALIZED == 0) { uint256 nextTokenId = tokenId + 1; // If the next slot's address is zero and not burned (i.e. packed value is zero). if (_packedOwnerships[nextTokenId] == 0) { // If the next slot is within bounds. if (nextTokenId != _currentIndex) { // Initialize the next slot to maintain correctness for `ownerOf(tokenId + 1)`. _packedOwnerships[nextTokenId] = prevOwnershipPacked; } } } } emit Transfer(from, address(0), tokenId); _afterTokenTransfers(from, address(0), tokenId, 1); // Overflow not possible, as `_burnCounter` cannot be exceed `_currentIndex + _spotMinted` times. unchecked { _burnCounter++; } } // ============================================================= // EXTRA DATA OPERATIONS // ============================================================= /** * @dev Directly sets the extra data for the ownership data `index`. */ function _setExtraDataAt(uint256 index, uint24 extraData) internal virtual { uint256 packed = _packedOwnerships[index]; if (packed == 0) _revert(OwnershipNotInitializedForExtraData.selector); uint256 extraDataCasted; // Cast `extraData` with assembly to avoid redundant masking. assembly { extraDataCasted := extraData } packed = (packed & _BITMASK_EXTRA_DATA_COMPLEMENT) | (extraDataCasted << _BITPOS_EXTRA_DATA); _packedOwnerships[index] = packed; } /** * @dev Called during each token transfer to set the 24bit `extraData` field. * Intended to be overridden by the cosumer contract. * * `previousExtraData` - the value of `extraData` before transfer. * * Calling conditions: * * - When `from` and `to` are both non-zero, `from`'s `tokenId` will be * transferred to `to`. * - When `from` is zero, `tokenId` will be minted for `to`. * - When `to` is zero, `tokenId` will be burned by `from`. * - `from` and `to` are never both zero. */ function _extraData( address from, address to, uint24 previousExtraData ) internal view virtual returns (uint24) {} /** * @dev Returns the next extra data for the packed ownership data. * The returned result is shifted into position. */ function _nextExtraData( address from, address to, uint256 prevOwnershipPacked ) private view returns (uint256) { uint24 extraData = uint24(prevOwnershipPacked >> _BITPOS_EXTRA_DATA); return uint256(_extraData(from, to, extraData)) << _BITPOS_EXTRA_DATA; } // ============================================================= // OTHER OPERATIONS // ============================================================= /** * @dev Returns the message sender (defaults to `msg.sender`). * * If you are writing GSN compatible contracts, you need to override this function. */ function _msgSenderERC721A() internal view virtual returns (address) { return msg.sender; } /** * @dev Converts a uint256 to its ASCII string decimal representation. */ function _toString(uint256 value) internal pure virtual returns (string memory str) { assembly { // The maximum value of a uint256 contains 78 digits (1 byte per digit), but // we allocate 0xa0 bytes to keep the free memory pointer 32-byte word aligned. // We will need 1 word for the trailing zeros padding, 1 word for the length, // and 3 words for a maximum of 78 digits. Total: 5 * 0x20 = 0xa0. let m := add(mload(0x40), 0xa0) // Update the free memory pointer to allocate. mstore(0x40, m) // Assign the `str` to the end. str := sub(m, 0x20) // Zeroize the slot after the string. mstore(str, 0) // Cache the end of the memory to calculate the length later. let end := str // We write the string from rightmost digit to leftmost digit. // The following is essentially a do-while loop that also handles the zero case. // prettier-ignore for { let temp := value } 1 {} { str := sub(str, 1) // Write the character to the pointer. // The ASCII index of the '0' character is 48. mstore8(str, add(48, mod(temp, 10))) // Keep dividing `temp` until zero. temp := div(temp, 10) // prettier-ignore if iszero(temp) { break } } let length := sub(end, str) // Move the pointer 32 bytes leftwards to make room for the length. str := sub(str, 0x20) // Store the length. mstore(str, length) } } /** * @dev For more efficient reverts. */ function _revert(bytes4 errorSelector) internal pure { assembly { mstore(0x00, errorSelector) revert(0x00, 0x04) } } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v5.0.0) (utils/cryptography/MerkleProof.sol) pragma solidity ^0.8.20; /** * @dev These functions deal with verification of Merkle Tree proofs. * * The tree and the proofs can be generated using our * https://github.com/OpenZeppelin/merkle-tree[JavaScript library]. * You will find a quickstart guide in the readme. * * WARNING: You should avoid using leaf values that are 64 bytes long prior to * hashing, or use a hash function other than keccak256 for hashing leaves. * This is because the concatenation of a sorted pair of internal nodes in * the Merkle tree could be reinterpreted as a leaf value. * OpenZeppelin's JavaScript library generates Merkle trees that are safe * against this attack out of the box. */ library MerkleProof { /** *@dev The multiproof provided is not valid. */ error MerkleProofInvalidMultiproof(); /** * @dev Returns true if a `leaf` can be proved to be a part of a Merkle tree * defined by `root`. For this, a `proof` must be provided, containing * sibling hashes on the branch from the leaf to the root of the tree. Each * pair of leaves and each pair of pre-images are assumed to be sorted. */ function verify(bytes32[] memory proof, bytes32 root, bytes32 leaf) internal pure returns (bool) { return processProof(proof, leaf) == root; } /** * @dev Calldata version of {verify} */ function verifyCalldata(bytes32[] calldata proof, bytes32 root, bytes32 leaf) internal pure returns (bool) { return processProofCalldata(proof, leaf) == root; } /** * @dev Returns the rebuilt hash obtained by traversing a Merkle tree up * from `leaf` using `proof`. A `proof` is valid if and only if the rebuilt * hash matches the root of the tree. When processing the proof, the pairs * of leafs & pre-images are assumed to be sorted. */ function processProof(bytes32[] memory proof, bytes32 leaf) internal pure returns (bytes32) { bytes32 computedHash = leaf; for (uint256 i = 0; i < proof.length; i++) { computedHash = _hashPair(computedHash, proof[i]); } return computedHash; } /** * @dev Calldata version of {processProof} */ function processProofCalldata(bytes32[] calldata proof, bytes32 leaf) internal pure returns (bytes32) { bytes32 computedHash = leaf; for (uint256 i = 0; i < proof.length; i++) { computedHash = _hashPair(computedHash, proof[i]); } return computedHash; } /** * @dev Returns true if the `leaves` can be simultaneously proven to be a part of a Merkle tree defined by * `root`, according to `proof` and `proofFlags` as described in {processMultiProof}. * * CAUTION: Not all Merkle trees admit multiproofs. See {processMultiProof} for details. */ function multiProofVerify( bytes32[] memory proof, bool[] memory proofFlags, bytes32 root, bytes32[] memory leaves ) internal pure returns (bool) { return processMultiProof(proof, proofFlags, leaves) == root; } /** * @dev Calldata version of {multiProofVerify} * * CAUTION: Not all Merkle trees admit multiproofs. See {processMultiProof} for details. */ function multiProofVerifyCalldata( bytes32[] calldata proof, bool[] calldata proofFlags, bytes32 root, bytes32[] memory leaves ) internal pure returns (bool) { return processMultiProofCalldata(proof, proofFlags, leaves) == root; } /** * @dev Returns the root of a tree reconstructed from `leaves` and sibling nodes in `proof`. The reconstruction * proceeds by incrementally reconstructing all inner nodes by combining a leaf/inner node with either another * leaf/inner node or a proof sibling node, depending on whether each `proofFlags` item is true or false * respectively. * * CAUTION: Not all Merkle trees admit multiproofs. To use multiproofs, it is sufficient to ensure that: 1) the tree * is complete (but not necessarily perfect), 2) the leaves to be proven are in the opposite order they are in the * tree (i.e., as seen from right to left starting at the deepest layer and continuing at the next layer). */ function processMultiProof( bytes32[] memory proof, bool[] memory proofFlags, bytes32[] memory leaves ) internal pure returns (bytes32 merkleRoot) { // This function rebuilds the root hash by traversing the tree up from the leaves. The root is rebuilt by // consuming and producing values on a queue. The queue starts with the `leaves` array, then goes onto the // `hashes` array. At the end of the process, the last hash in the `hashes` array should contain the root of // the Merkle tree. uint256 leavesLen = leaves.length; uint256 proofLen = proof.length; uint256 totalHashes = proofFlags.length; // Check proof validity. if (leavesLen + proofLen != totalHashes + 1) { revert MerkleProofInvalidMultiproof(); } // The xxxPos values are "pointers" to the next value to consume in each array. All accesses are done using // `xxx[xxxPos++]`, which return the current value and increment the pointer, thus mimicking a queue's "pop". bytes32[] memory hashes = new bytes32[](totalHashes); uint256 leafPos = 0; uint256 hashPos = 0; uint256 proofPos = 0; // At each step, we compute the next hash using two values: // - a value from the "main queue". If not all leaves have been consumed, we get the next leaf, otherwise we // get the next hash. // - depending on the flag, either another value from the "main queue" (merging branches) or an element from the // `proof` array. for (uint256 i = 0; i < totalHashes; i++) { bytes32 a = leafPos < leavesLen ? leaves[leafPos++] : hashes[hashPos++]; bytes32 b = proofFlags[i] ? (leafPos < leavesLen ? leaves[leafPos++] : hashes[hashPos++]) : proof[proofPos++]; hashes[i] = _hashPair(a, b); } if (totalHashes > 0) { if (proofPos != proofLen) { revert MerkleProofInvalidMultiproof(); } unchecked { return hashes[totalHashes - 1]; } } else if (leavesLen > 0) { return leaves[0]; } else { return proof[0]; } } /** * @dev Calldata version of {processMultiProof}. * * CAUTION: Not all Merkle trees admit multiproofs. See {processMultiProof} for details. */ function processMultiProofCalldata( bytes32[] calldata proof, bool[] calldata proofFlags, bytes32[] memory leaves ) internal pure returns (bytes32 merkleRoot) { // This function rebuilds the root hash by traversing the tree up from the leaves. The root is rebuilt by // consuming and producing values on a queue. The queue starts with the `leaves` array, then goes onto the // `hashes` array. At the end of the process, the last hash in the `hashes` array should contain the root of // the Merkle tree. uint256 leavesLen = leaves.length; uint256 proofLen = proof.length; uint256 totalHashes = proofFlags.length; // Check proof validity. if (leavesLen + proofLen != totalHashes + 1) { revert MerkleProofInvalidMultiproof(); } // The xxxPos values are "pointers" to the next value to consume in each array. All accesses are done using // `xxx[xxxPos++]`, which return the current value and increment the pointer, thus mimicking a queue's "pop". bytes32[] memory hashes = new bytes32[](totalHashes); uint256 leafPos = 0; uint256 hashPos = 0; uint256 proofPos = 0; // At each step, we compute the next hash using two values: // - a value from the "main queue". If not all leaves have been consumed, we get the next leaf, otherwise we // get the next hash. // - depending on the flag, either another value from the "main queue" (merging branches) or an element from the // `proof` array. for (uint256 i = 0; i < totalHashes; i++) { bytes32 a = leafPos < leavesLen ? leaves[leafPos++] : hashes[hashPos++]; bytes32 b = proofFlags[i] ? (leafPos < leavesLen ? leaves[leafPos++] : hashes[hashPos++]) : proof[proofPos++]; hashes[i] = _hashPair(a, b); } if (totalHashes > 0) { if (proofPos != proofLen) { revert MerkleProofInvalidMultiproof(); } unchecked { return hashes[totalHashes - 1]; } } else if (leavesLen > 0) { return leaves[0]; } else { return proof[0]; } } /** * @dev Sorts the pair (a, b) and hashes the result. */ function _hashPair(bytes32 a, bytes32 b) private pure returns (bytes32) { return a < b ? _efficientHash(a, b) : _efficientHash(b, a); } /** * @dev Implementation of keccak256(abi.encode(a, b)) that doesn't allocate or expand memory. */ function _efficientHash(bytes32 a, bytes32 b) private pure returns (bytes32 value) { /// @solidity memory-safe-assembly assembly { mstore(0x00, a) mstore(0x20, b) value := keccak256(0x00, 0x40) } } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v5.0.0) (utils/Strings.sol) pragma solidity ^0.8.20; import {Math} from "./math/Math.sol"; import {SignedMath} from "./math/SignedMath.sol"; /** * @dev String operations. */ library Strings { bytes16 private constant HEX_DIGITS = "0123456789abcdef"; uint8 private constant ADDRESS_LENGTH = 20; /** * @dev The `value` string doesn't fit in the specified `length`. */ error StringsInsufficientHexLength(uint256 value, uint256 length); /** * @dev Converts a `uint256` to its ASCII `string` decimal representation. */ function toString(uint256 value) internal pure returns (string memory) { unchecked { uint256 length = Math.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), HEX_DIGITS)) } value /= 10; if (value == 0) break; } return buffer; } } /** * @dev Converts a `int256` to its ASCII `string` decimal representation. */ function toStringSigned(int256 value) internal pure returns (string memory) { return string.concat(value < 0 ? "-" : "", toString(SignedMath.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, Math.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) { uint256 localValue = value; bytes memory buffer = new bytes(2 * length + 2); buffer[0] = "0"; buffer[1] = "x"; for (uint256 i = 2 * length + 1; i > 1; --i) { buffer[i] = HEX_DIGITS[localValue & 0xf]; localValue >>= 4; } if (localValue != 0) { revert StringsInsufficientHexLength(value, length); } 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 bytes(a).length == bytes(b).length && keccak256(bytes(a)) == keccak256(bytes(b)); } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v5.0.0) (interfaces/IERC2981.sol) pragma solidity ^0.8.20; import {IERC165} from "../utils/introspection/IERC165.sol"; /** * @dev Interface for the NFT Royalty Standard. * * A standardized way to retrieve royalty payment information for non-fungible tokens (NFTs) to enable universal * support for royalty payments across all NFT marketplaces and ecosystem participants. */ interface IERC2981 is IERC165 { /** * @dev Returns how much royalty is owed and to whom, based on a sale price that may be denominated in any unit of * exchange. The royalty amount is denominated and should be paid in that same unit of exchange. */ function royaltyInfo( uint256 tokenId, uint256 salePrice ) external view returns (address receiver, uint256 royaltyAmount); }
// SPDX-License-Identifier: MIT // ERC721A Contracts v4.2.3 // Creator: Chiru Labs pragma solidity ^0.8.4; /** * @dev Interface of ERC721A. */ interface IERC721A { /** * The caller must own the token or be an approved operator. */ error ApprovalCallerNotOwnerNorApproved(); /** * The token does not exist. */ error ApprovalQueryForNonexistentToken(); /** * Cannot query the balance for the zero address. */ error BalanceQueryForZeroAddress(); /** * Cannot mint to the zero address. */ error MintToZeroAddress(); /** * The quantity of tokens minted must be more than zero. */ error MintZeroQuantity(); /** * The token does not exist. */ error OwnerQueryForNonexistentToken(); /** * The caller must own the token or be an approved operator. */ error TransferCallerNotOwnerNorApproved(); /** * The token must be owned by `from`. */ error TransferFromIncorrectOwner(); /** * Cannot safely transfer to a contract that does not implement the * ERC721Receiver interface. */ error TransferToNonERC721ReceiverImplementer(); /** * Cannot transfer to the zero address. */ error TransferToZeroAddress(); /** * The token does not exist. */ error URIQueryForNonexistentToken(); /** * The `quantity` minted with ERC2309 exceeds the safety limit. */ error MintERC2309QuantityExceedsLimit(); /** * The `extraData` cannot be set on an unintialized ownership slot. */ error OwnershipNotInitializedForExtraData(); /** * `_sequentialUpTo()` must be greater than `_startTokenId()`. */ error SequentialUpToTooSmall(); /** * The `tokenId` of a sequential mint exceeds `_sequentialUpTo()`. */ error SequentialMintExceedsLimit(); /** * Spot minting requires a `tokenId` greater than `_sequentialUpTo()`. */ error SpotMintTokenIdTooSmall(); /** * Cannot mint over a token that already exists. */ error TokenAlreadyExists(); /** * The feature is not compatible with spot mints. */ error NotCompatibleWithSpotMints(); // ============================================================= // STRUCTS // ============================================================= struct TokenOwnership { // The address of the owner. address addr; // Stores the start time of ownership with minimal overhead for tokenomics. uint64 startTimestamp; // Whether the token has been burned. bool burned; // Arbitrary data similar to `startTimestamp` that can be set via {_extraData}. uint24 extraData; } // ============================================================= // TOKEN COUNTERS // ============================================================= /** * @dev Returns the total number of tokens in existence. * Burned tokens will reduce the count. * To get the total number of tokens minted, please see {_totalMinted}. */ function totalSupply() external view returns (uint256); // ============================================================= // IERC165 // ============================================================= /** * @dev Returns true if this contract implements the interface defined by * `interfaceId`. See the corresponding * [EIP section](https://eips.ethereum.org/EIPS/eip-165#how-interfaces-are-identified) * to learn more about how these ids are created. * * This function call must use less than 30000 gas. */ function supportsInterface(bytes4 interfaceId) external view returns (bool); // ============================================================= // IERC721 // ============================================================= /** * @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`, * 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 be 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, bytes calldata data ) external payable; /** * @dev Equivalent to `safeTransferFrom(from, to, tokenId, '')`. */ function safeTransferFrom( address from, address to, uint256 tokenId ) external payable; /** * @dev Transfers `tokenId` 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 payable; /** * @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 payable; /** * @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); // ============================================================= // IERC721Metadata // ============================================================= /** * @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); // ============================================================= // IERC2309 // ============================================================= /** * @dev Emitted when tokens in `fromTokenId` to `toTokenId` * (inclusive) is transferred from `from` to `to`, as defined in the * [ERC2309](https://eips.ethereum.org/EIPS/eip-2309) standard. * * See {_mintERC2309} for more details. */ event ConsecutiveTransfer(uint256 indexed fromTokenId, uint256 toTokenId, address indexed from, address indexed to); }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v5.0.0) (token/ERC20/utils/SafeERC20.sol) pragma solidity ^0.8.20; import {IERC20} from "../IERC20.sol"; import {IERC20Permit} from "../extensions/IERC20Permit.sol"; import {Address} from "../../../utils/Address.sol"; /** * @title SafeERC20 * @dev Wrappers around ERC20 operations that throw on failure (when the token * contract returns false). Tokens that return no value (and instead revert or * throw on failure) are also supported, non-reverting calls are assumed to be * successful. * To use this library you can add a `using SafeERC20 for IERC20;` statement to your contract, * which allows you to call the safe operations as `token.safeTransfer(...)`, etc. */ library SafeERC20 { using Address for address; /** * @dev An operation with an ERC20 token failed. */ error SafeERC20FailedOperation(address token); /** * @dev Indicates a failed `decreaseAllowance` request. */ error SafeERC20FailedDecreaseAllowance(address spender, uint256 currentAllowance, uint256 requestedDecrease); /** * @dev Transfer `value` amount of `token` from the calling contract to `to`. If `token` returns no value, * non-reverting calls are assumed to be successful. */ function safeTransfer(IERC20 token, address to, uint256 value) internal { _callOptionalReturn(token, abi.encodeCall(token.transfer, (to, value))); } /** * @dev Transfer `value` amount of `token` from `from` to `to`, spending the approval given by `from` to the * calling contract. If `token` returns no value, non-reverting calls are assumed to be successful. */ function safeTransferFrom(IERC20 token, address from, address to, uint256 value) internal { _callOptionalReturn(token, abi.encodeCall(token.transferFrom, (from, to, value))); } /** * @dev Increase the calling contract's allowance toward `spender` by `value`. If `token` returns no value, * non-reverting calls are assumed to be successful. */ function safeIncreaseAllowance(IERC20 token, address spender, uint256 value) internal { uint256 oldAllowance = token.allowance(address(this), spender); forceApprove(token, spender, oldAllowance + value); } /** * @dev Decrease the calling contract's allowance toward `spender` by `requestedDecrease`. If `token` returns no * value, non-reverting calls are assumed to be successful. */ function safeDecreaseAllowance(IERC20 token, address spender, uint256 requestedDecrease) internal { unchecked { uint256 currentAllowance = token.allowance(address(this), spender); if (currentAllowance < requestedDecrease) { revert SafeERC20FailedDecreaseAllowance(spender, currentAllowance, requestedDecrease); } forceApprove(token, spender, currentAllowance - requestedDecrease); } } /** * @dev Set the calling contract's allowance toward `spender` to `value`. If `token` returns no value, * non-reverting calls are assumed to be successful. Meant to be used with tokens that require the approval * to be set to zero before setting it to a non-zero value, such as USDT. */ function forceApprove(IERC20 token, address spender, uint256 value) internal { bytes memory approvalCall = abi.encodeCall(token.approve, (spender, value)); if (!_callOptionalReturnBool(token, approvalCall)) { _callOptionalReturn(token, abi.encodeCall(token.approve, (spender, 0))); _callOptionalReturn(token, approvalCall); } } /** * @dev Imitates a Solidity high-level call (i.e. a regular function call to a contract), relaxing the requirement * on the return value: the return value is optional (but if data is returned, it must not be false). * @param token The token targeted by the call. * @param data The call data (encoded using abi.encode or one of its variants). */ function _callOptionalReturn(IERC20 token, bytes memory data) private { // We need to perform a low level call here, to bypass Solidity's return data size checking mechanism, since // we're implementing it ourselves. We use {Address-functionCall} to perform this call, which verifies that // the target address contains contract code and also asserts for success in the low-level call. bytes memory returndata = address(token).functionCall(data); if (returndata.length != 0 && !abi.decode(returndata, (bool))) { revert SafeERC20FailedOperation(address(token)); } } /** * @dev Imitates a Solidity high-level call (i.e. a regular function call to a contract), relaxing the requirement * on the return value: the return value is optional (but if data is returned, it must not be false). * @param token The token targeted by the call. * @param data The call data (encoded using abi.encode or one of its variants). * * This is a variant of {_callOptionalReturn} that silents catches all reverts and returns a bool instead. */ function _callOptionalReturnBool(IERC20 token, bytes memory data) private returns (bool) { // We need to perform a low level call here, to bypass Solidity's return data size checking mechanism, since // we're implementing it ourselves. We cannot use {Address-functionCall} here since this should return false // and not revert is the subcall reverts. (bool success, bytes memory returndata) = address(token).call(data); return success && (returndata.length == 0 || abi.decode(returndata, (bool))) && address(token).code.length > 0; } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v5.0.0) (token/ERC20/ERC20.sol) pragma solidity ^0.8.20; import {IERC20} from "./IERC20.sol"; import {IERC20Metadata} from "./extensions/IERC20Metadata.sol"; import {Context} from "../../utils/Context.sol"; import {IERC20Errors} from "../../interfaces/draft-IERC6093.sol"; /** * @dev Implementation of the {IERC20} interface. * * This implementation is agnostic to the way tokens are created. This means * that a supply mechanism has to be added in a derived contract using {_mint}. * * TIP: For a detailed writeup see our guide * https://forum.openzeppelin.com/t/how-to-implement-erc20-supply-mechanisms/226[How * to implement supply mechanisms]. * * The default value of {decimals} is 18. To change this, you should override * this function so it returns a different value. * * We have followed general OpenZeppelin Contracts guidelines: functions revert * instead returning `false` on failure. This behavior is nonetheless * conventional and does not conflict with the expectations of ERC20 * applications. * * Additionally, an {Approval} event is emitted on calls to {transferFrom}. * This allows applications to reconstruct the allowance for all accounts just * by listening to said events. Other implementations of the EIP may not emit * these events, as it isn't required by the specification. */ abstract contract ERC20 is Context, IERC20, IERC20Metadata, IERC20Errors { mapping(address account => uint256) private _balances; mapping(address account => mapping(address spender => uint256)) private _allowances; uint256 private _totalSupply; string private _name; string private _symbol; /** * @dev Sets the values for {name} and {symbol}. * * All two of these values are immutable: they can only be set once during * construction. */ constructor(string memory name_, string memory symbol_) { _name = name_; _symbol = symbol_; } /** * @dev Returns the name of the token. */ function name() public view virtual returns (string memory) { return _name; } /** * @dev Returns the symbol of the token, usually a shorter version of the * name. */ function symbol() public view virtual returns (string memory) { return _symbol; } /** * @dev Returns the number of decimals used to get its user representation. * For example, if `decimals` equals `2`, a balance of `505` tokens should * be displayed to a user as `5.05` (`505 / 10 ** 2`). * * Tokens usually opt for a value of 18, imitating the relationship between * Ether and Wei. This is the default value returned by this function, unless * it's overridden. * * NOTE: This information is only used for _display_ purposes: it in * no way affects any of the arithmetic of the contract, including * {IERC20-balanceOf} and {IERC20-transfer}. */ function decimals() public view virtual returns (uint8) { return 18; } /** * @dev See {IERC20-totalSupply}. */ function totalSupply() public view virtual returns (uint256) { return _totalSupply; } /** * @dev See {IERC20-balanceOf}. */ function balanceOf(address account) public view virtual returns (uint256) { return _balances[account]; } /** * @dev See {IERC20-transfer}. * * Requirements: * * - `to` cannot be the zero address. * - the caller must have a balance of at least `value`. */ function transfer(address to, uint256 value) public virtual returns (bool) { address owner = _msgSender(); _transfer(owner, to, value); return true; } /** * @dev See {IERC20-allowance}. */ function allowance(address owner, address spender) public view virtual returns (uint256) { return _allowances[owner][spender]; } /** * @dev See {IERC20-approve}. * * NOTE: If `value` is the maximum `uint256`, the allowance is not updated on * `transferFrom`. This is semantically equivalent to an infinite approval. * * Requirements: * * - `spender` cannot be the zero address. */ function approve(address spender, uint256 value) public virtual returns (bool) { address owner = _msgSender(); _approve(owner, spender, value); return true; } /** * @dev See {IERC20-transferFrom}. * * Emits an {Approval} event indicating the updated allowance. This is not * required by the EIP. See the note at the beginning of {ERC20}. * * NOTE: Does not update the allowance if the current allowance * is the maximum `uint256`. * * Requirements: * * - `from` and `to` cannot be the zero address. * - `from` must have a balance of at least `value`. * - the caller must have allowance for ``from``'s tokens of at least * `value`. */ function transferFrom(address from, address to, uint256 value) public virtual returns (bool) { address spender = _msgSender(); _spendAllowance(from, spender, value); _transfer(from, to, value); return true; } /** * @dev Moves a `value` amount of tokens from `from` to `to`. * * This internal function is equivalent to {transfer}, and can be used to * e.g. implement automatic token fees, slashing mechanisms, etc. * * Emits a {Transfer} event. * * NOTE: This function is not virtual, {_update} should be overridden instead. */ function _transfer(address from, address to, uint256 value) internal { if (from == address(0)) { revert ERC20InvalidSender(address(0)); } if (to == address(0)) { revert ERC20InvalidReceiver(address(0)); } _update(from, to, value); } /** * @dev Transfers a `value` amount of tokens from `from` to `to`, or alternatively mints (or burns) if `from` * (or `to`) is the zero address. All customizations to transfers, mints, and burns should be done by overriding * this function. * * Emits a {Transfer} event. */ function _update(address from, address to, uint256 value) internal virtual { if (from == address(0)) { // Overflow check required: The rest of the code assumes that totalSupply never overflows _totalSupply += value; } else { uint256 fromBalance = _balances[from]; if (fromBalance < value) { revert ERC20InsufficientBalance(from, fromBalance, value); } unchecked { // Overflow not possible: value <= fromBalance <= totalSupply. _balances[from] = fromBalance - value; } } if (to == address(0)) { unchecked { // Overflow not possible: value <= totalSupply or value <= fromBalance <= totalSupply. _totalSupply -= value; } } else { unchecked { // Overflow not possible: balance + value is at most totalSupply, which we know fits into a uint256. _balances[to] += value; } } emit Transfer(from, to, value); } /** * @dev Creates a `value` amount of tokens and assigns them to `account`, by transferring it from address(0). * Relies on the `_update` mechanism * * Emits a {Transfer} event with `from` set to the zero address. * * NOTE: This function is not virtual, {_update} should be overridden instead. */ function _mint(address account, uint256 value) internal { if (account == address(0)) { revert ERC20InvalidReceiver(address(0)); } _update(address(0), account, value); } /** * @dev Destroys a `value` amount of tokens from `account`, lowering the total supply. * Relies on the `_update` mechanism. * * Emits a {Transfer} event with `to` set to the zero address. * * NOTE: This function is not virtual, {_update} should be overridden instead */ function _burn(address account, uint256 value) internal { if (account == address(0)) { revert ERC20InvalidSender(address(0)); } _update(account, address(0), value); } /** * @dev Sets `value` as the allowance of `spender` over the `owner` s tokens. * * This internal function is equivalent to `approve`, and can be used to * e.g. set automatic allowances for certain subsystems, etc. * * Emits an {Approval} event. * * Requirements: * * - `owner` cannot be the zero address. * - `spender` cannot be the zero address. * * Overrides to this logic should be done to the variant with an additional `bool emitEvent` argument. */ function _approve(address owner, address spender, uint256 value) internal { _approve(owner, spender, value, true); } /** * @dev Variant of {_approve} with an optional flag to enable or disable the {Approval} event. * * By default (when calling {_approve}) the flag is set to true. On the other hand, approval changes made by * `_spendAllowance` during the `transferFrom` operation set the flag to false. This saves gas by not emitting any * `Approval` event during `transferFrom` operations. * * Anyone who wishes to continue emitting `Approval` events on the`transferFrom` operation can force the flag to * true using the following override: * ``` * function _approve(address owner, address spender, uint256 value, bool) internal virtual override { * super._approve(owner, spender, value, true); * } * ``` * * Requirements are the same as {_approve}. */ function _approve(address owner, address spender, uint256 value, bool emitEvent) internal virtual { if (owner == address(0)) { revert ERC20InvalidApprover(address(0)); } if (spender == address(0)) { revert ERC20InvalidSpender(address(0)); } _allowances[owner][spender] = value; if (emitEvent) { emit Approval(owner, spender, value); } } /** * @dev Updates `owner` s allowance for `spender` based on spent `value`. * * Does not update the allowance value in case of infinite allowance. * Revert if not enough allowance is available. * * Does not emit an {Approval} event. */ function _spendAllowance(address owner, address spender, uint256 value) internal virtual { uint256 currentAllowance = allowance(owner, spender); if (currentAllowance != type(uint256).max) { if (currentAllowance < value) { revert ERC20InsufficientAllowance(spender, currentAllowance, value); } unchecked { _approve(owner, spender, currentAllowance - value, false); } } } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v5.0.1) (utils/Context.sol) pragma solidity ^0.8.20; /** * @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; } function _contextSuffixLength() internal view virtual returns (uint256) { return 0; } }
// SPDX-License-Identifier: MIT pragma solidity ^0.8.0; // solhint-disable-next-line interface-starts-with-i interface AggregatorV3Interface { function decimals() external view returns (uint8); function description() external view returns (string memory); function version() external view returns (uint256); function getRoundData( uint80 _roundId ) external view returns (uint80 roundId, int256 answer, uint256 startedAt, uint256 updatedAt, uint80 answeredInRound); function latestRoundData() external view returns (uint80 roundId, int256 answer, uint256 startedAt, uint256 updatedAt, uint80 answeredInRound); }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v5.0.0) (utils/math/Math.sol) pragma solidity ^0.8.20; /** * @dev Standard math utilities missing in the Solidity language. */ library Math { /** * @dev Muldiv operation overflow. */ error MathOverflowedMulDiv(); enum Rounding { Floor, // Toward negative infinity Ceil, // Toward positive infinity Trunc, // Toward zero Expand // Away from zero } /** * @dev Returns the addition of two unsigned integers, with an overflow flag. */ function tryAdd(uint256 a, uint256 b) internal pure returns (bool, uint256) { unchecked { uint256 c = a + b; if (c < a) return (false, 0); return (true, c); } } /** * @dev Returns the subtraction of two unsigned integers, with an overflow flag. */ function trySub(uint256 a, uint256 b) internal pure returns (bool, uint256) { unchecked { if (b > a) return (false, 0); return (true, a - b); } } /** * @dev Returns the multiplication of two unsigned integers, with an overflow flag. */ function tryMul(uint256 a, uint256 b) internal pure returns (bool, uint256) { unchecked { // Gas optimization: this is cheaper than requiring 'a' not being zero, but the // benefit is lost if 'b' is also tested. // See: https://github.com/OpenZeppelin/openzeppelin-contracts/pull/522 if (a == 0) return (true, 0); uint256 c = a * b; if (c / a != b) return (false, 0); return (true, c); } } /** * @dev Returns the division of two unsigned integers, with a division by zero flag. */ function tryDiv(uint256 a, uint256 b) internal pure returns (bool, uint256) { unchecked { if (b == 0) return (false, 0); return (true, a / b); } } /** * @dev Returns the remainder of dividing two unsigned integers, with a division by zero flag. */ function tryMod(uint256 a, uint256 b) internal pure returns (bool, uint256) { unchecked { if (b == 0) return (false, 0); return (true, a % b); } } /** * @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 towards infinity instead * of rounding towards zero. */ function ceilDiv(uint256 a, uint256 b) internal pure returns (uint256) { if (b == 0) { // Guarantee the same behavior as in a regular Solidity division. return a / b; } // (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 = x * y; // Least significant 256 bits of the product uint256 prod1; // Most significant 256 bits of the product assembly { let mm := mulmod(x, y, not(0)) 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. if (denominator <= prod1) { revert MathOverflowedMulDiv(); } /////////////////////////////////////////////// // 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. uint256 twos = denominator & (0 - denominator); 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 (unsignedRoundsUp(rounding) && 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 * towards zero. * * 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 + (unsignedRoundsUp(rounding) && result * result < a ? 1 : 0); } } /** * @dev Return the log in base 2 of a positive value rounded towards zero. * 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 + (unsignedRoundsUp(rounding) && 1 << result < value ? 1 : 0); } } /** * @dev Return the log in base 10 of a positive value rounded towards zero. * 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 + (unsignedRoundsUp(rounding) && 10 ** result < value ? 1 : 0); } } /** * @dev Return the log in base 256 of a positive value rounded towards zero. * 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 + (unsignedRoundsUp(rounding) && 1 << (result << 3) < value ? 1 : 0); } } /** * @dev Returns whether a provided rounding mode is considered rounding up for unsigned integers. */ function unsignedRoundsUp(Rounding rounding) internal pure returns (bool) { return uint8(rounding) % 2 == 1; } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v5.0.0) (utils/introspection/IERC165.sol) pragma solidity ^0.8.20; /** * @dev Interface of the ERC165 standard, as defined in the * https://eips.ethereum.org/EIPS/eip-165[EIP]. * * Implementers can declare support of contract interfaces, which can then be * queried by others ({ERC165Checker}). * * For an implementation, see {ERC165}. */ interface IERC165 { /** * @dev Returns true if this contract implements the interface defined by * `interfaceId`. See the corresponding * https://eips.ethereum.org/EIPS/eip-165#how-interfaces-are-identified[EIP section] * to learn more about how these ids are created. * * This function call must use less than 30 000 gas. */ function supportsInterface(bytes4 interfaceId) external view returns (bool); }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v5.0.0) (utils/math/SignedMath.sol) pragma solidity ^0.8.20; /** * @dev Standard signed math utilities missing in the Solidity language. */ library SignedMath { /** * @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 (last updated v5.0.0) (utils/Address.sol) pragma solidity ^0.8.20; /** * @dev Collection of functions related to the address type */ library Address { /** * @dev The ETH balance of the account is not enough to perform the operation. */ error AddressInsufficientBalance(address account); /** * @dev There's no code at `target` (it is not a contract). */ error AddressEmptyCode(address target); /** * @dev A call to an address target failed. The target may have reverted. */ error FailedInnerCall(); /** * @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.20/security-considerations.html#use-the-checks-effects-interactions-pattern[checks-effects-interactions pattern]. */ function sendValue(address payable recipient, uint256 amount) internal { if (address(this).balance < amount) { revert AddressInsufficientBalance(address(this)); } (bool success, ) = recipient.call{value: amount}(""); if (!success) { revert FailedInnerCall(); } } /** * @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 or custom error, it is bubbled * up by this function (like regular Solidity function calls). However, if * the call reverted with no returned reason, this function reverts with a * {FailedInnerCall} error. * * 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. */ function functionCall(address target, bytes memory data) internal returns (bytes memory) { return functionCallWithValue(target, data, 0); } /** * @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`. */ function functionCallWithValue(address target, bytes memory data, uint256 value) internal returns (bytes memory) { if (address(this).balance < value) { revert AddressInsufficientBalance(address(this)); } (bool success, bytes memory returndata) = target.call{value: value}(data); return verifyCallResultFromTarget(target, success, returndata); } /** * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], * but performing a static call. */ function functionStaticCall(address target, bytes memory data) internal view returns (bytes memory) { (bool success, bytes memory returndata) = target.staticcall(data); return verifyCallResultFromTarget(target, success, returndata); } /** * @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], * but performing a delegate call. */ function functionDelegateCall(address target, bytes memory data) internal returns (bytes memory) { (bool success, bytes memory returndata) = target.delegatecall(data); return verifyCallResultFromTarget(target, success, returndata); } /** * @dev Tool to verify that a low level call to smart-contract was successful, and reverts if the target * was not a contract or bubbling up the revert reason (falling back to {FailedInnerCall}) in case of an * unsuccessful call. */ function verifyCallResultFromTarget( address target, bool success, bytes memory returndata ) internal view returns (bytes memory) { if (!success) { _revert(returndata); } else { // only check if target is a contract if the call was successful and the return data is empty // otherwise we already know that it was a contract if (returndata.length == 0 && target.code.length == 0) { revert AddressEmptyCode(target); } return returndata; } } /** * @dev Tool to verify that a low level call was successful, and reverts if it wasn't, either by bubbling the * revert reason or with a default {FailedInnerCall} error. */ function verifyCallResult(bool success, bytes memory returndata) internal pure returns (bytes memory) { if (!success) { _revert(returndata); } else { return returndata; } } /** * @dev Reverts with returndata if present. Otherwise reverts with {FailedInnerCall}. */ function _revert(bytes memory returndata) 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 FailedInnerCall(); } } }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v5.0.0) (token/ERC20/IERC20.sol) pragma solidity ^0.8.20; /** * @dev Interface of the ERC20 standard as defined in the EIP. */ interface IERC20 { /** * @dev Emitted when `value` tokens are moved from one account (`from`) to * another (`to`). * * Note that `value` may be zero. */ event Transfer(address indexed from, address indexed to, uint256 value); /** * @dev Emitted when the allowance of a `spender` for an `owner` is set by * a call to {approve}. `value` is the new allowance. */ event Approval(address indexed owner, address indexed spender, uint256 value); /** * @dev Returns the value of tokens in existence. */ function totalSupply() external view returns (uint256); /** * @dev Returns the value of tokens owned by `account`. */ function balanceOf(address account) external view returns (uint256); /** * @dev Moves a `value` amount of tokens from the caller's account to `to`. * * Returns a boolean value indicating whether the operation succeeded. * * Emits a {Transfer} event. */ function transfer(address to, uint256 value) external returns (bool); /** * @dev Returns the remaining number of tokens that `spender` will be * allowed to spend on behalf of `owner` through {transferFrom}. This is * zero by default. * * This value changes when {approve} or {transferFrom} are called. */ function allowance(address owner, address spender) external view returns (uint256); /** * @dev Sets a `value` amount of tokens as the allowance of `spender` over the * caller's tokens. * * Returns a boolean value indicating whether the operation succeeded. * * IMPORTANT: Beware that changing an allowance with this method brings the risk * that someone may use both the old and the new allowance by unfortunate * transaction ordering. One possible solution to mitigate this race * condition is to first reduce the spender's allowance to 0 and set the * desired value afterwards: * https://github.com/ethereum/EIPs/issues/20#issuecomment-263524729 * * Emits an {Approval} event. */ function approve(address spender, uint256 value) external returns (bool); /** * @dev Moves a `value` amount of tokens from `from` to `to` using the * allowance mechanism. `value` is then deducted from the caller's * allowance. * * Returns a boolean value indicating whether the operation succeeded. * * Emits a {Transfer} event. */ function transferFrom(address from, address to, uint256 value) external returns (bool); }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v5.0.0) (interfaces/draft-IERC6093.sol) pragma solidity ^0.8.20; /** * @dev Standard ERC20 Errors * Interface of the https://eips.ethereum.org/EIPS/eip-6093[ERC-6093] custom errors for ERC20 tokens. */ interface IERC20Errors { /** * @dev Indicates an error related to the current `balance` of a `sender`. Used in transfers. * @param sender Address whose tokens are being transferred. * @param balance Current balance for the interacting account. * @param needed Minimum amount required to perform a transfer. */ error ERC20InsufficientBalance(address sender, uint256 balance, uint256 needed); /** * @dev Indicates a failure with the token `sender`. Used in transfers. * @param sender Address whose tokens are being transferred. */ error ERC20InvalidSender(address sender); /** * @dev Indicates a failure with the token `receiver`. Used in transfers. * @param receiver Address to which tokens are being transferred. */ error ERC20InvalidReceiver(address receiver); /** * @dev Indicates a failure with the `spender`’s `allowance`. Used in transfers. * @param spender Address that may be allowed to operate on tokens without being their owner. * @param allowance Amount of tokens a `spender` is allowed to operate with. * @param needed Minimum amount required to perform a transfer. */ error ERC20InsufficientAllowance(address spender, uint256 allowance, uint256 needed); /** * @dev Indicates a failure with the `approver` of a token to be approved. Used in approvals. * @param approver Address initiating an approval operation. */ error ERC20InvalidApprover(address approver); /** * @dev Indicates a failure with the `spender` to be approved. Used in approvals. * @param spender Address that may be allowed to operate on tokens without being their owner. */ error ERC20InvalidSpender(address spender); } /** * @dev Standard ERC721 Errors * Interface of the https://eips.ethereum.org/EIPS/eip-6093[ERC-6093] custom errors for ERC721 tokens. */ interface IERC721Errors { /** * @dev Indicates that an address can't be an owner. For example, `address(0)` is a forbidden owner in EIP-20. * Used in balance queries. * @param owner Address of the current owner of a token. */ error ERC721InvalidOwner(address owner); /** * @dev Indicates a `tokenId` whose `owner` is the zero address. * @param tokenId Identifier number of a token. */ error ERC721NonexistentToken(uint256 tokenId); /** * @dev Indicates an error related to the ownership over a particular token. Used in transfers. * @param sender Address whose tokens are being transferred. * @param tokenId Identifier number of a token. * @param owner Address of the current owner of a token. */ error ERC721IncorrectOwner(address sender, uint256 tokenId, address owner); /** * @dev Indicates a failure with the token `sender`. Used in transfers. * @param sender Address whose tokens are being transferred. */ error ERC721InvalidSender(address sender); /** * @dev Indicates a failure with the token `receiver`. Used in transfers. * @param receiver Address to which tokens are being transferred. */ error ERC721InvalidReceiver(address receiver); /** * @dev Indicates a failure with the `operator`’s approval. Used in transfers. * @param operator Address that may be allowed to operate on tokens without being their owner. * @param tokenId Identifier number of a token. */ error ERC721InsufficientApproval(address operator, uint256 tokenId); /** * @dev Indicates a failure with the `approver` of a token to be approved. Used in approvals. * @param approver Address initiating an approval operation. */ error ERC721InvalidApprover(address approver); /** * @dev Indicates a failure with the `operator` to be approved. Used in approvals. * @param operator Address that may be allowed to operate on tokens without being their owner. */ error ERC721InvalidOperator(address operator); } /** * @dev Standard ERC1155 Errors * Interface of the https://eips.ethereum.org/EIPS/eip-6093[ERC-6093] custom errors for ERC1155 tokens. */ interface IERC1155Errors { /** * @dev Indicates an error related to the current `balance` of a `sender`. Used in transfers. * @param sender Address whose tokens are being transferred. * @param balance Current balance for the interacting account. * @param needed Minimum amount required to perform a transfer. * @param tokenId Identifier number of a token. */ error ERC1155InsufficientBalance(address sender, uint256 balance, uint256 needed, uint256 tokenId); /** * @dev Indicates a failure with the token `sender`. Used in transfers. * @param sender Address whose tokens are being transferred. */ error ERC1155InvalidSender(address sender); /** * @dev Indicates a failure with the token `receiver`. Used in transfers. * @param receiver Address to which tokens are being transferred. */ error ERC1155InvalidReceiver(address receiver); /** * @dev Indicates a failure with the `operator`’s approval. Used in transfers. * @param operator Address that may be allowed to operate on tokens without being their owner. * @param owner Address of the current owner of a token. */ error ERC1155MissingApprovalForAll(address operator, address owner); /** * @dev Indicates a failure with the `approver` of a token to be approved. Used in approvals. * @param approver Address initiating an approval operation. */ error ERC1155InvalidApprover(address approver); /** * @dev Indicates a failure with the `operator` to be approved. Used in approvals. * @param operator Address that may be allowed to operate on tokens without being their owner. */ error ERC1155InvalidOperator(address operator); /** * @dev Indicates an array length mismatch between ids and values in a safeBatchTransferFrom operation. * Used in batch transfers. * @param idsLength Length of the array of token identifiers * @param valuesLength Length of the array of token amounts */ error ERC1155InvalidArrayLength(uint256 idsLength, uint256 valuesLength); }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v5.0.0) (token/ERC20/extensions/IERC20Permit.sol) pragma solidity ^0.8.20; /** * @dev Interface of the ERC20 Permit extension allowing approvals to be made via signatures, as defined in * https://eips.ethereum.org/EIPS/eip-2612[EIP-2612]. * * Adds the {permit} method, which can be used to change an account's ERC20 allowance (see {IERC20-allowance}) by * presenting a message signed by the account. By not relying on {IERC20-approve}, the token holder account doesn't * need to send a transaction, and thus is not required to hold Ether at all. * * ==== Security Considerations * * There are two important considerations concerning the use of `permit`. The first is that a valid permit signature * expresses an allowance, and it should not be assumed to convey additional meaning. In particular, it should not be * considered as an intention to spend the allowance in any specific way. The second is that because permits have * built-in replay protection and can be submitted by anyone, they can be frontrun. A protocol that uses permits should * take this into consideration and allow a `permit` call to fail. Combining these two aspects, a pattern that may be * generally recommended is: * * ```solidity * function doThingWithPermit(..., uint256 value, uint256 deadline, uint8 v, bytes32 r, bytes32 s) public { * try token.permit(msg.sender, address(this), value, deadline, v, r, s) {} catch {} * doThing(..., value); * } * * function doThing(..., uint256 value) public { * token.safeTransferFrom(msg.sender, address(this), value); * ... * } * ``` * * Observe that: 1) `msg.sender` is used as the owner, leaving no ambiguity as to the signer intent, and 2) the use of * `try/catch` allows the permit to fail and makes the code tolerant to frontrunning. (See also * {SafeERC20-safeTransferFrom}). * * Additionally, note that smart contract wallets (such as Argent or Safe) are not able to produce permit signatures, so * contracts should have entry points that don't rely on permit. */ interface IERC20Permit { /** * @dev Sets `value` as the allowance of `spender` over ``owner``'s tokens, * given ``owner``'s signed approval. * * IMPORTANT: The same issues {IERC20-approve} has related to transaction * ordering also apply here. * * Emits an {Approval} event. * * Requirements: * * - `spender` cannot be the zero address. * - `deadline` must be a timestamp in the future. * - `v`, `r` and `s` must be a valid `secp256k1` signature from `owner` * over the EIP712-formatted function arguments. * - the signature must use ``owner``'s current nonce (see {nonces}). * * For more information on the signature format, see the * https://eips.ethereum.org/EIPS/eip-2612#specification[relevant EIP * section]. * * CAUTION: See Security Considerations above. */ function permit( address owner, address spender, uint256 value, uint256 deadline, uint8 v, bytes32 r, bytes32 s ) external; /** * @dev Returns the current nonce for `owner`. This value must be * included whenever a signature is generated for {permit}. * * Every successful call to {permit} increases ``owner``'s nonce by one. This * prevents a signature from being used multiple times. */ function nonces(address owner) external view returns (uint256); /** * @dev Returns the domain separator used in the encoding of the signature for {permit}, as defined by {EIP712}. */ // solhint-disable-next-line func-name-mixedcase function DOMAIN_SEPARATOR() external view returns (bytes32); }
// SPDX-License-Identifier: MIT // OpenZeppelin Contracts (last updated v5.0.0) (token/ERC20/extensions/IERC20Metadata.sol) pragma solidity ^0.8.20; import {IERC20} from "../IERC20.sol"; /** * @dev Interface for the optional metadata functions from the ERC20 standard. */ interface IERC20Metadata is IERC20 { /** * @dev Returns the name of the token. */ function name() external view returns (string memory); /** * @dev Returns the symbol of the token. */ function symbol() external view returns (string memory); /** * @dev Returns the decimals places of the token. */ function decimals() external view returns (uint8); }
{ "optimizer": { "enabled": true, "mode": "3" }, "viaIR": true, "evmVersion": "paris", "outputSelection": { "*": { "*": [ "abi" ] } }, "detectMissingLibraries": false, "forceEVMLA": false, "enableEraVMExtensions": false, "libraries": {} }
Contract Security Audit
- No Contract Security Audit Submitted- Submit Audit Here
Contract ABI
API[{"inputs":[{"internalType":"string","name":"_name","type":"string"},{"internalType":"string","name":"_symbol","type":"string"},{"internalType":"string","name":"_contractURI","type":"string"},{"internalType":"uint256","name":"_maxSupply","type":"uint256"},{"internalType":"uint256","name":"_publicPrice","type":"uint256"},{"internalType":"string","name":"_defaultBaseURI","type":"string"},{"internalType":"string","name":"_notRevealedURI","type":"string"},{"internalType":"address payable","name":"_withdrawalRecipientAddress","type":"address"},{"internalType":"address payable","name":"_commissionRecipientAddress","type":"address"},{"internalType":"uint256","name":"_fixedCommisionThreshold","type":"uint256"},{"internalType":"uint256","name":"_commissionPercentageIn10000","type":"uint256"},{"internalType":"address payable","name":"_defaultRoyaltyRecipient","type":"address"},{"internalType":"uint256","name":"_defaultRoyaltyPercentageIn10000","type":"uint256"}],"stateMutability":"nonpayable","type":"constructor"},{"inputs":[{"internalType":"address","name":"target","type":"address"}],"name":"AddressEmptyCode","type":"error"},{"inputs":[{"internalType":"address","name":"account","type":"address"}],"name":"AddressInsufficientBalance","type":"error"},{"inputs":[],"name":"ApprovalCallerNotOwnerNorApproved","type":"error"},{"inputs":[],"name":"ApprovalQueryForNonexistentToken","type":"error"},{"inputs":[],"name":"BalanceQueryForZeroAddress","type":"error"},{"inputs":[],"name":"FailedInnerCall","type":"error"},{"inputs":[],"name":"MintERC2309QuantityExceedsLimit","type":"error"},{"inputs":[],"name":"MintToZeroAddress","type":"error"},{"inputs":[],"name":"MintZeroQuantity","type":"error"},{"inputs":[],"name":"NotCompatibleWithSpotMints","type":"error"},{"inputs":[{"internalType":"address","name":"owner","type":"address"}],"name":"OwnableInvalidOwner","type":"error"},{"inputs":[{"internalType":"address","name":"account","type":"address"}],"name":"OwnableUnauthorizedAccount","type":"error"},{"inputs":[],"name":"OwnerQueryForNonexistentToken","type":"error"},{"inputs":[],"name":"OwnershipNotInitializedForExtraData","type":"error"},{"inputs":[{"internalType":"address","name":"token","type":"address"}],"name":"SafeERC20FailedOperation","type":"error"},{"inputs":[],"name":"SequentialMintExceedsLimit","type":"error"},{"inputs":[],"name":"SequentialUpToTooSmall","type":"error"},{"inputs":[],"name":"SpotMintTokenIdTooSmall","type":"error"},{"inputs":[],"name":"TokenAlreadyExists","type":"error"},{"inputs":[],"name":"TransferCallerNotOwnerNorApproved","type":"error"},{"inputs":[],"name":"TransferFromIncorrectOwner","type":"error"},{"inputs":[],"name":"TransferToNonERC721ReceiverImplementer","type":"error"},{"inputs":[],"name":"TransferToZeroAddress","type":"error"},{"inputs":[],"name":"URIQueryForNonexistentToken","type":"error"},{"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":true,"internalType":"uint256","name":"fromTokenId","type":"uint256"},{"indexed":false,"internalType":"uint256","name":"toTokenId","type":"uint256"},{"indexed":true,"internalType":"address","name":"from","type":"address"},{"indexed":true,"internalType":"address","name":"to","type":"address"}],"name":"ConsecutiveTransfer","type":"event"},{"anonymous":false,"inputs":[],"name":"ContractURIUpdated","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"previousOwner","type":"address"},{"indexed":true,"internalType":"address","name":"newOwner","type":"address"}],"name":"OwnershipTransferred","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"from","type":"address"},{"indexed":true,"internalType":"address","name":"to","type":"address"},{"indexed":true,"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"Transfer","type":"event"},{"inputs":[],"name":"ERC20DiscountIn10000","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"ERC20FixedPricePerToken","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"ERC20PriceFeedAddress","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"ERC20TokenAddress","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_address","type":"address"}],"name":"addToBlacklist","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_to","type":"address"},{"internalType":"uint256","name":"_mintAmount","type":"uint256"}],"name":"adminMint","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"approve","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[{"internalType":"address","name":"owner","type":"address"}],"name":"balanceOf","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address[]","name":"recipients","type":"address[]"},{"internalType":"uint256[]","name":"amounts","type":"uint256[]"}],"name":"batchAdminMint","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"","type":"address"}],"name":"blacklist","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"commissionPercentageIn10000","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"commissionRecipientAddress","outputs":[{"internalType":"address payable","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"contractURI","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"defaultRoyaltyPercentageIn10000","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"defaultRoyaltyRecipient","outputs":[{"internalType":"address payable","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"tierId","type":"uint256"}],"name":"disableTier","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"emergencyWithdraw","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"contract ERC20","name":"erc20Token","type":"address"}],"name":"emergencyWithdrawERC20","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"tierId","type":"uint256"}],"name":"enableTier","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"ethPriceFeedAddress","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"getApproved","outputs":[{"internalType":"address","name":"","type":"address"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"getLaunchpadDetails","outputs":[{"internalType":"uint256","name":"","type":"uint256"},{"internalType":"uint256","name":"","type":"uint256"},{"internalType":"uint256","name":"","type":"uint256"},{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"getLaunchpadDetailsERC20","outputs":[{"internalType":"address","name":"","type":"address"},{"internalType":"uint256","name":"","type":"uint256"},{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"tierId","type":"uint256"},{"internalType":"address","name":"user","type":"address"},{"internalType":"bytes32[]","name":"proof","type":"bytes32[]"}],"name":"getMintEligibility","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"getPublicMintEligibility","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"ethPrice","type":"uint256"}],"name":"getRequiredERC20TokensChainlink","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"tierId","type":"uint256"}],"name":"getTierDetails","outputs":[{"internalType":"bytes32","name":"merkleRoot","type":"bytes32"},{"internalType":"uint256","name":"price","type":"uint256"},{"internalType":"uint256","name":"maxMintAmount","type":"uint256"},{"internalType":"uint256","name":"saleStartTime","type":"uint256"},{"internalType":"string","name":"title","type":"string"},{"internalType":"uint256","name":"ERC20Price","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"getTierIds","outputs":[{"internalType":"uint256[]","name":"","type":"uint256[]"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"owner","type":"address"},{"internalType":"address","name":"operator","type":"address"}],"name":"isApprovedForAll","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"isRevealed","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"maxSupply","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"_mintAmount","type":"uint256"}],"name":"mint","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_mintAmount","type":"uint256"}],"name":"mintWithERC20ChainlinkPrice","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_mintAmount","type":"uint256"}],"name":"mintWithFixedERC20Price","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"name","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"notRevealedURI","outputs":[{"internalType":"string","name":"","type":"string"}],"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":"publicMaxMintAmount","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"publicPrice","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"publicSaleStartTime","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"_address","type":"address"}],"name":"removeFromBlacklist","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"renounceOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_tokenId","type":"uint256"},{"internalType":"uint256","name":"_salePrice","type":"uint256"}],"name":"royaltyInfo","outputs":[{"internalType":"address","name":"","type":"address"},{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"address","name":"from","type":"address"},{"internalType":"address","name":"to","type":"address"},{"internalType":"uint256","name":"tokenId","type":"uint256"}],"name":"safeTransferFrom","outputs":[],"stateMutability":"payable","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":"payable","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":"newURI","type":"string"}],"name":"setContractURI","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address payable","name":"_defaultRoyaltyRecipient","type":"address"},{"internalType":"uint256","name":"_defaultRoyaltyPercentageIn10000","type":"uint256"}],"name":"setDefaultRoyaltyInfo","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_erc20DiscountIn10000","type":"uint256"}],"name":"setERC20DiscountIn10000","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_ERC20PriceFeedAddress","type":"address"},{"internalType":"uint256","name":"_maxStaleness","type":"uint256"}],"name":"setERC20PriceFeedAddress","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_ERC20TokenAddress","type":"address"}],"name":"setERC20TokenAddress","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"address","name":"_ethPriceFeedAddress","type":"address"},{"internalType":"uint256","name":"_maxStaleness","type":"uint256"}],"name":"setETHPriceFeedAddress","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_erc20FixedPricePerToken","type":"uint256"}],"name":"setErc20FixedPricePerToken","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_newMaxSupply","type":"uint256"}],"name":"setMaxSupply","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"string","name":"_notRevealedURI","type":"string"}],"name":"setNotRevealedURI","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_publicMaxMintAmount","type":"uint256"}],"name":"setPublicMaxMintAmount","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_newPrice","type":"uint256"}],"name":"setPublicPrice","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_publicSaleStartTime","type":"uint256"}],"name":"setPublicSaleStartTime","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"tierId","type":"uint256"},{"internalType":"string","name":"title","type":"string"},{"internalType":"bytes32","name":"merkleRoot","type":"bytes32"},{"internalType":"uint256","name":"price","type":"uint256"},{"internalType":"uint256","name":"erc20Price","type":"uint256"},{"internalType":"uint256","name":"maxMintAmount","type":"uint256"},{"internalType":"uint256","name":"saleStartTime","type":"uint256"}],"name":"setTier","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"tierId","type":"uint256"},{"internalType":"uint256","name":"maxMintAmount","type":"uint256"}],"name":"setTierMaxMintAmount","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"tierId","type":"uint256"},{"internalType":"bytes32","name":"merkleRoot","type":"bytes32"}],"name":"setTierMerkleRoot","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"tierId","type":"uint256"},{"internalType":"uint256","name":"price","type":"uint256"},{"internalType":"uint256","name":"erc20Price","type":"uint256"}],"name":"setTierPrice","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"tierId","type":"uint256"},{"internalType":"uint256","name":"saleStartTime","type":"uint256"}],"name":"setTierSaleStartTime","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"_tokenId","type":"uint256"},{"internalType":"address payable","name":"_royaltyRecipient","type":"address"},{"internalType":"uint256","name":"_royaltyPercentage","type":"uint256"}],"name":"setTokenRoyaltyInfo","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"bool","name":"_tradingEnabled","type":"bool"}],"name":"setTradingEnabled","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"bytes4","name":"interfaceId","type":"bytes4"}],"name":"supportsInterface","outputs":[{"internalType":"bool","name":"","type":"bool"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"symbol","outputs":[{"internalType":"string","name":"","type":"string"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"}],"name":"tierIds","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"}],"name":"tiers","outputs":[{"internalType":"string","name":"title","type":"string"},{"internalType":"bytes32","name":"merkleRoot","type":"bytes32"},{"internalType":"uint256","name":"price","type":"uint256"},{"internalType":"uint256","name":"erc20Price","type":"uint256"},{"internalType":"uint128","name":"maxMintAmount","type":"uint128"},{"internalType":"uint128","name":"saleStartTime","type":"uint128"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"togglePublicSaleActive","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"toggleReveal","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"}],"name":"tokenRoyaltyPercentage","outputs":[{"internalType":"uint256","name":"","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[{"internalType":"uint256","name":"","type":"uint256"}],"name":"tokenRoyaltyRecipient","outputs":[{"internalType":"address payable","name":"","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":"result","type":"uint256"}],"stateMutability":"view","type":"function"},{"inputs":[],"name":"tradingEnabled","outputs":[{"internalType":"bool","name":"","type":"bool"}],"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":"payable","type":"function"},{"inputs":[{"internalType":"address","name":"newOwner","type":"address"}],"name":"transferOwnership","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"tierId","type":"uint256"},{"internalType":"uint256","name":"amount","type":"uint256"},{"internalType":"bytes32[]","name":"proof","type":"bytes32[]"}],"name":"whitelistMint","outputs":[],"stateMutability":"payable","type":"function"},{"inputs":[{"internalType":"uint256","name":"tierId","type":"uint256"},{"internalType":"uint256","name":"amount","type":"uint256"},{"internalType":"bytes32[]","name":"proof","type":"bytes32[]"}],"name":"whitelistMintWithERC20ChainlinkPrice","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"uint256","name":"tierId","type":"uint256"},{"internalType":"uint256","name":"amount","type":"uint256"},{"internalType":"bytes32[]","name":"proof","type":"bytes32[]"}],"name":"whitelistMintWithFixedERC20Price","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"withdraw","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[{"internalType":"contract ERC20","name":"erc20Token","type":"address"}],"name":"withdrawERC20","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"withdrawFixedCommission","outputs":[],"stateMutability":"nonpayable","type":"function"},{"inputs":[],"name":"withdrawalRecipientAddress","outputs":[{"internalType":"address payable","name":"","type":"address"}],"stateMutability":"view","type":"function"}]
Contract Creation Code
9c4d535b000000000000000000000000000000000000000000000000000000000000000001000b5d901f2091438d38d53c812b2ee6e90cc7b21967ff988204bed67ffbc60000000000000000000000000000000000000000000000000000000000000060000000000000000000000000000000000000000000000000000000000000030000000000000000000000000000000000000000000000000000000000000001a000000000000000000000000000000000000000000000000000000000000001e0000000000000000000000000000000000000000000000000000000000000022000000000000000000000000000000000000000000000000000000000000027100000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000028000000000000000000000000000000000000000000000000000000000000002a0000000000000000000000000c4abd8d2315f51c6eb95052d64ef52ba9767e373000000000000000000000000c4abd8d2315f51c6eb95052d64ef52ba9767e37300000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000c4abd8d2315f51c6eb95052d64ef52ba9767e3730000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000001441627374726163742078207a6b4d61726b65747300000000000000000000000000000000000000000000000000000000000000000000000000000000000000066162737a6b6d00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000035697066733a2f2f516d62786f5731595662646e776845686a394a3576644d4a4e37686179536e44514579695445696350794e445a77000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000035697066733a2f2f516d564e70426d6f353253317432786b4d575675457151663837415452434142484d623861787479703954746b430000000000000000000000
Deployed Bytecode
0x000200000000000200180000000000020001000000010355000000600310027000000a560030019d00000a56033001970000000100200190000007590000c13d0000008004000039000000400040043f000000040030008c0000014e0000413d000000000201043b000000e00220027000000a760020009c000001b40000c13d000000440030008c0000014e0000413d0000000002000416000000000002004b0000014e0000c13d0000000402100370000000000202043b00000a590020009c0000014e0000213d0000002304200039000000000034004b0000014e0000813d0000000404200039000000000441034f000000000404043b000a00000004001d00000a590040009c0000014e0000213d000800240020003d0000000a0200002900000005022002100000000802200029000000000032004b0000014e0000213d0000002402100370000000000202043b00000a590020009c0000014e0000213d0000002304200039000000000034004b0000014e0000813d0000000404200039000000000141034f000000000101043b000900000001001d00000a590010009c0000014e0000213d000700240020003d000000090100002900000005011002100000000701100029000000000031004b0000014e0000213d000000000100041a00000a5b011001970000000002000411000000000021004b000011c00000c13d0000000a0000006b000003230000613d000600800000003d000b00000000001d0000004a0000013d0000000b020000290000000102200039000b00000002001d0000000a0020006c000003230000813d0000000b01000029000000050110021000000008031000290000000102000367000000000332034f000000000303043b001100000003001d00000a5b0030009c0000014e0000213d0000000b04000029000000090040006c0000018b0000813d0000000701100029000000000112034f000000000301043b000000000100041a00000a5b011001970000000002000411000000000021004b000001910000c13d0000000201000039000000000101041a00000b4c011001670000000102000039000000000402041a0000000001140019000000000031001a00000ca20000413d000c00000003001d0000000001310019000000400300043d0000000c02000039000000000202041a000000000021004b0000019c0000213d000f00000003001d00000b1a0030009c000011f10000213d0000000f010000290000002002100039000000400020043f00000000000104350000000c0000006b000001ac0000613d001200000004001d000d00000002001d00000a65010000410000000000100443000000000100041400000a560010009c00000a5601008041000000c00110021000000a66011001c70000800b020000392953294e0000040f000000010020019000001dc20000613d000000000101043b001000000001001d0000001201000029000000000010043f0000000501000039000000200010043f000000000100041400000a560010009c00000a5601008041000000c00110021000000b0e011001c700008010020000392953294e0000040f00000001002001900000014e0000613d0000001002000029000000a0022002100000000c03000029000000010030008c000000000300001900000b1e03006041000000000223019f0000001103000029000000000232019f000000000101043b000000000021041b000000000030043f0000000601000039000000200010043f000000000100041400000a560010009c00000a5601008041000000c00110021000000b0e011001c700008010020000392953294e0000040f00000001002001900000014e0000613d0000000c0400002900000b1f024000d1000000000101043b000000000301041a0000000002230019000000000021041b000000110000006b000001b00000613d001000120040002d0000000001000019000000c40000013d0000001207000029000000000100041400000a560010009c00000a5601008041000000c00110021000000a5d011001c70000800d02000039000000040300003900000b200400004100000000050000190000001106000029001200000007001d295329490000040f000000010020019000000001010000390000014e0000613d0000000100100190000000b40000613d00000012070000290000000107700039000000100070006c000000b50000c13d00000001010000390000001002000029000000000021041b00000af801000041000000000010044300000011010000290000000400100443000000000100041400000a560010009c00000a5601008041000000c00110021000000af9011001c700008002020000392953294e0000040f000000010020019000001dc20000613d000000000101043b000000000001004b000000450000613d000000400700043d0000000101000039000000000201041a000e00000002001d0000000c0320006a00000064017000390000008002000039000000000021043500000b210100004100000000001704350000000401700039000000000200041100000000002104350000004401700039001000000003001d0000000000310435000000240170003900000000000104350000000f01000029000000000101043300000084027000390000000000120435000000a402700039000000000001004b0000000d06000029000000ff0000613d000000000300001900000000042300190000000005630019000000000505043300000000005404350000002003300039000000000013004b000000f80000413d0000001f0310003900000b4d0330019700000000012100190000000000010435000000a40130003900000a560010009c00000a5601008041000000600110021000000a560070009c00000a560200004100000000020740190000004002200210000000000121019f000000000200041400000a560020009c00000a5602008041000000c002200210000000000112019f0000001102000029001200000007001d295329490000040f000000120a000029000000600310027000000a5603300197000000200030008c00000020040000390000000004034019000000200640019000000000056a0019000001230000613d000000000701034f00000000080a0019000000007907043c0000000008980436000000000058004b0000011f0000c13d0000001f07400190000001300000613d000000000661034f0000000307700210000000000805043300000000087801cf000000000878022f000000000606043b0000010007700089000000000676022f00000000067601cf000000000686019f00000000006504350000000100200190000001500000613d0000001f01400039000000600210018f0000000001a20019000000000021004b0000000002000039000000010200403900000a590010009c000011f10000213d0000000100200190000011f10000c13d000000400010043f000000200030008c0000014e0000413d00000000020a043300000b22002001980000014e0000c13d00000b230220019700000b210020009c000001560000c13d000000100300002900000001033000390000000e0030006c0000000007010019000000e20000413d0000000101000039000000000101041a0000000e0010006c000000450000613d00000000010000190000295500010430000000000003004b0000015a0000c13d00000060020000390000000001020433000000000001004b000001820000c13d00000b2401000041000000000010043f00000afc0100004100002955000104300000001f0230003900000a57022001970000003f0220003900000af704200197000000400200043d0000000004420019000000000024004b0000000005000039000000010500403900000a590040009c000011f10000213d0000000100500190000011f10000c13d000000400040043f0000001f0430018f000000000632043600000a5805300198000600000006001d0000000003560019000001740000613d000000000601034f0000000607000029000000006806043c0000000007870436000000000037004b000001700000c13d000000000004004b000001530000613d000000000151034f0000000304400210000000000503043300000000054501cf000000000545022f000000000101043b0000010004400089000000000141022f00000000014101cf000000000151019f0000000000130435000001530000013d000000060200002900000a560020009c00000a5602008041000000400220021000000a560010009c00000a56010080410000006001100210000000000121019f000029550001043000000b1d01000041000000000010043f0000003201000039000000040010043f00000a75010000410000295500010430000000400100043d00000b0102000041000000000021043500000004021000390000000003000411000000000032043500000a560010009c00000a5601008041000000400110021000000a75011001c70000295500010430000000440130003900000b2702000041000000000021043500000024013000390000001502000039000000000021043500000a7201000041000000000013043500000004013000390000002002000039000000000021043500000a560030009c00000a5603008041000000400130021000000a73011001c7000029550001043000000b2601000041000000000010043f00000afc01000041000029550001043000000b2501000041000000000010043f00000afc010000410000295500010430001200000004001d00000a770020009c000001ca0000213d00000ab20020009c0000021e0000a13d00000ab30020009c000003030000213d00000ac20020009c000003dd0000a13d00000ac30020009c0000051d0000213d00000ac70020009c000008a40000613d00000ac80020009c000008a90000613d00000ac90020009c0000014e0000c13d0000000001000416000000000001004b0000014e0000c13d0000001001000039000005ca0000013d00000a780020009c000002cf0000a13d00000a790020009c000003250000213d00000a880020009c000003ea0000a13d00000a890020009c0000053a0000213d00000a8d0020009c000008c00000613d00000a8e0020009c000008cd0000613d00000a8f0020009c0000014e0000c13d000000440030008c0000014e0000413d0000000002000416000000000002004b0000014e0000c13d0000000401100370000000000201043b00000a5b0020009c0000014e0000213d000000000100041a00000a5b031001970000000001000411000000000013004b00000ca80000c13d0000001a03000039000000000103041a00000a5c01100197000000000121019f000000000013041b00000b0301000041000000800010043f000000000100041400000a560010009c00000a5601008041000000c00110021000000b04011001c72953294e0000040f000000600310027000000a5603300197000000200030008c000000200400003900000000040340190000001f0640018f000000200740019000000080057001bf000002020000613d0000008008000039000000000901034f000000009a09043c0000000008a80436000000000058004b000001fe0000c13d000000000006004b0000020f0000613d000000000771034f0000000306600210000000000805043300000000086801cf000000000868022f000000000707043b0000010006600089000000000767022f00000000066701cf000000000686019f00000000006504350000000100200190000013690000613d0000001f01400039000000600110018f00000080011001bf000000400010043f000000200030008c0000014e0000413d000000800100043d000000ff0010008c0000014e0000213d000000c00110021000000b05011001970000001a0300003900000b2a0000013d00000ad00020009c0000039f0000a13d00000ad10020009c000003f70000a13d00000ad20020009c0000055f0000213d00000ad60020009c0000090d0000613d00000ad70020009c000009120000613d00000ad80020009c0000014e0000c13d0000000001000416000000000001004b0000014e0000c13d000000000100041a00120a5b0010019b0000000001000411000000120010006c000002590000613d00000aef0100004100000000001004430000000001000412000000040010044300000020010000390000002400100443000000000100041400000a560010009c00000a5601008041000000c00110021000000af0011001c700008005020000392953294e0000040f000000010020019000001dc20000613d000000000101043b00000a5b011001970000000002000411000000000012004b000002590000613d00000aef010000410000000000100443000000000100041200000004001004430000002400000443000000000100041400000a560010009c00000a5601008041000000c00110021000000af0011001c700008005020000392953294e0000040f000000010020019000001dc20000613d000000000101043b00000a5b011001970000000002000411000000000012004b000013550000c13d0000001301000039000000000101041a001000000001001d0000001201000039000000000101041a001100000001001d00000aff01000041000000000010044300000000010004100000000400100443000000000100041400000a560010009c00000a5601008041000000c00110021000000af9011001c70000800a020000392953294e0000040f000000010020019000001dc20000613d0000001102000039000000000202041a000f00000002001d000000000101043b000e00000001001d00000aef0100004100000000001004430000000001000412000000040010044300000060010000390000002400100443000000000100041400000a560010009c00000a5601008041000000c00110021000000af0011001c700008005020000392953294e0000040f000000010020019000001dc20000613d000000000101043b0000000f0110006c00000ca20000413d0000000e0110006b00000ca20000413d000f00110010007400000ca20000413d0000000f02000029000e00100020007400000ca20000413d00000aef0100004100000000001004430000000001000412000000040010044300000040010000390000002400100443000000000100041400000a560010009c00000a5601008041000000c00110021000000af0011001c700008005020000392953294e0000040f000000010020019000001dc20000613d000000000201043b0000000e012000b90000000f04000029000000100040006c000002a10000613d0000000e031000fa000000000023004b00000ca20000c13d000027100110011a000f00000001001d000e000e0010007300000ca20000413d00000aef0100004100000000001004430000000001000412000000040010044300000020010000390000002400100443000000000100041400000a560010009c00000a5601008041000000c00110021000000af0011001c700008005020000392953294e0000040f000000010020019000001dc20000613d000000000101043b00000a5b011001970000000002000411000000000012004b000019ba0000c13d0000001301000039000000000101041a0000000e0010002a00000ca20000413d0000000e011000290000001302000039000000000012041b0000000f02000029000000110020002a00000ca20000413d0000000f0200002900000011032000290000001201000039000000000001041b000000000100041400000a560010009c00000a5601008041000000c001100210000000000003004b000019e50000c13d0000000002000411000019e90000013d00000a960020009c000003ce0000a13d00000a970020009c0000042e0000a13d00000a980020009c000005750000213d00000a9c0020009c000009200000613d00000a9d0020009c000009410000613d00000a9e0020009c0000014e0000c13d000000640030008c0000014e0000413d0000000002000416000000000002004b0000014e0000c13d0000000401100370000000000101043b001200000001001d295325f00000040f0000001201000029000000000010043f0000002001000039000000200010043f00000040020000390000000001000019295329160000040f0000000101100039000000000101041a000000000001004b0000000001000039000000010100c039295321e30000040f0000001201000029000000000010043f0000002001000039000000200010043f00000000010000190000004002000039295329160000040f00000001020003670000002403200370000000000303043b0000000204100039000000000034041b0000004402200370000000000202043b0000000301100039000000000021041b0000000001000019000029540001042e00000ab40020009c000004b40000a13d00000ab50020009c000005880000213d00000ab90020009c000009670000613d00000aba0020009c000009740000613d00000abb0020009c0000014e0000c13d0000000001000416000000000001004b0000014e0000c13d000000000100041a00000a5b021001970000000005000411000000000052004b000011c60000c13d00000a5c01100197000000000010041b000000000100041400000a560010009c00000a5601008041000000c00110021000000a5d011001c70000800d02000039000000030300003900000a5e040000410000000006000019295329490000040f00000001002001900000014e0000613d0000000001000019000029540001042e00000a7a0020009c000004c60000a13d00000a7b0020009c000005b50000213d00000a7f0020009c0000097f0000613d00000a800020009c000009e40000613d00000a810020009c0000014e0000c13d0000000001000416000000000001004b0000014e0000c13d000000000100041a00000a5b011001970000000002000411000000000012004b0000034b0000613d00000aef0100004100000000001004430000000001000412000000040010044300000020010000390000002400100443000000000100041400000a560010009c00000a5601008041000000c00110021000000af0011001c700008005020000392953294e0000040f000000010020019000001dc20000613d000000000101043b00000a5b011001970000000002000411000000000012004b000013550000c13d0000001101000039000000000101041a001200000001001d00000aef0100004100000000001004430000000001000412000000040010044300000060010000390000002400100443000000000100041400000a560010009c00000a5601008041000000c00110021000000af0011001c700008005020000392953294e0000040f000000010020019000001dc20000613d000000000101043b001100120010007400000ca20000413d00000aff01000041000000000010044300000000010004100000000400100443000000000100041400000a560010009c00000a5601008041000000c00110021000000af9011001c70000800a020000392953294e0000040f000000010020019000001dc20000613d000000000101043b000000110010006b0000037f0000a13d00000aff01000041000000000010044300000000010004100000000400100443000000000100041400000a560010009c00000a5601008041000000c00110021000000af9011001c70000800a020000392953294e0000040f000000010020019000001dc20000613d000000000101043b001100000001001d0000001102000029000000120020002a00000ca20000413d000000110200002900000012012000290000001102000039000000000012041b00000aef0100004100000000001004430000000001000412000000040010044300000020010000390000002400100443000000000100041400000a560010009c00000a5601008041000000c00110021000000af0011001c700008005020000392953294e0000040f000000010020019000001dc20000613d000000000101043b00000a5b04100197000000000100041400000a560010009c00000a5601008041000000c001100210000000110000006b000016d60000c13d0000000002040019000016da0000013d00000adf0020009c000004e30000213d00000ae60020009c000005d10000a13d00000ae70020009c000009f90000613d00000ae80020009c00000a360000613d00000ae90020009c0000014e0000c13d000000240030008c0000014e0000413d0000000002000416000000000002004b0000014e0000c13d0000000401100370000000000101043b001100000001001d000000000001004b000014e10000613d0000000101000039000000000101041a000000110010006b000014e10000813d0000001101000029001200000001001d000000000010043f0000000501000039000000200010043f000000000100041400000a560010009c00000a5601008041000000c00110021000000b0e011001c700008010020000392953294e0000040f00000001002001900000014e0000613d000000000101043b000000000101041a000000000001004b000014d50000c13d0000001201000029000000000001004b000000010110008a000003b80000c13d00000ca20000013d00000aa50020009c000004f90000213d00000aac0020009c000005e30000a13d00000aad0020009c00000a550000613d00000aae0020009c00000a5a0000613d00000aaf0020009c0000014e0000c13d0000000001000416000000000001004b0000014e0000c13d000000000100041a0000114d0000013d00000aca0020009c000005f00000a13d00000acb0020009c00000a650000613d00000acc0020009c00000a720000613d00000acd0020009c0000014e0000c13d0000000001000416000000000001004b0000014e0000c13d0000001e01000039000005ca0000013d00000a900020009c0000060c0000a13d00000a910020009c00000a820000613d00000a920020009c00000a870000613d00000a930020009c0000014e0000c13d0000000001000416000000000001004b0000014e0000c13d0000000c0100003900000c7d0000013d00000ad90020009c000006580000a13d00000ada0020009c00000aa50000613d00000adb0020009c00000aaa0000613d00000adc0020009c0000014e0000c13d000000440030008c0000014e0000413d0000000002000416000000000002004b0000014e0000c13d0000000401100370000000000101043b001200000001001d000000000010043f0000001d01000039000000200010043f000000000100041400000a560010009c00000a5601008041000000c00110021000000b0e011001c700008010020000392953294e0000040f00000001002001900000014e0000613d000000000101043b000000000101041a001100000001001d000000000001004b0000041b0000c13d0000001601000039000000000101041a001100000001001d0000001201000029000000000010043f0000001c01000039000000200010043f000000000100041400000a560010009c00000a5601008041000000c00110021000000b0e011001c700008010020000392953294e0000040f00000001002001900000014e0000613d000000000101043b000000000101041a00000a5b00100198000013750000c13d0000001501000039000013830000013d00000a9f0020009c000006680000a13d00000aa00020009c00000aaf0000613d00000aa10020009c00000aba0000613d00000aa20020009c0000014e0000c13d000000e40030008c0000014e0000413d0000000002000416000000000002004b0000014e0000c13d0000002402100370000000000202043b00000a590020009c0000014e0000213d0000002304200039000000000034004b0000014e0000813d001000040020003d0000001004100360000000000404043b001100000004001d00000a590040009c0000014e0000213d0000002404200039000f00000004001d0000001102400029000000000032004b0000014e0000213d0000000402100370000000000202043b001200000002001d0000004402100370000000000202043b000e00000002001d0000006402100370000000000202043b000b00000002001d0000008402100370000000000202043b000c00000002001d000000a402100370000000000202043b000d00000002001d000000c401100370000000000101043b000a00000001001d000000000100041a00000a5b021001970000000001000411000000000012004b00000ca80000c13d0000001201000029000000000010043f0000002001000039000000200010043f000000000100041400000a560010009c00000a5601008041000000c00110021000000b0e011001c700008010020000392953294e0000040f00000001002001900000014e0000613d000000000201043b00000001012000390000000e03000029000000000031041b000e00000002001d000000000102041a000000010010019000000001021002700000007f0220618f000900000002001d0000001f0020008c00000000020000390000000102002039000000000121013f000000010010019000000bc70000c13d0000000901000029000000200010008c000004a00000413d0000000e01000029000000000010043f000000000100041400000a560010009c00000a5601008041000000c00110021000000b2c011001c700008010020000392953294e0000040f00000001002001900000014e0000613d00000011030000290000001f023000390000000502200270000000200030008c0000000002004019000000000301043b00000009010000290000001f01100039000000050110027000000000011300190000000002230019000000000012004b000004a00000813d000000000002041b0000000102200039000000000012004b0000049c0000413d0000001101000029000000200010008c000018590000413d0000000e01000029000000000010043f000000000100041400000a560010009c00000a5601008041000000c00110021000000b2c011001c700008010020000392953294e0000040f00000001002001900000014e0000613d000000200200008a0000001102200180000000000101043b000018e20000c13d0000000003000019000018ed0000013d00000abc0020009c000006710000a13d00000abd0020009c00000ad20000613d00000abe0020009c00000ae00000613d00000abf0020009c0000014e0000c13d000000240030008c0000014e0000413d0000000001000416000000000001004b0000014e0000c13d295325f00000040f00000004010000390000000101100367000000000101043b00000b5a0000013d00000a820020009c0000067f0000a13d00000a830020009c00000ae50000613d00000a840020009c00000b360000613d00000a850020009c0000014e0000c13d000000240030008c0000014e0000413d0000000001000416000000000001004b0000014e0000c13d295325f00000040f00000004010000390000000101100367000000000101043b000000000010043f0000002001000039000000200010043f00000040020000390000000001000019295329160000040f0000000401100039000000000201041a00000aee022001c7000000000021041b0000000001000019000029540001042e00000ae00020009c000006920000a13d00000ae10020009c00000b3b0000613d00000ae20020009c00000b400000613d00000ae30020009c0000014e0000c13d000000240030008c0000014e0000413d0000000002000416000000000002004b0000014e0000c13d0000000401100370000000000101043b000000000010043f0000001d01000039000000200010043f00000040020000390000000001000019295329160000040f00000c7d0000013d00000aa60020009c000006f00000a13d00000aa70020009c00000b5e0000613d00000aa80020009c00000bb90000613d00000aa90020009c0000014e0000c13d000000240030008c0000014e0000413d0000000202000039000000000202041a00000b4c022001670000000103000039000000000303041a00000000022300190000000401100370000000000101043b001200000001001d000000000012001a00000ca20000413d00000012012000290000000c02000039000000000202041a000000000021004b000011cb0000a13d00000a7201000041000000800010043f0000002001000039000000840010043f0000001501000039000000a40010043f00000b2701000041000000c40010043f00000b2f01000041000029550001043000000ac40020009c00000bcd0000613d00000ac50020009c00000c320000613d00000ac60020009c0000014e0000c13d000000440030008c0000014e0000413d0000000002000416000000000002004b0000014e0000c13d0000000401100370000000000101043b001200000001001d00000a5b0010009c0000014e0000213d295325f00000040f0000001501000039000000000201041a00000a5c0220019700000012022001af000000000021041b00000024010000390000000101100367000000000101043b0000001602000039000000000012041b0000000001000019000029540001042e00000a8a0020009c00000c3e0000613d00000a8b0020009c00000c530000613d00000a8c0020009c0000014e0000c13d0000000001000416000000000001004b0000014e0000c13d0000001403000039000000000203041a000000010420019000000001012002700000007f0110618f0000001f0010008c00000000050000390000000105002039000000000552013f000000010050019000000bc70000c13d000000800010043f000000000004004b000012220000613d000000000030043f000000000001004b0000000002000019000012270000613d00000a68030000410000000002000019000000000403041a000000a005200039000000000045043500000001033000390000002002200039000000000012004b000005570000413d000012270000013d00000ad30020009c00000c790000613d00000ad40020009c00000c810000613d00000ad50020009c0000014e0000c13d000000000103001929531ff00000040f001200000001001d001100000002001d001000000003001d000000400100043d000f00000001001d29531f250000040f0000000f040000290000000000040435000000120100002900000011020000290000001003000029295323e20000040f0000000001000019000029540001042e00000a990020009c00000cad0000613d00000a9a0020009c00000cc20000613d00000a9b0020009c0000014e0000c13d000000240030008c0000014e0000413d0000000001000416000000000001004b0000014e0000c13d295325f00000040f00000004010000390000000101100367000000000101043b0000000d02000039000000000012041b0000000001000019000029540001042e00000ab60020009c00000d9e0000613d00000ab70020009c00000da80000613d00000ab80020009c0000014e0000c13d000000440030008c0000014e0000413d0000000002000416000000000002004b0000014e0000c13d0000000401100370000000000101043b001200000001001d295325f00000040f0000001201000029000000000010043f0000002001000039000000200010043f00000040020000390000000001000019295329160000040f0000000101100039000000000101041a000000000001004b0000000001000039000000010100c039295321e30000040f0000001201000029000000000010043f0000002001000039000000200010043f00000024010000390000000101100367000000000101043b001200000001001d00000000010000190000004002000039295329160000040f000000120200002900000080022002100000000401100039000000000301041a00000aed0330019700000ef10000013d00000a7c0020009c00000ea50000613d00000a7d0020009c00000ecb0000613d00000a7e0020009c0000014e0000c13d000000240030008c0000014e0000413d0000000002000416000000000002004b0000014e0000c13d0000000401100370000000000101043b00000a5b0010009c0000014e0000213d000000000010043f0000001f01000039000000200010043f00000040020000390000000001000019295329160000040f000000000101041a000000ff001001900000000001000039000000010100c039000000800010043f00000aec01000041000029540001042e00000aea0020009c00000ef50000613d00000aeb0020009c0000014e0000c13d000000240030008c0000014e0000413d0000000002000416000000000002004b0000014e0000c13d0000000401100370000000000101043b00000b22001001980000014e0000c13d00000b230210019700000b480020009c0000126c0000c13d0000000101000039000012790000013d00000ab00020009c00000f020000613d00000ab10020009c0000014e0000c13d000000240030008c0000014e0000413d0000000002000416000000000002004b0000014e0000c13d0000000401100370000000000101043b295321f70000040f0000116b0000013d00000ace0020009c00000f200000613d00000acf0020009c0000014e0000c13d000000240030008c0000014e0000413d0000000002000416000000000002004b0000014e0000c13d0000000401100370000000000101043b001200000001001d00000a5b0010009c0000014e0000213d295325f00000040f0000001201000029000000000010043f0000001f01000039000000200010043f00000040020000390000000001000019295329160000040f000000000301041a00000b4e0230019700000001022001bf000000000021041b0000000001000019000029540001042e00000a940020009c000010e90000613d00000a950020009c0000014e0000c13d000000640030008c0000014e0000413d0000000002000416000000000002004b0000014e0000c13d0000000402100370000000000202043b001100000002001d0000002401100370000000000101043b001000000001001d00000a5b0010009c0000014e0000213d000000000100041a00000a5b021001970000000001000411000000000012004b00000ca80000c13d000000110000006b000010f30000613d0000001101000029000000000010043f0000000501000039000000200010043f000000000100041400000a560010009c00000a5601008041000000c00110021000000b0e011001c700008010020000392953294e0000040f00000001002001900000014e0000613d000000000101043b000000000101041a000000000001004b0000064d0000c13d0000000101000039000000000101041a000000110010006c000010f30000a13d001200110000002d0000001201000029000000010110008a001200000001001d000000000010043f0000000501000039000000200010043f000000000100041400000a560010009c00000a5601008041000000c00110021000000b0e011001c700008010020000392953294e0000040f00000001002001900000014e0000613d000000000101043b000000000101041a000000000001004b0000063a0000613d00000b0f00100198000010f30000c13d00000a5b001001980000170c0000c13d000000400100043d000000440210003900000b1003000041000000000032043500000024021000390000001403000039000019820000013d00000add0020009c000010f70000613d00000ade0020009c0000014e0000c13d0000000001000416000000000001004b0000014e0000c13d0000000201000039000000000101041a00000b4c011001670000000102000039000000000202041a0000000001120019000000800010043f00000aec01000041000029540001042e00000aa30020009c000010fc0000613d00000aa40020009c0000014e0000c13d0000000001000416000000000001004b0000014e0000c13d0000000d0100003900000c7d0000013d00000ac00020009c0000113f0000613d00000ac10020009c0000014e0000c13d000000240030008c0000014e0000413d0000000002000416000000000002004b0000014e0000c13d0000000401100370000000000101043b295326000000040f00000a5b011001970000116b0000013d00000a860020009c000011510000613d00000a870020009c0000014e0000c13d000000240030008c0000014e0000413d0000000002000416000000000002004b0000014e0000c13d0000000401100370000000000101043b001200000001001d00000a5b0010009c0000014e0000213d295325f00000040f0000001701000039000000000201041a00000a5c0220019700000cbe0000013d00000ae40020009c000011720000613d00000ae50020009c0000014e0000c13d000000440030008c0000014e0000413d0000000402100370000000000202043b001100000002001d00000a5b0020009c0000014e0000213d0000002401100370000000000101043b001000000001001d000000000001004b000010f30000613d0000001001000029000000000010043f0000000501000039000000200010043f000000000100041400000a560010009c00000a5601008041000000c00110021000000b0e011001c700008010020000392953294e0000040f00000001002001900000014e0000613d000000000101043b000000000101041a000000000001004b000006cb0000c13d0000000101000039000000000101041a000000100010006c000010f30000a13d001200100000002d0000001201000029000000010110008a001200000001001d000000000010043f0000000501000039000000200010043f000000000100041400000a560010009c00000a5601008041000000c00110021000000b0e011001c700008010020000392953294e0000040f00000001002001900000014e0000613d000000000101043b000000000101041a000000000001004b000006b80000613d00000b0f00100198000010f30000c13d00120a5b0010019b0000000002000411000000120020006c000016780000c13d0000001001000029000000000010043f0000000701000039000000200010043f000000000100041400000a560010009c00000a5601008041000000c00110021000000b0e011001c700008010020000392953294e0000040f00000001002001900000014e0000613d000000110200002900000a5b06200197000000000101043b000000000201041a00000a5c02200197000000000262019f000000000021041b000000000100041400000a560010009c00000a5601008041000000c00110021000000a5d011001c70000800d02000039000000040300003900000b450400004100000012050000290000001007000029000003200000013d00000aaa0020009c000011870000613d00000aab0020009c0000014e0000c13d000000240030008c0000014e0000413d0000000002000416000000000002004b0000014e0000c13d0000000402100370000000000502043b00000a590050009c0000014e0000213d0000002302500039000000000032004b0000014e0000813d0000000406500039000000000261034f000000000202043b00000a590020009c000011f10000213d0000001f0720003900000b4d077001970000003f0770003900000b4d0770019700000af60070009c000011f10000213d00000024055000390000008007700039000000400070043f000000800020043f0000000005520019000000000035004b0000014e0000213d0000002003600039000000000331034f00000b4d052001980000001f0620018f000000a0015000390000071e0000613d000000a007000039000000000803034f000000008908043c0000000007970436000000000017004b0000071a0000c13d000000000006004b0000072b0000613d000000000353034f0000000305600210000000000601043300000000065601cf000000000656022f000000000303043b0000010005500089000000000353022f00000000035301cf000000000363019f0000000000310435000000a0012000390000000000010435000000000100041a00000a5b021001970000000001000411000000000012004b000014cb0000c13d000000800200043d00000a590020009c000011f10000213d0000001401000039000000000501041a000000010050019000000001035002700000007f0330618f0000001f0030008c00000000060000390000000106002039000000000565013f000000010050019000000bc70000c13d000000200030008c000007510000413d000000000010043f0000001f05200039000000050550027000000a670550009a000000200020008c00000a68050040410000001f03300039000000050330027000000a670330009a000000000035004b000007510000813d000000000005041b0000000105500039000000000035004b0000074d0000413d0000001f0020008c0000187b0000a13d000000000010043f00000b4d04200198000018940000c13d000000a00500003900000a6803000041000018a20000013d0000012004000039000000400040043f0000000002000416000000000002004b0000014e0000c13d0000001f0230003900000a57022001970000012002200039000000400020043f0000001f0530018f00000a580630019800000120026000390000076b0000613d000000000701034f000000007807043c0000000004840436000000000024004b000007670000c13d000000000005004b000007780000613d000000000161034f0000000304500210000000000502043300000000054501cf000000000545022f000000000101043b0000010004400089000000000141022f00000000014101cf000000000151019f0000000000120435000001a00030008c0000014e0000413d000001200400043d00000a590040009c0000014e0000213d0000001f01400039000000000031004b000000000200001900000a5a0200804100000a5a01100197000000000001004b000000000500001900000a5a0500404100000a5a0010009c000000000502c019000000000005004b0000014e0000c13d0000012001400039000000000201043300000a590020009c000011f10000213d0000001f0120003900000b4d011001970000003f0110003900000b4d01100197000000400600043d0000000005160019000f00000006001d000000000065004b0000000001000039000000010100403900000a590050009c000011f10000213d0000000100100190000011f10000c13d0000012001300039000000400050043f0000000f050000290000000005250436000e00000005001d00000140044000390000000005420019000000000015004b0000014e0000213d000000000002004b0000000e08000029000007af0000613d000000000500001900000000065800190000000007450019000000000707043300000000007604350000002005500039000000000025004b000007a80000413d0000000f0220002900000020022000390000000000020435000001400400043d00000a590040009c0000014e0000213d0000001f02400039000000000032004b000000000500001900000a5a0500804100000a5a02200197000000000002004b000000000600001900000a5a0600404100000a5a0020009c000000000605c019000000000006004b0000014e0000c13d0000012002400039000000000202043300000a590020009c000011f10000213d0000001f0520003900000b4d055001970000003f0550003900000b4d05500197000000400600043d0000000005560019000d00000006001d000000000065004b0000000006000039000000010600403900000a590050009c000011f10000213d0000000100600190000011f10000c13d000000400050043f0000000d050000290000000005250436000c00000005001d00000140044000390000000005420019000000000015004b0000014e0000213d000000000002004b0000000c08000029000007e60000613d000000000500001900000000065800190000000007450019000000000707043300000000007604350000002005500039000000000025004b000007df0000413d0000000d0220002900000020022000390000000000020435000001600400043d00000a590040009c0000014e0000213d0000001f02400039000000000032004b000000000500001900000a5a0500804100000a5a02200197000000000002004b000000000600001900000a5a0600404100000a5a0020009c000000000605c019000000000006004b0000014e0000c13d0000012002400039000000000202043300000a590020009c000011f10000213d0000001f0520003900000b4d055001970000003f0550003900000b4d05500197000000400600043d0000000005560019000b00000006001d000000000065004b0000000006000039000000010600403900000a590050009c000011f10000213d0000000100600190000011f10000c13d000000400050043f0000000b050000290000000005250436001200000005001d00000140044000390000000005420019000000000015004b0000014e0000213d000000000002004b0000081c0000613d000000000500001900000012065000290000000007450019000000000707043300000000007604350000002005500039000000000025004b000008150000413d0000000b0220002900000020022000390000000000020435000001c00400043d00000a590040009c0000014e0000213d0000001f02400039000000000032004b000000000500001900000a5a0500804100000a5a02200197000000000002004b000000000600001900000a5a0600404100000a5a0020009c000000000605c019000000000006004b0000014e0000c13d0000012002400039000000000202043300000a590020009c000011f10000213d0000001f0520003900000b4d055001970000003f0550003900000b4d05500197000000400600043d0000000005560019000a00000006001d000000000065004b0000000006000039000000010600403900000a590050009c000011f10000213d0000000100600190000011f10000c13d000001a00600043d000900000006001d000001800600043d000800000006001d000000400050043f0000000a050000290000000005250436001100000005001d00000140044000390000000005420019000000000015004b0000014e0000213d000000000002004b000008560000613d000000000500001900000011065000290000000007450019000000000707043300000000007604350000002005500039000000000025004b0000084f0000413d0000000a0220002900000020022000390000000000020435000001e00400043d00000a590040009c0000014e0000213d0000001f02400039000000000032004b000000000300001900000a5a0300804100000a5a02200197000000000002004b000000000500001900000a5a0500404100000a5a0020009c000000000503c019000000000005004b0000014e0000c13d0000012002400039000000000202043300000a590020009c000011f10000213d0000001f0320003900000b4d033001970000003f0330003900000b4d03300197000000400500043d0000000003350019000700000005001d000000000053004b0000000005000039000000010500403900000a590030009c000011f10000213d0000000100500190000011f10000c13d000000400030043f00000007030000290000000003230436001000000003001d00000140034000390000000004320019000000000014004b0000014e0000213d000000000002004b0000088c0000613d000000000100001900000010041000290000000005310019000000000505043300000000005404350000002001100039000000000021004b000008850000413d000000070120002900000020011000390000000000010435000002000100043d000600000001001d00000a5b0010009c0000014e0000213d000002200100043d000500000001001d00000a5b0010009c0000014e0000213d000002800100043d000200000001001d00000a5b0010009c0000014e0000213d0000000006000411000000000006004b00001b3e0000c13d000000400100043d00000a7402000041000000000021043500000004021000390000000000020435000001970000013d0000000001000416000000000001004b0000014e0000c13d00000015010000390000114c0000013d000000240030008c0000014e0000413d0000000002000416000000000002004b0000014e0000c13d0000000401100370000000000101043b001200000001001d00000a5b0010009c0000014e0000213d295325f00000040f0000001201000029000000000010043f0000001f01000039000000200010043f00000040020000390000000001000019295329160000040f000000000301041a00000b4e02300197000000000021041b0000000001000019000029540001042e0000000001000416000000000001004b0000014e0000c13d0000002102000039000000000102041a000000800010043f000000000020043f0000002002000039000000000001004b000011e30000c13d000000a0010000390000000004020019000012820000013d0000000001000416000000000001004b0000014e0000c13d000000000100041a00000a5b021001970000000001000411000000000012004b00000ca80000c13d00000aef0100004100000000001004430000000001000412000000040010044300000080010000390000002400100443000000000100041400000a560010009c00000a5601008041000000c00110021000000af0011001c700008005020000392953294e0000040f000000010020019000001dc20000613d000000000101043b00120b0a001000a400000ca20000813d00000a65010000410000000000100443000000000100041400000a560010009c00000a5601008041000000c00110021000000a66011001c70000800b020000392953294e0000040f000000010020019000001dc20000613d000000000101043b000000120010006c0000164c0000a13d00000aff01000041000000000010044300000000010004100000000400100443000000000100041400000a560010009c00000a5601008041000000c00110021000000af9011001c70000800a020000392953294e0000040f000000010020019000001dc20000613d000000000200041a00000a5b04200197000000000301043b000000000100041400000a560010009c00000a5601008041000000c001100210000000000003004b000017890000c13d00000000020400190000178c0000013d0000000001000416000000000001004b0000014e0000c13d000000190100003900000c7d0000013d000000240030008c0000014e0000413d0000000002000416000000000002004b0000014e0000c13d0000000401100370000000000101043b0000002102000039000000000302041a000000000031004b0000014e0000813d000000000020043f00000b2d0110009a00000c7d0000013d000000840030008c0000014e0000413d0000000402100370000000000202043b001200000002001d00000a5b0020009c0000014e0000213d0000002402100370000000000202043b001100000002001d00000a5b0020009c0000014e0000213d0000006402100370000000000402043b00000a590040009c0000014e0000213d0000002302400039000000000032004b0000014e0000813d0000000402400039000000000121034f000000000201043b0000002401400039295320020000040f00000044020000390000000102200367000000000302043b000000000401001900000012010000290000001102000029295323e20000040f0000000001000019000029540001042e000000440030008c0000014e0000413d0000000002000416000000000002004b0000014e0000c13d0000000401100370000000000101043b001200000001001d295325f00000040f0000001201000029000000000010043f0000002001000039000000200010043f00000040020000390000000001000019295329160000040f0000000101100039000000000101041a000000000001004b0000000001000039000000010100c039295321e30000040f0000001201000029000000000010043f0000002001000039000000200010043f00000024010000390000000101100367000000000101043b001200000001001d00000000010000190000004002000039295329160000040f00000001011000390000001202000029000000000021041b0000000001000019000029540001042e000000240030008c0000014e0000413d0000000001000416000000000001004b0000014e0000c13d295325f00000040f00000004010000390000000101100367000000000101043b0000000c02000039000000000012041b0000000001000019000029540001042e000000240030008c0000014e0000413d0000000002000416000000000002004b0000014e0000c13d0000000401100370000000000101043b00000a5b0010009c0000014e0000213d295321cb0000040f0000116b0000013d000000240030008c0000014e0000413d0000000002000416000000000002004b0000014e0000c13d0000000402100370000000000502043b00000a590050009c0000014e0000213d0000002302500039000000000032004b0000014e0000813d0000000406500039000000000261034f000000000202043b00000a590020009c000011f10000213d0000001f0720003900000b4d077001970000003f0770003900000b4d0770019700000af60070009c000011f10000213d00000024055000390000008007700039000000400070043f000000800020043f0000000005520019000000000035004b0000014e0000213d0000002003600039000000000331034f00000b4d052001980000001f0620018f000000a001500039000009a90000613d000000a007000039000000000803034f000000008908043c0000000007970436000000000017004b000009a50000c13d000000000006004b000009b60000613d000000000353034f0000000305600210000000000601043300000000065601cf000000000656022f000000000303043b0000010005500089000000000353022f00000000035301cf000000000363019f0000000000310435000000a0012000390000000000010435000000000100041a00000a5b021001970000000001000411000000000012004b000014cb0000c13d000000800200043d00000a590020009c000011f10000213d0000000b01000039000000000501041a000000010050019000000001035002700000007f0330618f0000001f0030008c00000000060000390000000106002039000000000565013f000000010050019000000bc70000c13d000000200030008c000009dc0000413d000000000010043f0000001f05200039000000050550027000000a6d0550009a000000200020008c00000a6e050040410000001f03300039000000050330027000000a6d0330009a000000000035004b000009dc0000813d000000000005041b0000000105500039000000000035004b000009d80000413d0000001f0020008c000017b60000a13d000000000010043f00000b4d04200198000018b70000c13d000000a00500003900000a6e03000041000018d30000013d000000240030008c0000014e0000413d0000000002000416000000000002004b0000014e0000c13d0000000401100370000000000601043b00000a5b0060009c0000014e0000213d000000000100041a00000a5b021001970000000005000411000000000052004b000011c60000c13d000000000006004b000013960000c13d00000a7401000041000000800010043f000000840000043f00000b02010000410000295500010430000000240030008c0000014e0000413d0000000002000416000000000002004b0000014e0000c13d0000000401100370000000000101043b000000000010043f0000002001000039000000200010043f00000040020000390000000001000019295329160000040f000f00000001001d29531f7c0000040f0000000f030000290000000102300039000000000202041a001200000002001d0000000202300039000000000202041a001100000002001d0000000302300039000000000202041a001000000002001d0000000402300039000000000202041a000e00000002001d000000c002000039000000400300043d000d00000003001d0000000002230436000f00000002001d000000c00230003929531fc90000040f0000000e0500002900000080025002700000000d04000029000000a003400039000000000023043500000aed025001970000008003400039000000000023043500000060024000390000001003000029000000000032043500000040024000390000001103000029000000000032043500000012020000290000000f030000290000000000230435000000000141004900000a560010009c00000a5601008041000000600110021000000a560040009c00000a56040080410000004002400210000000000121019f000029540001042e0000000001000416000000000001004b0000014e0000c13d0000000303000039000000000203041a000000010420019000000001012002700000007f0110618f0000001f0010008c00000000050000390000000105002039000000000552013f000000010050019000000bc70000c13d000000800010043f000000000004004b000012220000613d000000000030043f000000000001004b0000000002000019000012270000613d00000a60030000410000000002000019000000000403041a000000a005200039000000000045043500000001033000390000002002200039000000000012004b00000a4d0000413d000012270000013d0000000001000416000000000001004b0000014e0000c13d0000001b010000390000114c0000013d0000000001000416000000000001004b0000014e0000c13d0000000001000412001400000001001d001300000000003d000080050100003900000044030000390000000004000415000000140440008a00000adc0000013d000000240030008c0000014e0000413d0000000001000416000000000001004b0000014e0000c13d295325f00000040f00000004010000390000000101100367000000000101043b0000000e02000039000000000012041b0000000001000019000029540001042e0000000001000416000000000001004b0000014e0000c13d0000000001000412001800000001001d001700400000003d000080050100003900000044030000390000000004000415000000180440008a000000050440021000000aef020000412953292b0000040f000000800010043f00000aec01000041000029540001042e0000000001000416000000000001004b0000014e0000c13d0000000e0100003900000c7d0000013d000000240030008c0000014e0000413d0000000002000416000000000002004b0000014e0000c13d0000000401100370000000000101043b001200000001001d295328b00000040f0000001801000039000000000101041a001100000001001d0000001701000039000000000101041a00100a5b0010019c0000000001000039000000010100c039295321b70000040f000000110000006b0000000001000039000000010100c039295323ce0000040f00000011010000290000001202000029295321a90000040f000000000401001900000000020004110000000003000410000000100100002900000f1a0000013d0000000001000416000000000001004b0000014e0000c13d00000017010000390000114c0000013d000000000103001929531ff00000040f295320670000040f0000000001000019000029540001042e0000000001000416000000000001004b0000014e0000c13d00000080010000390000001b02000039000000000202041a001200000002001d00000a5b02200198000011f70000c13d0000000003000019000011fd0000013d000000240030008c0000014e0000413d0000000002000416000000000002004b0000014e0000c13d0000000401100370000000000101043b000000000200041a00000a5b032001970000000002000411000000000023004b0000120c0000c13d000027110010008c0000127d0000413d00000a7201000041000000800010043f0000002001000039000000840010043f0000001b01000039000000a40010043f00000b2e01000041000000c40010043f00000b2f0100004100002955000104300000000001000416000000000001004b0000014e0000c13d0000000001000412001600000001001d001500200000003d000080050100003900000044030000390000000004000415000000160440008a000000050440021000000aef020000412953292b0000040f0000114d0000013d0000000001000416000000000001004b0000014e0000c13d0000000f0100003900000c7d0000013d000000440030008c0000014e0000413d0000000002000416000000000002004b0000014e0000c13d0000000401100370000000000201043b00000a5b0020009c0000014e0000213d000000000100041a00000a5b031001970000000001000411000000000013004b00000ca80000c13d0000001b03000039000000000103041a00000a5c01100197000000000121019f000000000013041b00000b0301000041000000800010043f000000000100041400000a560010009c00000a5601008041000000c00110021000000b04011001c72953294e0000040f000000600310027000000a5603300197000000200030008c000000200400003900000000040340190000001f0640018f000000200740019000000080057001bf00000b0f0000613d0000008008000039000000000901034f000000009a09043c0000000008a80436000000000058004b00000b0b0000c13d000000000006004b00000b1c0000613d000000000771034f0000000306600210000000000805043300000000086801cf000000000868022f000000000707043b0000010006600089000000000767022f00000000066701cf000000000686019f00000000006504350000000100200190000013a50000613d0000001f01400039000000600110018f00000080011001bf000000400010043f000000200030008c0000014e0000413d000000800100043d000000ff0010008c0000014e0000213d000000c00110021000000b05011001970000001b03000039000000000203041a00000b0602200197000000000112019f00000024020000390000000102200367000000000202043b000000a00220021000000b0702200197000000000121019f000000000013041b0000000001000019000029540001042e0000000001000416000000000001004b0000014e0000c13d0000001a010000390000114c0000013d0000000001000416000000000001004b0000014e0000c13d000000160100003900000c7d0000013d0000000001000416000000000001004b0000014e0000c13d000000000100041a00000a5b021001970000000001000411000000000012004b00000ca80000c13d0000000f01000039000000000101041a001200000001001d00000a65010000410000000000100443000000000100041400000a560010009c00000a5601008041000000c00110021000000a66011001c70000800b020000392953294e0000040f000000010020019000001dc20000613d000000000101043b000000010200008a000000120010006c00000000010280190000000f02000039000000000012041b0000000001000019000029540001042e000000240030008c0000014e0000413d0000000002000416000000000002004b0000014e0000c13d0000000401100370000000000101043b000000000010043f0000002001000039000000200010043f000000000100041400000a560010009c00000a5601008041000000c00110021000000b0e011001c700008010020000392953294e0000040f00000001002001900000014e0000613d000000000101043b001200000001001d0000001b01000039000000000101041a001100000001001d00000a5b0210019800000b7d0000613d0000001a01000039000000000101041a000f00000001001d00100a5b0010019c000013b10000c13d000000400100043d001100000001001d00000012010000290000000301100039000000000101041a000f00000001001d0000001201000029000000000101041a000000010210019000000001031002700000007f0330618f001000000003001d0000001f0030008c00000000030000390000000103002039000000000331013f000000010030019000000bc70000c13d00000012040000290000000403400039000000000303041a000b00000003001d0000000203400039000000000303041a000d00000003001d0000000103400039000000000303041a000c00000003001d000000110300002900000010040000290000000003430436000e00000003001d000000000002004b000014920000613d0000001201000029000000000010043f000000000100041400000a560010009c00000a5601008041000000c00110021000000b2c011001c700008010020000392953294e0000040f00000001002001900000014e0000613d0000001006000029000000000006004b00000000020000190000000e05000029000014980000613d000000000101043b00000000020000190000000003250019000000000401041a000000000043043500000001011000390000002002200039000000000062004b00000bb10000413d000014980000013d0000000001000416000000000001004b0000014e0000c13d0000000403000039000000000203041a000000010420019000000001012002700000007f0110618f0000001f0010008c00000000050000390000000105002039000000000552013f0000000100500190000012110000613d00000b1d01000041000000000010043f0000002201000039000000040010043f00000a75010000410000295500010430000000240030008c0000014e0000413d0000000002000416000000000002004b0000014e0000c13d0000000402100370000000000502043b00000a590050009c0000014e0000213d0000002302500039000000000032004b0000014e0000813d0000000406500039000000000261034f000000000202043b00000a590020009c000011f10000213d0000001f0720003900000b4d077001970000003f0770003900000b4d0770019700000af60070009c000011f10000213d00000024055000390000008007700039000000400070043f000000800020043f0000000005520019000000000035004b0000014e0000213d0000002003600039000000000331034f00000b4d052001980000001f0620018f000000a00150003900000bf70000613d000000a007000039000000000803034f000000008908043c0000000007970436000000000017004b00000bf30000c13d000000000006004b00000c040000613d000000000353034f0000000305600210000000000601043300000000065601cf000000000656022f000000000303043b0000010005500089000000000353022f00000000035301cf000000000363019f0000000000310435000000a0012000390000000000010435000000000100041a00000a5b021001970000000001000411000000000012004b000014cb0000c13d000000800200043d00000a590020009c000011f10000213d0000000a01000039000000000501041a000000010050019000000001035002700000007f0330618f0000001f0030008c00000000060000390000000106002039000000000565013f000000010050019000000bc70000c13d000000200030008c00000c2a0000413d000000000010043f0000001f05200039000000050550027000000a6a0550009a000000200020008c00000a6b050040410000001f03300039000000050330027000000a6a0330009a000000000035004b00000c2a0000813d000000000005041b0000000105500039000000000035004b00000c260000413d0000001f0020008c000017b60000a13d000000000010043f00000b4d04200198000018c50000c13d000000a00500003900000a6b03000041000018d30000013d0000000001000416000000000001004b0000014e0000c13d295325f00000040f0000001001000039000000000201041a00000b4e03200197000000ff0020019000000001033061bf000000000031041b0000000001000019000029540001042e0000000001000416000000000001004b0000014e0000c13d0000000f01000039000000000101041a0000000102000039000000000202041a0000000203000039000000000303041a0000000d04000039000000000404041a0000000c05000039000000000505041a000000800050043f000000a00040043f00000b4c033001670000000002320019000000c00020043f000000e00010043f00000b0901000041000029540001042e000000440030008c0000014e0000413d0000000002000416000000000002004b0000014e0000c13d0000000402100370000000000202043b001200000002001d00000a5b0020009c0000014e0000213d0000002401100370000000000301043b000000000100041a00000a5b021001970000000001000411000000000012004b00000ca80000c13d0000000201000039000000000101041a00000b4c011001670000000102000039000000000202041a0000000001120019000000000031001a00000ca20000413d001100000003001d00000000013100190000000c02000039000000000202041a000000000021004b0000000001000039000000010100a039295325dc0000040f00000012010000290000001102000029295327760000040f0000000001000019000029540001042e0000000001000416000000000001004b0000014e0000c13d0000001801000039000000000101041a000000800010043f00000aec01000041000029540001042e000000240030008c0000014e0000413d0000000002000416000000000002004b0000014e0000c13d0000000401100370000000000101043b001200000001001d00000a5b0010009c0000014e0000213d000000000100041a00000a5b021001970000000001000411000000000012004b00000ca80000c13d00000aef0100004100000000001004430000000001000412000000040010044300000080010000390000002400100443000000000100041400000a560010009c00000a5601008041000000c00110021000000af0011001c700008005020000392953294e0000040f000000010020019000001dc20000613d000000000101043b00110b0a001000a4000014e50000413d00000b1d01000041000000000010043f0000001101000039000000040010043f00000a7501000041000029550001043000000b0102000041000000800020043f000000840010043f00000b02010000410000295500010430000000240030008c0000014e0000413d0000000002000416000000000002004b0000014e0000c13d0000000401100370000000000201043b000000000002004b0000000001000039000000010100c039001200000002001d000000000012004b0000014e0000c13d295325f00000040f0000001e01000039000000000201041a00000b4e0220019700000012022001af000000000021041b0000000001000019000029540001042e000000640030008c0000014e0000413d0000002402100370000000000202043b000f00000002001d0000000402100370000000000202043b0000004404100370000000000404043b00000a590040009c0000014e0000213d0000002305400039000000000035004b0000014e0000813d000e00040040003d0000000e01100360000000000101043b001200000001001d00000a590010009c0000014e0000213d000000240140003900000012040000290000000504400210000c00000001001d001100000004001d000d00000014001d0000000d0030006b0000014e0000213d000000000020043f0000002001000039000000200010043f000000000100041400000a560010009c00000a5601008041000000c00110021000000b0e011001c700008010020000392953294e0000040f00000001002001900000014e0000613d000000000101043b000b00000001001d0000000101100039000000000101041a000a00000001001d000000000001004b000011b90000613d0000000b010000290000000401100039000000000101041a000900000001001d00000a65010000410000000000100443000000000100041400000a560010009c00000a5601008041000000c00110021000000a66011001c70000800b020000392953294e0000040f000000010020019000001dc20000613d000000400200043d00000009030000290000008003300270000000000101043b000000000031004b000016680000413d0000001401000039000000000112043600000000030004110000006003300210000000000031043500000b280020009c000011f10000213d0000004003200039000000400030043f00000a560010009c00000a56010080410000004001100210000000000202043300000a560020009c00000a56020080410000006002200210000000000112019f000000000200041400000a560020009c00000a5602008041000000c002200210000000000112019f00000a5d011001c700008010020000392953294e0000040f00000001002001900000014e0000613d000000000101043b00000011020000290000003f0220003900000b2902200197000000400300043d0000000002230019001100000003001d000000000032004b0000000003000039000000010300403900000a590020009c000011f10000213d0000000100300190000011f10000c13d000000400020043f000000120200002900000011040000290000000002240436001000000002001d0000000d02000029000000000020007c0000014e0000213d000000120000006b00000d660000613d0000000e020000290000002002200039000000010220036700000011030000290000000c050000290000000d060000290000002003300039000000002402043c00000000004304350000002005500039000000000065004b00000d3f0000413d00000011020000290000000002020433000000000002004b00000d660000613d001200000000001d0000001202000029000000050220021000000010022000290000000002020433000000000021004b00000d540000813d000000000010043f000000200020043f000000000100041400000d570000013d000000000020043f000000200010043f000000000100041400000a560010009c00000a5601008041000000c00110021000000b0e011001c700008010020000392953294e0000040f00000001002001900000014e0000613d000000000101043b0000001203000029001200010030003d00000011020000290000000002020433000000120020006b00000d4a0000413d0000000a0010006c000019a40000c13d000000000100041100000a5b01100197001200000001001d000000000010043f0000000b010000290000000501100039001100000001001d000000200010043f000000000100041400000a560010009c00000a5601008041000000c00110021000000b0e011001c700008010020000392953294e0000040f00000001002001900000014e0000613d000000090200002900000aed02200197000000000101043b000000000101041a000000000112004b00000ca20000413d0000000f0010006c000018550000413d0000000201000039000000000101041a00000b4c011001670000000102000039000000000202041a00000000011200190000000f0010002a00000ca20000413d0000000f011000290000000c02000039000000000202041a000000000021004b00001ac80000213d0000000b010000290000000201100039000000000201041a0000000f012000b90000000f0000006b00000d970000613d0000000f031000fa000000000023004b00000ca20000c13d0000000002000416000000000012004b00001afd0000813d000000400100043d000000440210003900000b2a03000041000016270000013d0000000001000416000000000001004b0000014e0000c13d29531f420000040f0000002002000039000000400300043d001200000003001d000000000223043629531fc90000040f0000122e0000013d000000640030008c0000014e0000413d0000000002000416000000000002004b0000014e0000c13d0000002402100370000000000202043b000f00000002001d00000a5b0020009c0000014e0000213d0000004402100370000000000202043b00000a590020009c0000014e0000213d0000002304200039000000000034004b0000014e0000813d000e00040020003d0000000e04100360000000000404043b001200000004001d00000a590040009c0000014e0000213d000000240220003900000012040000290000000504400210000c00000002001d001100000004001d000d00000024001d0000000d0030006b0000014e0000213d0000000401100370000000000101043b000b00000001001d000000000010043f0000002001000039000000200010043f000000000100041400000a560010009c00000a5601008041000000c00110021000000b0e011001c700008010020000392953294e0000040f00000001002001900000014e0000613d000000000101043b0000000101100039000000000101041a000a00000001001d000000400100043d000000140200003900000000022104360000000f030000290000006003300210000000000032043500000b280010009c000011f10000213d0000004003100039000000400030043f00000a560020009c00000a56020080410000004002200210000000000101043300000a560010009c00000a56010080410000006001100210000000000121019f000000000200041400000a560020009c00000a5602008041000000c002200210000000000112019f00000a5d011001c700008010020000392953294e0000040f00000001002001900000014e0000613d000000000101043b00000011020000290000003f0220003900000b2902200197000000400300043d0000000002230019001100000003001d000000000032004b0000000003000039000000010300403900000a590020009c000011f10000213d0000000100300190000011f10000c13d000000400020043f000000110200002900000012040000290000000002420436001000000002001d0000000d02000029000000000020007c0000014e0000213d000000120000006b00000e3b0000613d0000000e020000290000002002200039000000010220036700000011030000290000000c050000290000000d060000290000002003300039000000002402043c00000000004304350000002005500039000000000065004b00000e140000413d00000011020000290000000002020433000000000002004b00000e3b0000613d0000000003000019001200000003001d000000050230021000000010022000290000000002020433000000000021004b00000e290000813d000000000010043f000000200020043f000000000100041400000e2c0000013d000000000020043f000000200010043f000000000100041400000a560010009c00000a5601008041000000c00110021000000b0e011001c700008010020000392953294e0000040f00000001002001900000014e0000613d000000000101043b0000001203000029000000010330003900000011020000290000000002020433000000000023004b00000e1f0000413d0000000a0010006c00000000010000190000116b0000c13d0000000b01000029000000000010043f0000002001000039000000200010043f000000000100041400000a560010009c00000a5601008041000000c00110021000000b0e011001c700008010020000392953294e0000040f00000001002001900000014e0000613d000000000101043b0000000f0200002900000a5b02200197001200000002001d000000000020043f0000000501100039000000200010043f000000000100041400000a560010009c00000a5601008041000000c00110021000000b0e011001c700008010020000392953294e0000040f00000001002001900000014e0000613d000000000101043b000000000101041a001100000001001d0000000b01000029000000000010043f0000002001000039000000200010043f000000000100041400000a560010009c00000a5601008041000000c00110021000000b0e011001c700008010020000392953294e0000040f00000001002001900000014e0000613d000000000101043b0000000401100039000000000101041a00000aed01100197000000110010006b00000000010000190000116b0000813d0000000b01000029000000000010043f0000002001000039000000200010043f000000000100041400000a560010009c00000a5601008041000000c00110021000000b0e011001c700008010020000392953294e0000040f00000001002001900000014e0000613d000000000101043b0000000401100039000000000101041a001100000001001d0000000b01000029000000000010043f0000002001000039000000200010043f000000000100041400000a560010009c00000a5601008041000000c00110021000000b0e011001c700008010020000392953294e0000040f00000001002001900000014e0000613d000000000101043b0000001202000029000000000020043f0000000501100039000000200010043f000000000100041400000a560010009c00000a5601008041000000c00110021000000b0e011001c700008010020000392953294e0000040f00000001002001900000014e0000613d000000110200002900000aed02200197000000000101043b000000000101041a000000000112004b00000ca20000413d0000116b0000013d000000240030008c0000014e0000413d0000000002000416000000000002004b0000014e0000c13d0000000401100370000000000101043b001100000001001d00000a5b0010009c0000014e0000213d000000000100041a00000a5b011001970000000002000411000000000012004b000015500000613d00000aef0100004100000000001004430000000001000412000000040010044300000020010000390000002400100443000000000100041400000a560010009c00000a5601008041000000c00110021000000af0011001c700008005020000392953294e0000040f000000010020019000001dc20000613d000000000101043b00000a5b011001970000000002000411000000000012004b0000153b0000c13d000000400100043d001200000001001d000015500000013d000000440030008c0000014e0000413d0000000002000416000000000002004b0000014e0000c13d0000000401100370000000000101043b001200000001001d295325f00000040f0000001201000029000000000010043f0000002001000039000000200010043f00000040020000390000000001000019295329160000040f0000000101100039000000000101041a000000000001004b0000000001000039000000010100c039295321e30000040f0000001201000029000000000010043f0000002001000039000000200010043f00000024010000390000000101100367000000000101043b001200000001001d00000000010000190000004002000039295329160000040f000000120200002900000aed022001970000000401100039000000000301041a00000aee03300197000000000223019f000000000021041b0000000001000019000029540001042e000000240030008c0000014e0000413d0000000001000416000000000001004b0000014e0000c13d295325f00000040f00000004010000390000000101100367000000000101043b0000001802000039000000000012041b0000000001000019000029540001042e000000240030008c0000014e0000413d0000000002000416000000000002004b0000014e0000c13d0000000401100370000000000101043b001200000001001d295328b00000040f0000001701000039000000000101041a00110a5b0010019c0000000001000039000000010100c039295321b70000040f0000000d01000039000000000101041a0000001202000029295321a90000040f295321f70000040f0000000004010019000000000200041100000000030004100000001101000029295326d40000040f00000000010004110000001202000029295327760000040f0000000001000019000029540001042e000000640030008c0000014e0000413d0000000002000416000000000002004b0000014e0000c13d0000002402100370000000000202043b000f00000002001d0000000402100370000000000202043b0000004404100370000000000404043b00000a590040009c0000014e0000213d0000002305400039000000000035004b0000014e0000813d0000000405400039000000000151034f000000000101043b001200000001001d00000a590010009c0000014e0000213d000000240140003900000012040000290000000504400210000d00000001001d001100000004001d000e00000014001d0000000e0030006b0000014e0000213d000000000020043f0000002001000039000000200010043f000000000100041400000a560010009c00000a5601008041000000c00110021000000b0e011001c700008010020000392953294e0000040f00000001002001900000014e0000613d000000000101043b000c00000001001d0000000101100039000000000101041a000b00000001001d000000000001004b000011b90000613d0000000c010000290000000401100039000000000101041a000a00000001001d00000a65010000410000000000100443000000000100041400000a560010009c00000a5601008041000000c00110021000000a66011001c70000800b020000392953294e0000040f000000010020019000001dc20000613d000000400200043d0000000a030000290000008003300270000000000101043b000000000031004b000016680000413d0000001401000039000000000112043600000000030004110000006003300210000000000031043500000b280020009c000011f10000213d0000004003200039000000400030043f00000a560010009c00000a56010080410000004001100210000000000202043300000a560020009c00000a56020080410000006002200210000000000112019f000000000200041400000a560020009c00000a5602008041000000c002200210000000000112019f00000a5d011001c700008010020000392953294e0000040f00000001002001900000014e0000613d000000000101043b00000011020000290000003f0220003900000b2902200197000000400300043d0000000002230019001100000003001d000000000032004b0000000003000039000000010300403900000a590020009c000011f10000213d0000000100300190000011f10000c13d000000400020043f000000120200002900000011040000290000000002240436001000000002001d0000000e02000029000000000020007c0000014e0000213d000000120000006b00000fc50000613d0000000d05000029000000010250036700000011030000290000000e060000290000002003300039000000002402043c00000000004304350000002005500039000000000065004b00000f9e0000413d00000011020000290000000002020433000000000002004b00000fc50000613d001200000000001d0000001202000029000000050220021000000010022000290000000002020433000000000021004b00000fb30000813d000000000010043f000000200020043f000000000100041400000fb60000013d000000000020043f000000200010043f000000000100041400000a560010009c00000a5601008041000000c00110021000000b0e011001c700008010020000392953294e0000040f00000001002001900000014e0000613d000000000101043b0000001203000029001200010030003d00000011020000290000000002020433000000120020006b00000fa90000413d0000000b0010006c000019a40000c13d000000000100041100000a5b01100197001200000001001d000000000010043f0000000c010000290000000501100039001100000001001d000000200010043f000000000100041400000a560010009c00000a5601008041000000c00110021000000b0e011001c700008010020000392953294e0000040f00000001002001900000014e0000613d0000000a0200002900000aed02200197000000000101043b000000000101041a000000000112004b00000ca20000413d0000000f0010006c000018550000413d0000000201000039000000000101041a00000b4c011001670000000102000039001000000002001d000000000202041a00000000011200190000000f0010002a00000ca20000413d0000000f011000290000000c02000039000000000202041a000000000021004b00001ac80000213d0000001701000039000000000101041a00000a5b0010019800001ad30000613d0000000c010000290000000201100039000000000101041a000d000f001000bd000000000001004b00000ffb0000613d0000000d011000f90000000f0010006c00000ca20000c13d000000400100043d000e00000001001d0000001b01000039000000000101041a000c00000001001d00000a5b0210019800001b2b0000613d0000001a01000039000000000101041a000b00000001001d000a0a5b0010019c00001b2b0000613d00000b30010000410000000e030000290000000001130436000900000001001d00000a560030009c00000a560100004100000000010340190000004001100210000000000300041400000a560030009c00000a5603008041000000c003300210000000000113019f00000afc011001c72953294e0000040f000000600310027000000a5603300197000000a00030008c000000a00400003900000000040340190000001f0640018f000000e0074001900000000e05700029000010250000613d000000000801034f0000000e09000029000000008a08043c0000000009a90436000000000059004b000010210000c13d000000000006004b000010320000613d000000000771034f0000000306600210000000000805043300000000086801cf000000000868022f000000000707043b0000010006600089000000000767022f00000000066701cf000000000686019f0000000000650435000000010020019000001b320000613d0000001f01400039000001e00210018f0000000e01200029000000000021004b0000000002000039000000010200403900000a590010009c000011f10000213d0000000100200190000011f10000c13d000000400010043f000000a00030008c0000014e0000413d0000000e02000029000000000202043300000b310020009c0000014e0000213d0000000e020000290000008002200039000000000202043300000b310020009c0000014e0000213d00000009020000290000000002020433000900000002001d00000afa0020009c0000197d0000213d000000090000006b0000197d0000613d0000000e0100002900000060011000390000000001010433000e00000001001d00000a65010000410000000000100443000000000100041400000a560010009c00000a5601008041000000c00110021000000a66011001c70000800b020000392953294e0000040f000000010020019000001dc20000613d0000000c02000029000000a00220027000000a5602200197000000000101043b000000000121004b00000ca20000413d0000000e0010006b0000197c0000a13d000000400200043d000800000002001d00000b30010000410000000001120436000e00000001001d00000a560020009c00000a560100004100000000010240190000004001100210000000000200041400000a560020009c00000a5602008041000000c002200210000000000112019f00000afc011001c70000000a020000292953294e0000040f000000600310027000000a5603300197000000a00030008c000000a00400003900000000040340190000001f0640018f000000e007400190000000080b0000290000000805700029000010890000613d000000000801034f00000000090b0019000000008a08043c0000000009a90436000000000059004b000010850000c13d000000000006004b000010960000613d000000000771034f0000000306600210000000000805043300000000086801cf000000000868022f000000000707043b0000010006600089000000000767022f00000000066701cf000000000686019f0000000000650435000000010020019000001b7f0000613d0000001f01400039000001e00210018f0000000001b20019000000000021004b0000000002000039000000010200403900000a590010009c000011f10000213d0000000100200190000011f10000c13d000000400010043f000000a00030008c0000014e0000413d00000000020b043300000b310020009c0000014e0000213d0000008002b00039000000000202043300000b310020009c0000014e0000213d0000000e020000290000000002020433000a00000002001d00000afa0020009c0000197d0000213d0000000a0000006b0000197d0000613d0000006001b000390000000001010433000e00000001001d00000a65010000410000000000100443000000000100041400000a560010009c00000a5601008041000000c00110021000000a66011001c70000800b020000392953294e0000040f000000010020019000001dc20000613d0000000b02000029000000a00220027000000a5602200197000000000101043b000000000121004b00000ca20000413d0000000e0010006b0000197c0000a13d0000000c01000029000000c00110027000000a5602100197000000120120008900000a560010009c00000ca20000213d000000120020008c000010da0000613d0000000a02000039001000010000003d000000010010019000000000032200a9000000010200603900100010002000bd00000001011002720000000002030019000010d30000c13d0000000901000029000e0010001000bd0000000e011000f9000000100010006c00000ca20000c13d0000000b01000029000000c00110027000000a5602100197000000120120008900000a560010009c00000ca20000213d000000120020008c00001b8b0000c13d000000010200003900001b940000013d000000240030008c0000014e0000413d0000000002000416000000000002004b0000014e0000c13d0000000401100370000000000101043b001100000001001d000000000001004b000012380000c13d00000b4601000041000000000010043f00000afc0100004100002955000104300000000001000416000000000001004b0000014e0000c13d2953204a0000040f0000116b0000013d000000440030008c0000014e0000413d0000000002000416000000000002004b0000014e0000c13d0000000402100370000000000202043b001200000002001d00000a5b0020009c0000014e0000213d0000002401100370000000000201043b000000000002004b0000000001000039000000010100c039001100000002001d000000000012004b0000014e0000c13d0000000001000411000000000010043f0000000801000039000000200010043f000000000100041400000a560010009c00000a5601008041000000c00110021000000b0e011001c700008010020000392953294e0000040f00000001002001900000014e0000613d000000000101043b0000001202000029000000000020043f000000200010043f000000000100041400000a560010009c00000a5601008041000000c00110021000000b0e011001c700008010020000392953294e0000040f00000001002001900000014e0000613d000000000101043b000000000201041a00000b4e022001970000001103000029000000000232019f000000000021041b000000400100043d000000000031043500000a560010009c00000a56010080410000004001100210000000000200041400000a560020009c00000a5602008041000000c002200210000000000112019f00000b2c011001c70000800d02000039000000030300003900000b340400004100000000050004110000001206000029000003200000013d000000240030008c0000014e0000413d0000000002000416000000000002004b0000014e0000c13d0000000401100370000000000101043b000000000010043f0000001c01000039000000200010043f00000040020000390000000001000019295329160000040f000000000101041a00000a5b01100197000000800010043f00000aec01000041000029540001042e000000440030008c0000014e0000413d0000000002000416000000000002004b0000014e0000c13d0000000402100370000000000202043b00000a5b0020009c0000014e0000213d0000002401100370000000000101043b001200000001001d00000a5b0010009c0000014e0000213d000000000020043f0000000801000039000000200010043f00000040020000390000000001000019295329160000040f00000012020000292953203a0000040f000000000101041a000000ff001001900000000001000039000000010100c039000000400200043d000000000012043500000a560020009c00000a5602008041000000400120021000000b08011001c7000029540001042e000000240030008c0000014e0000413d0000000001000416000000000001004b0000014e0000c13d295325f00000040f00000004010000390000000101100367000000000101043b000000000010043f0000002001000039000000200010043f00000040020000390000000001000019295329160000040f0000000401100039000000000201041a00000aed02200197000000000021041b0000000001000019000029540001042e000000640030008c0000014e0000413d0000000002000416000000000002004b0000014e0000c13d0000002402100370000000000202043b000f00000002001d0000000402100370000000000202043b0000004404100370000000000404043b00000a590040009c0000014e0000213d0000002305400039000000000035004b0000014e0000813d000e00040040003d0000000e01100360000000000101043b001200000001001d00000a590010009c0000014e0000213d000000240140003900000012040000290000000504400210000c00000001001d001100000004001d000d00000014001d0000000d0030006b0000014e0000213d000000000020043f0000002001000039000000200010043f000000000100041400000a560010009c00000a5601008041000000c00110021000000b0e011001c700008010020000392953294e0000040f00000001002001900000014e0000613d000000000101043b000b00000001001d0000000101100039000000000101041a000a00000001001d000000000001004b000016530000c13d000000400100043d000000440210003900000b4203000041000000000032043500000024021000390000001303000039000019820000013d00000b0101000041000000800010043f0000000001000411000000840010043f00000b0201000041000029550001043000000b0101000041000000800010043f000000840050043f00000b020100004100002955000104300000000f01000039000000000101041a001100000001001d00000a65010000410000000000100443000000000100041400000a560010009c00000a5601008041000000c00110021000000a66011001c70000800b020000392953294e0000040f000000010020019000001dc20000613d000000000101043b000000110010006c0000147f0000813d000000400100043d000000440210003900000b3903000041000000000032043500000024021000390000001603000039000019820000013d000000a00500003900000b0d0300004100000000040000190000000006050019000000000503041a000000000556043600000001033000390000000104400039000000000014004b000011e60000413d000000410160008a00000b4d0410019700000af60040009c000012810000a13d00000b1d01000041000000000010043f0000004101000039000000040010043f00000a750100004100002955000104300000001a03000039000000000303041a001000000003001d00110a5b0030019c00000000030000190000129a0000c13d0000001702000039000000000202041a0000001804000039000000000404041a000000400510003900000000003504350000002003100039000000000043043500000a5b02200197000000000021043500000a560010009c00000a5601008041000000400110021000000b33011001c7000029540001042e00000b0101000041000000800010043f000000840020043f00000b02010000410000295500010430000000800010043f000000000004004b000012220000613d000000000030043f000000000001004b0000000002000019000012270000613d00000a63030000410000000002000019000000000403041a000000a005200039000000000045043500000001033000390000002002200039000000000012004b0000121a0000413d000012270000013d00000b4e02200197000000a00020043f000000000001004b000000200200003900000000020060390000002002200039000000800100003929531f300000040f000000400100043d001200000001001d000000800200003929531fdb0000040f0000001202000029000000000121004900000a560010009c00000a5601008041000000600110021000000a560020009c00000a56020080410000004002200210000000000121019f000029540001042e0000001101000029000000000010043f0000000501000039000000200010043f000000000100041400000a560010009c00000a5601008041000000c00110021000000b0e011001c700008010020000392953294e0000040f00000001002001900000014e0000613d000000000101043b000000000101041a000000000001004b000012610000c13d0000000101000039000000000101041a000000110010006c000010f30000a13d001200110000002d0000001201000029000000010110008a001200000001001d000000000010043f0000000501000039000000200010043f000000000100041400000a560010009c00000a5601008041000000c00110021000000b0e011001c700008010020000392953294e0000040f00000001002001900000014e0000613d000000000101043b000000000101041a000000000001004b0000124e0000613d00000b0f00100198000010f30000c13d00000a5b001001980000162b0000c13d000000400100043d000000640210003900000b1b030000410000000000320435000000440210003900000b1c030000410000135b0000013d00000b490020009c0000000001000039000000010100603900000b4a0020009c00000001011061bf00000b4b0020009c00000001011061bf00000b4a0020009c000012790000613d00000b4b0020009c000012790000613d00000b490020009c000000000100c019000000010110018f000000800010043f00000aec01000041000029540001042e0000001902000039000000000012041b0000000001000019000029540001042e0000008001400039000000400010043f0000000000210435000000a002400039000000800300043d0000000000320435000000c002400039000000000003004b000012910000613d000000a0040000390000000005000019000000004604043400000000026204360000000105500039000000000035004b0000128c0000413d000000000212004900000a560020009c00000a5602008041000000600220021000000a560010009c00000a56010080410000004001100210000000000112019f000029540001042e0000000d01000039000000000101041a000f00000001001d00000b3001000041000000800010043f000000000100041400000a560010009c00000a5601008041000000c00110021000000b04011001c72953294e0000040f000000600310027000000a5603300197000000a00030008c000000a00400003900000000040340190000001f0640018f000000e0074001900000008005700039000012b40000613d0000008008000039000000000901034f000000009a09043c0000000008a80436000000000058004b000012b00000c13d000000000006004b000012c10000613d000000000771034f0000000306600210000000000805043300000000086801cf000000000868022f000000000707043b0000010006600089000000000767022f00000000066701cf000000000686019f00000000006504350000000100200190000014860000613d0000001f01400039000001e00210018f0000008001200039000000400010043f000000a00030008c0000014e0000413d000000800300043d00000b310030009c0000014e0000213d000001000300043d00000b310030009c0000014e0000213d000000a00300043d000e00000003001d00000afa0030009c000017c00000213d0000000e0000006b000017c00000613d000000e00100043d000d00000001001d00000a65010000410000000000100443000000000100041400000a560010009c00000a5601008041000000c00110021000000a66011001c70000800b020000392953294e0000040f000000010020019000001dc20000613d0000001202000029000000a00220027000000a5602200197000000000101043b000000000121004b00000ca20000413d0000000d0010006b0000197c0000a13d000000400200043d000d00000002001d00000b30010000410000000001120436000c00000001001d00000a560020009c00000a560100004100000000010240190000004001100210000000000200041400000a560020009c00000a5602008041000000c002200210000000000112019f00000afc011001c700000011020000292953294e0000040f000000600310027000000a5603300197000000a00030008c000000a00400003900000000040340190000001f0640018f000000e0074001900000000d0b0000290000000d057000290000130b0000613d000000000801034f00000000090b0019000000008a08043c0000000009a90436000000000059004b000013070000c13d000000000006004b000013180000613d000000000771034f0000000306600210000000000805043300000000086801cf000000000868022f000000000707043b0000010006600089000000000767022f00000000066701cf000000000686019f00000000006504350000000100200190000019ae0000613d0000001f01400039000001e00210018f0000000001b20019000000000021004b0000000002000039000000010200403900000a590010009c000011f10000213d0000000100200190000011f10000c13d000000400010043f000000a00030008c0000014e0000413d00000000020b043300000b310020009c0000014e0000213d0000008002b00039000000000202043300000b310020009c0000014e0000213d0000000c020000290000000002020433001100000002001d00000afa0020009c0000197d0000213d000000110000006b0000197d0000613d0000006001b000390000000001010433000d00000001001d00000a65010000410000000000100443000000000100041400000a560010009c00000a5601008041000000c00110021000000a66011001c70000800b020000392953294e0000040f000000010020019000001dc20000613d0000001002000029000000a00220027000000a5602200197000000000101043b000000000121004b00000ca20000413d0000000d0010006b0000197c0000a13d0000001201000029000000c00110027000000a5602100197000000120120008900000a560010009c00000ca20000213d000000120020008c00001be10000c13d000000010200003900001bea0000013d000000400100043d000000640210003900000af1030000410000000000320435000000440210003900000af203000041000000000032043500000024021000390000002f03000039000000000032043500000a7202000041000000000021043500000004021000390000002003000039000000000032043500000a560010009c00000a5601008041000000400110021000000af3011001c700002955000104300000001f0530018f00000a5806300198000000400200043d0000000004620019000015fc0000613d000000000701034f0000000008020019000000007907043c0000000008980436000000000048004b000013700000c13d000015fc0000013d0000001201000029000000000010043f0000001c01000039000000200010043f000000000100041400000a560010009c00000a5601008041000000c00110021000000b0e011001c700008010020000392953294e0000040f00000001002001900000014e0000613d000000000101043b000000000101041a001200000001001d00000024010000390000000101100367000000000101043b0000001102000029295321a90000040f000027100110011a000000400200043d00000020032000390000000000130435000000120100002900000a5b01100197000000000012043500000a560020009c00000a5602008041000000400120021000000b43011001c7000029540001042e00000a5c01100197000000000161019f000000000010041b000000000100041400000a560010009c00000a5601008041000000c00110021000000a5d011001c70000800d02000039000000030300003900000a5e04000041295329490000040f0000000100200190000003230000c13d0000014e0000013d0000001f0530018f00000a5806300198000000400200043d0000000004620019000015fc0000613d000000000701034f0000000008020019000000007907043c0000000008980436000000000048004b000013ac0000c13d000015fc0000013d0000000d01000039000000000101041a000c00000001001d000000400300043d000e00000003001d00000b30010000410000000001130436000d00000001001d00000a560030009c00000a560100004100000000010340190000004001100210000000000300041400000a560030009c00000a5603008041000000c003300210000000000113019f00000afc011001c72953294e0000040f000000600310027000000a5603300197000000a00030008c000000a00400003900000000040340190000001f0640018f000000e0074001900000000e0b0000290000000e05700029000013d40000613d000000000801034f00000000090b0019000000008a08043c0000000009a90436000000000059004b000013d00000c13d000000000006004b000013e10000613d000000000771034f0000000306600210000000000805043300000000086801cf000000000868022f000000000707043b0000010006600089000000000767022f00000000066701cf000000000686019f00000000006504350000000100200190000015f10000613d0000001f01400039000001e00210018f0000000001b20019000000000021004b0000000002000039000000010200403900000a590010009c000011f10000213d0000000100200190000011f10000c13d000000400010043f000000a00030008c0000014e0000413d00000000020b043300000b310020009c0000014e0000213d0000008002b00039000000000202043300000b310020009c0000014e0000213d0000000d020000290000000002020433000d00000002001d00000afa0020009c0000197d0000213d0000000d0000006b0000197d0000613d0000006001b000390000000001010433000e00000001001d00000a65010000410000000000100443000000000100041400000a560010009c00000a5601008041000000c00110021000000a66011001c70000800b020000392953294e0000040f000000010020019000001dc20000613d0000001102000029000000a00220027000000a5602200197000000000101043b000000000121004b00000ca20000413d0000000e0010006b0000197c0000a13d000000400200043d000e00000002001d00000b30010000410000000001120436000b00000001001d00000a560020009c00000a560100004100000000010240190000004001100210000000000200041400000a560020009c00000a5602008041000000c002200210000000000112019f00000afc011001c700000010020000292953294e0000040f000000600310027000000a5603300197000000a00030008c000000a00400003900000000040340190000001f0640018f000000e0074001900000000e0b0000290000000e05700029000014350000613d000000000801034f00000000090b0019000000008a08043c0000000009a90436000000000059004b000014310000c13d000000000006004b000014420000613d000000000771034f0000000306600210000000000805043300000000086801cf000000000868022f000000000707043b0000010006600089000000000767022f00000000066701cf000000000686019f0000000000650435000000010020019000001a3d0000613d0000001f01400039000001e00210018f0000000001b20019000000000021004b0000000002000039000000010200403900000a590010009c000011f10000213d0000000100200190000011f10000c13d000000400010043f000000a00030008c0000014e0000413d00000000020b043300000b310020009c0000014e0000213d0000008002b00039000000000202043300000b310020009c0000014e0000213d0000000b020000290000000002020433001000000002001d00000afa0020009c0000197d0000213d000000100000006b0000197d0000613d0000006001b000390000000001010433000e00000001001d00000a65010000410000000000100443000000000100041400000a560010009c00000a5601008041000000c00110021000000a66011001c70000800b020000392953294e0000040f000000010020019000001dc20000613d0000000f02000029000000a00220027000000a5602200197000000000101043b000000000121004b00000ca20000413d0000000e0010006b0000197c0000a13d0000001101000029000000c00110027000000a5602100197000000120120008900000a560010009c00000ca20000213d000000120020008c00001bf90000c13d000000010200003900001c020000013d000000000100041100000a5b011001980000160f0000c13d00000b3801000041000000000010043f00000afc0100004100002955000104300000001f0530018f00000a5806300198000000400200043d0000000004620019000015fc0000613d000000000701034f0000000008020019000000007907043c0000000008980436000000000048004b0000148d0000c13d000015fc0000013d00000b4e011001970000000e020000290000000000120435000000100000006b000000200200003900000000020060390000003f0120003900000b4d031001970000001101300029000000000031004b0000000003000039000000010300403900000a590010009c000011f10000213d0000000100300190000011f10000c13d000000400010043f0000008003100039000000c00400003900000000004304350000000b0500002900000080035002700000006004100039000000000034043500000aed035001970000004004100039000000000034043500000020031000390000000d0400002900000000004304350000000c030000290000000000310435000000c004100039000000110300002900000000030304330000000000340435000000e004100039000000000003004b0000000e08000029000014c20000613d000000000500001900000000064500190000000007580019000000000707043300000000007604350000002005500039000000000035004b000014bb0000413d00000000044300190000000000040435000000a0041000390000000f0500002900000000005404350000001f0330003900000b4d02300197000000e002200039000012920000013d000000400200043d00000b010300004100000000003204350000000403200039000000000013043500000a560020009c00000a5602008041000000400120021000000a75011001c7000029550001043000000b0f00100198000014e10000c13d0000001101000029000000000010043f0000000701000039000000200010043f00000040020000390000000001000019295329160000040f000000000101041a00000a5b011001970000116b0000013d00000b4701000041000000000010043f00000afc01000041000029550001043000000a65010000410000000000100443000000000100041400000a560010009c00000a5601008041000000c00110021000000a66011001c70000800b020000392953294e0000040f000000010020019000001dc20000613d000000400200043d001000000002001d0000000402200039000000000101043b000000110010006c000016a20000a13d00000af40100004100000010030000290000000000130435000000000100041000000a5b01100197000000000012043500000a560030009c00000a560100004100000000010340190000004001100210000000000200041400000a560020009c00000a5602008041000000c002200210000000000112019f00000a75011001c700000012020000292953294e0000040f000000600310027000000a5603300197000000200030008c000000200400003900000000040340190000001f0640018f00000020074001900000001005700029000015170000613d000000000801034f0000001009000029000000008a08043c0000000009a90436000000000059004b000015130000c13d000000000006004b000015240000613d000000000771034f0000000306600210000000000805043300000000086801cf000000000868022f000000000707043b0000010006600089000000000767022f00000000066701cf000000000686019f00000000006504350000000100200190000017ce0000613d0000001f01400039000000600210018f0000001001200029000000000021004b0000000002000039000000010200403900000a590010009c000011f10000213d0000000100200190000011f10000c13d000000400010043f000000200030008c0000014e0000413d00000010010000290000000003010433000000000100041a00000a5b021001970000001201000029295326350000040f0000000001000019000029540001042e00000aef010000410000000000100443000000000100041200000004001004430000002400000443000000000100041400000a560010009c00000a5601008041000000c00110021000000af0011001c700008005020000392953294e0000040f000000010020019000001dc20000613d000000400200043d001200000002001d000000000101043b00000a5b011001970000000002000411000000000012004b000016ad0000c13d00000af4010000410000001203000029001200000003001d0000000000130435000000000100041000000a5b011001970000000402300039000000000012043500000a560030009c00000a560100004100000000010340190000004001100210000000000200041400000a560020009c00000a5602008041000000c002200210000000000112019f00000a75011001c700000011020000292953294e0000040f000000600310027000000a5603300197000000200030008c000000200400003900000000040340190000001f0640018f00000020074001900000001205700029000015730000613d000000000801034f0000001209000029000000008a08043c0000000009a90436000000000059004b0000156f0000c13d000000000006004b000015800000613d000000000771034f0000000306600210000000000805043300000000086801cf000000000868022f000000000707043b0000010006600089000000000767022f00000000066701cf000000000686019f00000000006504350000000100200190000015e50000613d0000001f01400039000000600210018f0000001201200029000000000021004b0000000002000039000000010200403900000a590010009c000011f10000213d0000000100200190000011f10000c13d000000400010043f000000200030008c00000012010000290000014e0000413d0000000001010433001200000001001d00000aef0100004100000000001004430000000001000412000000040010044300000040010000390000002400100443000000000100041400000a560010009c00000a5601008041000000c00110021000000af0011001c700008005020000392953294e0000040f000000010020019000001dc20000613d000000000201043b000000120000006b000003230000613d00000012012000b900000012031000fa000000000023004b00000ca20000c13d000027100310011a001000000003001d000f00120030007300000ca20000413d000027100010008c000019200000813d0000001002000029000000120020006b000003230000613d00000aef010000410000000000100443000000000100041200000004001004430000002400000443000000000100041400000a560010009c00000a5601008041000000c00110021000000af0011001c700008005020000392953294e0000040f000000010020019000001dc20000613d000000000301043b000000400100043d00000044021000390000000f040000290000000000420435000000200210003900000af504000041000000000042043500000a5b03300197000000240410003900000000003404350000004403000039000000000031043500000af60010009c000011f10000213d0000008003100039000000400030043f00000a560020009c00000a56020080410000004002200210000000000101043300000a560010009c00000a56010080410000006001100210000000000121019f000000000200041400000a560020009c00000a5602008041000000c002200210000000000121019f0000001102000029295329490000040f000000600310027000000a560330019800001a4d0000c13d001200600000003d001000800000003d00001a760000013d0000001f0530018f00000a5806300198000000400200043d0000000004620019000015fc0000613d000000000701034f0000000008020019000000007907043c0000000008980436000000000048004b000015ec0000c13d000015fc0000013d0000001f0530018f00000a5806300198000000400200043d0000000004620019000015fc0000613d000000000701034f0000000008020019000000007907043c0000000008980436000000000048004b000015f80000c13d000000000005004b000016090000613d000000000161034f0000000305500210000000000604043300000000065601cf000000000656022f000000000101043b0000010005500089000000000151022f00000000015101cf000000000161019f0000000000140435000000600130021000000a560020009c00000a56020080410000004002200210000000000112019f0000295500010430000000000010043f0000000601000039000000200010043f000000000100041400000a560010009c00000a5601008041000000c00110021000000b0e011001c700008010020000392953294e0000040f00000001002001900000014e0000613d0000000e02000039000000000202041a000000000101043b000000000101041a00000a5901100197000000000112004b0000000001004019000000120010006c000016c10000813d000000400100043d000000440210003900000b3703000041000000000032043500000024021000390000001b03000039000019820000013d0000001001000039000000000101041a000000ff001001900000169b0000c13d0000000b05000039000000000405041a000000010640019000000001024002700000007f0220618f0000001f0020008c00000000010000390000000101002039000000000114013f000000010010019000000bc70000c13d000000400100043d0000000003210436000000000006004b000018680000613d000000000050043f000000000002004b00000000040000190000186d0000613d00000a6e0500004100000000040000190000000006430019000000000705041a000000000076043500000001055000390000002004400039000000000024004b000016440000413d0000186d0000013d000000400100043d000000440210003900000b0c03000041000000000032043500000024021000390000001f03000039000019820000013d0000000b010000290000000401100039000000000101041a000900000001001d00000a65010000410000000000100443000000000100041400000a560010009c00000a5601008041000000c00110021000000a66011001c70000800b020000392953294e0000040f000000010020019000001dc20000613d000000400200043d00000009030000290000008003300270000000000101043b000000000031004b000017da0000813d000000440120003900000b2b03000041000000000031043500000024012000390000001503000039000000000031043500000a7201000041000000000012043500000004012000390000002003000039000000000031043500000a560020009c00000a5602008041000000400120021000000a73011001c700002955000104300000001201000029000000000010043f0000000801000039000000200010043f000000000100041400000a560010009c00000a5601008041000000c00110021000000b0e011001c700008010020000392953294e0000040f00000001002001900000014e0000613d000000000101043b000000000200041100000a5b02200197000000000020043f000000200010043f000000000100041400000a560010009c00000a5601008041000000c00110021000000b0e011001c700008010020000392953294e0000040f00000001002001900000014e0000613d000000000101043b000000000101041a000000ff00100190000006d10000c13d00000b4401000041000000000010043f00000afc010000410000295500010430000000110100002900000b110010009c000017330000413d0000004001000039000000110200002900000b110220012a0000173c0000013d00000a72010000410000001003000029000000000013043500000020010000390000000000120435000000440130003900000b0c02000041000000000021043500000024013000390000001f02000039000001a60000013d0000001203000029000000640130003900000af1020000410000000000210435000000440130003900000af202000041000000000021043500000024013000390000002f02000039000000000021043500000a7201000041000000000013043500000004013000390000002002000039000000000021043500000a560030009c00000a5603008041000000400130021000000af3011001c700002955000104300000000d01000039000000000201041a00000012012000b9000000000002004b000016c90000613d00000000022100d9000000120020006c00000ca20000c13d0000000002000416000000000012004b00000f1b0000813d000000400100043d000000640210003900000b35030000410000000000320435000000440210003900000b36030000410000000000320435000000240210003900000023030000390000135e0000013d00000a5d011001c7000080090200003900000011030000290000000005000019295329490000040f000000600310027000000a5603300198000017030000613d0000001f0430003900000a57044001970000003f0440003900000af704400197000000400500043d0000000004450019000000000054004b0000000006000039000000010600403900000a590040009c000011f10000213d0000000100600190000011f10000c13d000000400040043f0000001f0430018f000000000635043600000a58053001980000000003560019000016f60000613d000000000701034f000000007807043c0000000006860436000000000036004b000016f20000c13d000000000004004b000017030000613d000000000151034f0000000304400210000000000503043300000000054501cf000000000545022f000000000101043b0000010004400089000000000141022f00000000014101cf000000000151019f00000000001304350000000100200190000003230000c13d000000400100043d000000440210003900000b0003000041000000000032043500000024021000390000000f03000039000019820000013d0000001101000029000000000010043f0000001c01000039000000200010043f000000000100041400000a560010009c00000a5601008041000000c00110021000000b0e011001c700008010020000392953294e0000040f00000001002001900000014e0000613d000000100200002900000a5b02200197000000000101043b000000000301041a00000a5c03300197000000000223019f000000000021041b00000044010000390000000101100367000000000101043b001200000001001d0000001101000029000000000010043f0000001d01000039000000200010043f000000000100041400000a560010009c00000a5601008041000000c00110021000000b0e011001c700008010020000392953294e0000040f00000001002001900000014e0000613d000000000101043b000009630000013d000000110200002900000b130020009c00000b120220212a0000000001000039000000200100203900000b140020009c00000010011081bf00000aed0220819700000b140220812a00000b150020009c000000080110803900000a590220819700000b150220812a000027100020008c000000040110803900000a5602208197000027100220811a000000640020008c00000002011080390000ffff0220818f000000640220811a000000090020008c000000010110203900000b4d051001970000005f0350003900000b4d06300197000000400400043d0000000003460019000000000063004b0000000006000039000000010600403900000a590030009c000011f10000213d0000000100600190000011f10000c13d000000400030043f00000001031000390000000003340436000000200550003900000b4d065001980000001f0550018f000017650000613d0000000006630019000000000700003100000001077003670000000008030019000000007907043c0000000008980436000000000068004b000017610000c13d000000000005004b000000000114001900000021011000390000001107000029000000090070008c0000000a5770011a0000000305500210000000010110008a000000000601043300000b160660019700000b170550021f00000b1805500197000000000565019f0000000000510435000017690000213d0000000a08000039000000000708041a000000010970019000000001057002700000007f0550618f0000001f0050008c00000000010000390000000101002039000000000117013f000000010010019000000bc70000c13d000000400100043d000000000005004b000018860000c13d00000b1a0010009c000011f10000213d0000002002100039000000400020043f0000000000010435000000400300043d000018790000013d00000a5d011001c700008009020000390000000005000019295329490000040f000000600310027000000a5603300198000013a20000613d0000001f0430003900000a57044001970000003f0440003900000af704400197000000400500043d0000000004450019000000000054004b0000000006000039000000010600403900000a590040009c000011f10000213d0000000100600190000011f10000c13d000000400040043f0000001f0430018f000000000635043600000a58053001980000000003560019000017a80000613d000000000701034f000000007807043c0000000006860436000000000036004b000017a40000c13d000000000004004b000013a20000613d000000000151034f0000000304400210000000000503043300000000054501cf000000000545022f000000000101043b0000010004400089000000000141022f00000000014101cf000000000151019f0000000000130435000013a20000013d000000000002004b0000000003000019000017ba0000613d000000a00300043d000000030420021000000b4c0440027f00000b4c04400167000000000443016f0000000103200210000018de0000013d00000a72030000410000000000310435000000c40320003900000b32040000410000000000430435000000a4032000390000001c040000390000000000430435000000840220003900000020030000390000000000320435000000400110021000000a73011001c700002955000104300000001f0530018f00000a5806300198000000400200043d0000000004620019000015fc0000613d000000000701034f0000000008020019000000007907043c0000000008980436000000000048004b000017d50000c13d000015fc0000013d0000001401000039000000000112043600000000030004110000006003300210000000000031043500000b280020009c000011f10000213d0000004003200039000000400030043f00000a560010009c00000a56010080410000004001100210000000000202043300000a560020009c00000a56020080410000006002200210000000000112019f000000000200041400000a560020009c00000a5602008041000000c002200210000000000112019f00000a5d011001c700008010020000392953294e0000040f00000001002001900000014e0000613d000000000101043b00000011020000290000003f0220003900000b2902200197000000400300043d0000000002230019001100000003001d000000000032004b0000000003000039000000010300403900000a590020009c000011f10000213d0000000100300190000011f10000c13d000000400020043f000000120200002900000011040000290000000002240436001000000002001d0000000d02000029000000000020007c0000014e0000213d000000120000006b0000183a0000613d0000000e020000290000002002200039000000010220036700000011030000290000000c050000290000000d060000290000002003300039000000002402043c00000000004304350000002005500039000000000065004b000018130000413d00000011020000290000000002020433000000000002004b0000183a0000613d001200000000001d0000001202000029000000050220021000000010022000290000000002020433000000000021004b000018280000813d000000000010043f000000200020043f00000000010004140000182b0000013d000000000020043f000000200010043f000000000100041400000a560010009c00000a5601008041000000c00110021000000b0e011001c700008010020000392953294e0000040f00000001002001900000014e0000613d000000000101043b0000001203000029001200010030003d00000011020000290000000002020433000000120020006b0000181e0000413d0000000a0010006c000019a40000c13d000000000100041100000a5b01100197001200000001001d000000000010043f0000000b010000290000000501100039001100000001001d000000200010043f000000000100041400000a560010009c00000a5601008041000000c00110021000000b0e011001c700008010020000392953294e0000040f00000001002001900000014e0000613d000000090200002900000aed02200197000000000101043b000000000101041a000000000112004b00000ca20000413d0000000f0010006c00001abb0000813d000000400100043d000000440210003900000b41030000410000197f0000013d000000110000006b0000000001000019000018fc0000613d0000001103000029000000030130021000000b4c0110027f00000b4c01100167000000100200002900000020022000390000000102200367000000000202043b000000000112016f0000000102300210000000000121019f000018fc0000013d00000b4e044001970000000000430435000000000002004b000000200400003900000000040060390000003f0240003900000b4d022001970000000004120019000000000024004b0000000002000039000000010200403900000a590040009c000011f10000213d0000000100200190000011f10000c13d0000000003040019000000400040043f000000200200003900000da40000013d000000000002004b00000000030000190000187f0000613d000000a00300043d000000030420021000000b4c0440027f00000b4c04400167000000000343016f0000000102200210000000000223019f000018ad0000013d0000002006100039000000000009004b0000198d0000613d000000000080043f00000a6b0700004100000000080000190000000009680019000000000a07041a0000000000a9043500000001077000390000002008800039000000000058004b0000188c0000413d0000198f0000013d00000a68030000410000002006000039000000010540008a000000050550027000000a690550009a000000000706001900000080066000390000000006060433000000000063041b00000020067000390000000103300039000000000053004b000018990000c13d000000a005700039000000000024004b000018ab0000813d0000000304200210000000f80440018f00000b4c0440027f00000b4c044001670000000005050433000000000445016f000000000043041b000000010220021000000001022001bf000000000021041b000000000100041400000a560010009c00000a5601008041000000c00110021000000a5d011001c70000800d02000039000000010300003900000b3a04000041000003200000013d00000a6e030000410000002006000039000000010540008a000000050550027000000a6f0550009a000000000706001900000080066000390000000006060433000000000063041b00000020067000390000000103300039000000000053004b000018bc0000c13d000018d20000013d00000a6b030000410000002006000039000000010540008a000000050550027000000a6c0550009a000000000706001900000080066000390000000006060433000000000063041b00000020067000390000000103300039000000000053004b000018ca0000c13d000000a005700039000000000024004b000018dc0000813d0000000304200210000000f80440018f00000b4c0440027f00000b4c044001670000000005050433000000000445016f000000000043041b00000001030000390000000104200210000000000234019f000000000021041b0000000001000019000029540001042e000000010400036700000000030000190000000f060000290000000005630019000000000554034f000000000505043b000000000051041b00000001011000390000002003300039000000000023004b000018e50000413d000000110020006c000018f90000813d00000011020000290000000302200210000000f80220018f00000b4c0220027f00000b4c022001670000000f033000290000000103300367000000000303043b000000000223016f000000000021041b0000001101000029000000010110021000000001011001bf0000000e03000029000000000013041b00000002013000390000000b02000029000000000021041b00000003013000390000000c02000029000000000021041b0000000a0100002900000080011002100000000d0200002900000aed02200197000000000112019f0000000402300039000000000012041b0000002102000039000000000102041a000000000001004b000019140000c13d0000000103100039000000000032041b000000000020043f00000b2d0110009a000009630000013d000000000020043f000000000300001900000b2d0430009a000000000404041a000000120040006c000003230000613d0000000103300039000000000013004b000019160000413d00000a590010009c000011f10000213d0000190f0000013d00000aef0100004100000000001004430000000001000412000000040010044300000020010000390000002400100443000000000100041400000a560010009c00000a5601008041000000c00110021000000af0011001c700008005020000392953294e0000040f000000010020019000001dc20000613d000000000301043b000000400100043d000000440210003900000010040000290000000000420435000000200210003900000af504000041000000000042043500000a5b03300197000000240410003900000000003404350000004403000039000000000031043500000af60010009c000011f10000213d0000008003100039000000400030043f00000a560020009c00000a56020080410000004002200210000000000101043300000a560010009c00000a56010080410000006001100210000000000121019f000000000200041400000a560020009c00000a5602008041000000c002200210000000000121019f0000001102000029295329490000040f000000600310027000000a560330019800001a130000c13d000e00600000003d000d00800000003d0000000e010000290000000001010433000000010020019000001a490000613d000000000001004b0000196e0000c13d00000af801000041000000000010044300000011010000290000000400100443000000000100041400000a560010009c00000a5601008041000000c00110021000000af9011001c700008002020000392953294e0000040f000000010020019000001dc20000613d000000000101043b000000000001004b00001a8c0000613d0000000e010000290000000001010433000000000001004b000015ae0000613d00000afa0010009c0000014e0000213d000000200010008c0000014e0000413d0000000d010000290000000001010433000000000001004b0000000002000039000000010200c039000000000021004b0000014e0000c13d000000000001004b000015ae0000c13d00001aeb0000013d000000400100043d000000440210003900000b3203000041000000000032043500000024021000390000001c03000039000000000032043500000a7202000041000000000021043500000004021000390000002003000039000000000032043500000a560010009c00000a5601008041000000400110021000000a73011001c7000029550001043000000b4e07700197000000000076043500000000055600190000000004040433000000000004004b0000199b0000613d000000000600001900000000075600190000000008360019000000000808043300000000008704350000002006600039000000000046004b000019940000413d000000000354001900000b1904000041000000000043043500000000031300490000001b0430008a0000000000410435000000240330003900000b4d023001970000186f0000013d000000400100043d000000640210003900000b3c030000410000000000320435000000440210003900000b3d030000410000000000320435000000240210003900000021030000390000135e0000013d0000001f0530018f00000a5806300198000000400200043d0000000004620019000015fc0000613d000000000701034f0000000008020019000000007907043c0000000008980436000000000048004b000019b50000c13d000015fc0000013d00000aef010000410000000000100443000000000100041200000004001004430000002400000443000000000100041400000a560010009c00000a5601008041000000c00110021000000af0011001c700008005020000392953294e0000040f000000010020019000001dc20000613d000000000101043b00000a5b041001970000000001000411000000120010006c000019cf0000613d000000000041004b000003230000c13d0000000f02000029000000110020002a00000ca20000413d0000000f0200002900000011012000290000001202000039000000000012041b0000000e02000029000000100020002a00000ca20000413d0000000e0200002900000010032000290000001301000039000000000001041b000000000100041400000a560010009c00000a5601008041000000c001100210000000000003004b00001a990000c13d000000000204001900001a9c0000013d00000a5d011001c7000080090200003900000000040004110000000005000019295329490000040f000000600310027000000a5603300198000003210000613d0000001f0430003900000a57044001970000003f0440003900000af704400197000000400500043d0000000004450019000000000054004b0000000006000039000000010600403900000a590040009c000011f10000213d0000000100600190000011f10000c13d000000400040043f0000001f0430018f000000000635043600000a5805300198000000000356001900001a050000613d000000000701034f000000007807043c0000000006860436000000000036004b00001a010000c13d000000000004004b000003210000613d000000000151034f0000000304400210000000000503043300000000054501cf000000000545022f000000000101043b0000010004400089000000000141022f00000000014101cf000000000151019f0000000000130435000003210000013d0000001f0430003900000a57044001970000003f0440003900000af704400197000000400500043d0000000004450019000e00000005001d000000000054004b0000000005000039000000010500403900000a590040009c000011f10000213d0000000100500190000011f10000c13d000000400040043f0000001f0430018f0000000e05000029000000000635043600000a5805300198000d00000006001d000000000356001900001a2f0000613d000000000601034f0000000d07000029000000006806043c0000000007870436000000000037004b00001a2b0000c13d000000000004004b000019540000613d000000000151034f0000000304400210000000000503043300000000054501cf000000000545022f000000000101043b0000010004400089000000000141022f00000000014101cf000000000151019f0000000000130435000019540000013d0000001f0530018f00000a5806300198000000400200043d0000000004620019000015fc0000613d000000000701034f0000000008020019000000007907043c0000000008980436000000000048004b00001a440000c13d000015fc0000013d000000000001004b00001a910000613d0000000d02000029000001830000013d0000001f0430003900000a57044001970000003f0440003900000af704400197000000400500043d0000000004450019001200000005001d000000000054004b0000000005000039000000010500403900000a590040009c000011f10000213d0000000100500190000011f10000c13d000000400040043f0000001f0430018f0000001205000029000000000635043600000a5805300198001000000006001d000000000356001900001a690000613d000000000601034f0000001007000029000000006806043c0000000007870436000000000037004b00001a650000c13d000000000004004b00001a760000613d000000000151034f0000000304400210000000000503043300000000054501cf000000000545022f000000000101043b0000010004400089000000000141022f00000000014101cf000000000151019f000000000013043500000012010000290000000001010433000000010020019000001a8f0000613d000000000001004b00001ade0000c13d00000af801000041000000000010044300000011010000290000000400100443000000000100041400000a560010009c00000a5601008041000000c00110021000000af9011001c700008002020000392953294e0000040f000000010020019000001dc20000613d000000000101043b000000000001004b00001ada0000c13d000000400100043d00000afe0200004100001aed0000013d000000000001004b00001ab90000c13d000000400100043d00000afb02000041000000000021043500000a560010009c00000a5601008041000000400110021000000afc011001c7000029550001043000000a5d011001c700008009020000390000000005000019295329490000040f000000600310027000000a5603300198000003210000613d0000001f0430003900000a57044001970000003f0440003900000af704400197000000400500043d0000000004450019000000000054004b0000000006000039000000010600403900000a590040009c000011f10000213d0000000100600190000011f10000c13d000000400040043f0000001f0430018f000000000635043600000a5805300198000000000356001900001a050000613d000000000701034f000000007807043c0000000006860436000000000036004b00001ab40000c13d00001a050000013d0000001002000029000001830000013d0000000201000039000000000101041a00000b4c011001670000000102000039000000000202041a00000000011200190000000f0010002a00000ca20000413d0000000f011000290000000c02000039000000000202041a000000000021004b00001acf0000a13d000000400100043d000000440210003900000b4003000041000000000032043500000024021000390000001203000039000019820000013d0000001701000039000000000101041a00000a5b0010019800001af10000c13d000000400100043d000000440210003900000b3f03000041000000000032043500000024021000390000001d03000039000019820000013d00000012010000290000000001010433000000000001004b000003230000613d00000afa0010009c0000014e0000213d000000200010008c0000014e0000413d00000010010000290000000001010433000000000001004b0000000002000039000000010200c039000000000021004b0000014e0000c13d000000000001004b000003230000c13d000000400100043d00000afd02000041000000000021043500000004021000390000001103000029000001960000013d0000000b010000290000000301100039000000000101041a000000000001004b00001b150000c13d000000400100043d000000440210003900000b3b03000041000000000032043500000024021000390000001703000039000019820000013d0000001201000029000000000010043f0000001101000029000000200010043f000000000100041400000a560010009c00000a5601008041000000c00110021000000b0e011001c700008010020000392953294e0000040f00000001002001900000014e0000613d000000000101043b000000000301041a0000000f0030002a00000ca20000413d0000000f020000290000000003230019000000000031041b0000000001000411295327760000040f0000000001000019000029540001042e0010000f001000bd00000010011000f90000000f0010006c00000ca20000c13d0000001201000029000000000010043f0000001101000029000000200010043f000000000100041400000a560010009c00000a5601008041000000c00110021000000b0e011001c700008010020000392953294e0000040f00000001002001900000014e0000613d000000000101043b000000000201041a0000000f0020002a00000ca20000413d00001c9e0000013d0000000e03000029000000440130003900000b3e02000041000000000021043500000024013000390000001c02000039000001a10000013d0000001f0530018f00000a5806300198000000400200043d0000000004620019000015fc0000613d000000000701034f0000000008020019000000007907043c0000000008980436000000000048004b00001b390000c13d000015fc0000013d000002600100043d000300000001001d000002400100043d000400000001001d000002a00100043d000100000001001d000000000100041a00000a5c02100197000000000262019f000000000020041b000000000200041400000a5b0510019700000a560020009c00000a5602008041000000c00120021000000a5d011001c70000800d02000039000000030300003900000a5e04000041295329490000040f00000001002001900000014e0000613d0000000f01000029000000000101043300000a590010009c000011f10000213d0000000302000039000000000202041a000000010320019000000001022002700000007f0220618f0000001f0020008c00000000040000390000000104002039000000000043004b00000bc70000c13d000000200020008c00001b740000413d0000000303000039000000000030043f0000001f03100039000000050330027000000a5f0330009a000000200010008c00000a60030040410000001f02200039000000050220027000000a5f0220009a000000000023004b00001b740000813d000000000003041b0000000103300039000000000023004b00001b700000413d000000200010008c0000000102100210000000030310021000001c110000413d0000000304000039000000000040043f00000b4d0610019800001c1b0000c13d000000200500003900000a600400004100001c270000013d0000001f0530018f00000a5806300198000000400200043d0000000004620019000015fc0000613d000000000701034f0000000008020019000000007907043c0000000008980436000000000048004b00001b860000c13d000015fc0000013d0000000a030000390000000102000039000000010010019000000000043300a9000000010300603900000000022300a90000000101100272000000000304001900001b8d0000c13d0000000a012000b9001000000001001d0000000a011000fa000000000021004b00000ca20000c13d0000001701000039000000000201041a000000400300043d000c00000003001d00000b0301000041000000000013043500000a560030009c00000a560100004100000000010340190000004001100210000000000300041400000a560030009c00000a5603008041000000c003300210000000000113019f00000afc011001c700000a5b022001972953294e0000040f000000600310027000000a5603300197000000200030008c000000200400003900000000040340190000001f0640018f00000020074001900000000c0b0000290000000c0570002900001bbb0000613d000000000801034f00000000090b0019000000008a08043c0000000009a90436000000000059004b00001bb70000c13d000000000006004b00001bc80000613d000000000771034f0000000306600210000000000805043300000000086801cf000000000868022f000000000707043b0000010006600089000000000767022f00000000066701cf000000000686019f0000000000650435000000010020019000001c5d0000613d0000001f01400039000000600210018f0000000001b20019000000000021004b0000000002000039000000010200403900000a590010009c000011f10000213d0000000100200190000011f10000c13d000000400010043f000000200030008c0000014e0000413d00000000010b0433000000ff0010008c0000014e0000213d0000001202100089000000ff0020008c00000ca20000213d000000120010008c00001c690000c13d000000010100003900001c720000013d0000000a030000390000000102000039000000010010019000000000043300a9000000010300603900000000022300a90000000101100272000000000304001900001be30000c13d0000000e032000b9001200000003001d0000000e013000fa000000000021004b00000ca20000c13d0000001001000029000000c00110027000000a5602100197000000120120008900000a560010009c00000ca20000213d000000120020008c00001cac0000c13d000000010200003900001cb50000013d0000000a030000390000000102000039000000010010019000000000043300a9000000010300603900000000022300a90000000101100272000000000304001900001bfb0000c13d0000000d032000b9000e00000003001d0000000d013000fa000000000021004b00000ca20000c13d0000000f01000029000000c00110027000000a5602100197000000120120008900000a560010009c00000ca20000213d000000120020008c00001d020000c13d000000010200003900001d0b0000013d000000000001004b000000000100001900001c310000613d00000b4c0130027f00000b4c011001670000000e030000290000000003030433000000000113016f000000000121019f00001c310000013d00000a60040000410000002005000039000000010760008a000000050770027000000a610770009a0000000f085000290000000008080433000000000084041b00000020055000390000000104400039000000000074004b00001c200000c13d000000000016004b00001c300000813d000000f80130018f00000b4c0110027f00000b4c011001670000000f035000290000000003030433000000000113016f000000000014041b00000001012001bf0000000302000039000000000012041b0000000d01000029000000000201043300000a590020009c000011f10000213d0000000401000039000000000401041a000000010040019000000001034002700000007f0330618f0000001f0030008c00000000050000390000000105002039000000000454013f000000010040019000000bc70000c13d000000200030008c00001c530000413d000000000010043f0000001f04200039000000050440027000000a620440009a000000200020008c00000a63040040410000001f03300039000000050330027000000a620330009a000000000034004b00001c530000813d000000000004041b0000000104400039000000000034004b00001c4f0000413d000000200020008c0000000103200210000000030420021000001d5a0000413d000000000010043f00000b4d0720019800001d640000c13d000000200600003900000a630500004100001d700000013d0000001f0530018f00000a5806300198000000400200043d0000000004620019000015fc0000613d000000000701034f0000000008020019000000007907043c0000000008980436000000000048004b00001c640000c13d000015fc0000013d0000000a030000390000000101000039000000010020019000000000043300a9000000010300603900000000011300a90000000102200272000000000304001900001c6b0000c13d0000000e020000290000000d022000b90000000d0000006b00001c790000613d0000000d032000fa0000000e0030006c00000ca20000c13d000000100000006b00001e6e0000613d000000000001004b00001e6e0000613d00000010022000fa00100000001200e10000001903000039000000000303041a000000000003004b00001c8d0000613d000027100030008c00000ca20000213d000027100430008900000010034000b9000000000021004b00001c8c0000213d00000010013000fa000000000041004b00000ca20000c13d00102710003001220000001201000029000000000010043f0000001101000029000000200010043f000000000100041400000a560010009c00000a5601008041000000c00110021000000b0e011001c700008010020000392953294e0000040f00000001002001900000014e0000613d000000000101043b000000000201041a0000000f0020002a00000ca20000413d0000000f02200029000000000021041b0000001701000039000000000101041a00000a5b01100197000000000300041000000000020004110000001004000029295326d40000040f00000000010004110000000f02000029295327760000040f0000000001000019000029540001042e0000000a030000390000000102000039000000010010019000000000043300a9000000010300603900000000022300a90000000101100272000000000304001900001cae0000c13d00000011012000b9001000000001001d00000011011000fa000000000021004b00000ca20000c13d0000001701000039000000000201041a000000400300043d001100000003001d00000b0301000041000000000013043500000a560030009c00000a560100004100000000010340190000004001100210000000000300041400000a560030009c00000a5603008041000000c003300210000000000113019f00000afc011001c700000a5b022001972953294e0000040f000000600310027000000a5603300197000000200030008c000000200400003900000000040340190000001f0640018f0000002007400190000000110b000029000000110570002900001cdc0000613d000000000801034f00000000090b0019000000008a08043c0000000009a90436000000000059004b00001cd80000c13d000000000006004b00001ce90000613d000000000771034f0000000306600210000000000805043300000000086801cf000000000868022f000000000707043b0000010006600089000000000767022f00000000066701cf000000000686019f0000000000650435000000010020019000001dc30000613d0000001f01400039000000600210018f0000000001b20019000000000021004b0000000002000039000000010200403900000a590010009c000011f10000213d0000000100200190000011f10000c13d000000400010043f000000200030008c0000014e0000413d00000000020b0433000000ff0020008c0000014e0000213d0000001203200089000000ff0030008c00000ca20000213d000000120020008c00001dcf0000c13d000000010200003900001dd80000013d0000000a030000390000000102000039000000010010019000000000043300a9000000010300603900000000022300a90000000101100272000000000304001900001d040000c13d00000010012000b9000f00000001001d00000010011000fa000000000021004b00000ca20000c13d0000001701000039000000000201041a000000400300043d001000000003001d00000b0301000041000000000013043500000a560030009c00000a560100004100000000010340190000004001100210000000000300041400000a560030009c00000a5603008041000000c003300210000000000113019f00000afc011001c700000a5b022001972953294e0000040f000000600310027000000a5603300197000000200030008c000000200400003900000000040340190000001f0640018f0000002007400190000000100b000029000000100570002900001d320000613d000000000801034f00000000090b0019000000008a08043c0000000009a90436000000000059004b00001d2e0000c13d000000000006004b00001d3f0000613d000000000771034f0000000306600210000000000805043300000000086801cf000000000868022f000000000707043b0000010006600089000000000767022f00000000066701cf000000000686019f0000000000650435000000010020019000001df40000613d0000001f01400039000000600110018f0000000002b10019000000000012004b00000000010000390000000101004039001100000002001d00000a590020009c000011f10000213d0000000100100190000011f10000c13d0000001101000029000000400010043f000000200030008c0000014e0000413d00000000010b0433000000ff0010008c0000014e0000213d0000001202100089000000ff0020008c00000ca20000213d000000120010008c00001e000000c13d000000010100003900001e090000013d000000000002004b000000000200001900001d7a0000613d00000b4c0240027f00000b4c022001670000000c040000290000000004040433000000000224016f000000000232019f00001d7a0000013d00000a63050000410000002006000039000000010870008a000000050880027000000a640880009a0000000d096000290000000009090433000000000095041b00000020066000390000000105500039000000000085004b00001d690000c13d000000000027004b00001d790000813d000000f80240018f00000b4c0220027f00000b4c022001670000000d046000290000000004040433000000000224016f000000000025041b00000001023001bf000000000021041b0000000101000039000000000011041b0000000f01000039000000010200008a000000000021041b00000a65010000410000000000100443000000000100041400000a560010009c00000a5601008041000000c00110021000000a66011001c70000800b020000392953294e0000040f000000010020019000001dc20000613d000000000101043b000001000010043f0000001e03000039000000000403041a00000b4e0440019700000001044001bf000000000043041b0000000c030000390000000804000029000000000043041b0000000d030000390000000904000029000000000043041b0000000b03000029000000000403043300000a590040009c000011f10000213d0000001403000039000000000603041a000000010060019000000001056002700000007f0550618f0000001f0050008c00000000070000390000000107002039000000000676013f000000010060019000000bc70000c13d000000200050008c00001db80000413d000000000030043f0000001f06400039000000050660027000000a670660009a000000200040008c00000a68060040410000001f05500039000000050550027000000a670550009a000000000056004b00001db80000813d000000000006041b0000000106600039000000000056004b00001db40000413d000000200040008c0000000105400210000000030640021000001e250000413d000000000030043f00000b4d0940019800001e2f0000c13d000000200800003900000a680700004100001e3b0000013d000000000001042f0000001f0530018f00000a5806300198000000400200043d0000000004620019000015fc0000613d000000000701034f0000000008020019000000007907043c0000000008980436000000000048004b00001dca0000c13d000015fc0000013d0000000a040000390000000102000039000000010030019000000000054400a9000000010400603900000000022400a90000000103300272000000000405001900001dd10000c13d00000012030000290000000f033000b90000000f0000006b00001ddf0000613d0000000f043000fa000000120040006c00000ca20000c13d000000100000006b00001e6e0000613d000000000002004b00001e6e0000613d00000010043000fa00000000032400d90000001905000039000000000505041a000000000005004b000011fd0000613d000027100050008c00000ca20000213d000027100650008900000000053600a9000000000042004b00001df20000213d00000000023500d9000000000062004b00000ca20000c13d000027100350011a000011fd0000013d0000001f0530018f00000a5806300198000000400200043d0000000004620019000015fc0000613d000000000701034f0000000008020019000000007907043c0000000008980436000000000048004b00001dfb0000c13d000015fc0000013d0000000a030000390000000101000039000000010020019000000000043300a9000000010300603900000000011300a90000000102200272000000000304001900001e020000c13d0000000e020000290000000c022000b90000000c0000006b00001e100000613d0000000c032000fa0000000e0030006c00000ca20000c13d0000000f0000006b00001e6e0000613d000000000001004b00001e6e0000613d0000000f022000fa000f0000001200e10000001903000039000000000303041a000000000003004b00000b830000613d000027100030008c00000ca20000213d00002710043000890000000f034000b9000000000021004b00001e230000213d0000000f013000fa000000000041004b00000ca20000c13d000027100130011a00000b820000013d000000000004004b000000000400001900001e450000613d00000b4c0460027f00000b4c0440016700000012060000290000000006060433000000000446016f000000000454019f00001e450000013d00000a68070000410000002008000039000000010a90008a000000050aa0027000000a690aa0009a0000000b0b800029000000000b0b04330000000000b7041b000000200880003900000001077000390000000000a7004b00001e340000c13d000000000049004b00001e440000813d000000f80460018f00000b4c0440027f00000b4c044001670000000b068000290000000006060433000000000446016f000000000047041b00000001045001bf000000000043041b0000000a03000029000000000403043300000a590040009c000011f10000213d0000000a03000039000000000603041a000000010060019000000001056002700000007f0550618f0000001f0050008c00000000070000390000000107002039000000000676013f000000010060019000000bc70000c13d000000200050008c00001e660000413d000000000030043f0000001f06400039000000050660027000000a6a0660009a000000200040008c00000a6b060040410000001f05500039000000050550027000000a6a0550009a000000000056004b00001e660000813d000000000006041b0000000106600039000000000056004b00001e620000413d000000200040008c00001e740000413d000000000030043f00000b4d0740019800001e800000c13d000000200600003900000a6b0500004100001e8c0000013d00000b1d01000041000000000010043f0000001201000039000000040010043f00000a75010000410000295500010430000000000004004b000000000500001900001e980000613d000000030540021000000b4c0550027f00000b4c0550016700000011060000290000000006060433000000000556016f0000000104400210000000000545019f00001e980000013d00000a6b050000410000002006000039000000010870008a000000050880027000000a6c0880009a0000000a096000290000000009090433000000000095041b00000020066000390000000105500039000000000085004b00001e850000c13d000000000047004b00001e960000813d0000000307400210000000f80770018f00000b4c0770027f00000b4c077001670000000a066000290000000006060433000000000676016f000000000065041b000000010440021000000001054001bf000000000053041b0000000703000029000000000503043300000a590050009c000011f10000213d0000000b04000039000000000604041a000000010060019000000001036002700000007f0330618f0000001f0030008c00000000070000390000000107002039000000000676013f000000010060019000000bc70000c13d000000200030008c00001eb90000413d000000000040043f0000001f06500039000000050660027000000a6d0660009a000000200050008c00000a6e060040410000001f03300039000000050330027000000a6d0330009a000000000036004b00001eb90000813d000000000006041b0000000106600039000000000036004b00001eb50000413d0000002003000039000000200050008c00001ed90000413d000000000040043f00000b4d0850019800000a6e06000041000000000703001900001ecc0000613d0000002007000039000000010980008a000000050990027000000a6f0990009a000000070a700029000000000a0a04330000000000a6041b00000020077000390000000106600039000000000096004b00001ec50000c13d000000000058004b00001ed60000813d0000000308500210000000f80880018f00000b4c0880027f00000b4c0880016700000007077000290000000007070433000000000787016f000000000076041b000000010550021000000001065001bf00001ee40000013d000000000005004b000000000600001900001ee40000613d000000030650021000000b4c0660027f00000b4c0660016700000010070000290000000007070433000000000667016f0000000105500210000000000656019f000000000064041b00002710040000390000000e05000039000000000045041b0000000604000029000000800040043f0000000504000029000000a00040043f0000000404000029000000e00040043f0000000304000029000027110040008c00001efc0000413d000000400100043d000000440210003900000a7104000041000000000042043500000024021000390000001d04000039000000000042043500000a720200004100000000002104350000000402100039000019870000013d0000000307000029000000c00070043f000000020400002900000a5b044001970000001505000039000000000605041a00000a5c06600197000000000446019f000000000045041b00000016040000390000000105000029000000000054041b0000001004000039000000000504041a00000b4e0250019700000007050000290000000005050433000000000005004b00000001022061bf000000000024041b00000140000004430000000602000029000001600020044300000180003004430000000502000029000001a0002004430000004002000039000001c000200443000001e000700443000000600200003900000200002004430000000402000029000002200020044300000080020000390000024000200443000002600010044300000100003004430000000501000039000001200010044300000a7001000041000029540001042e00000b4f0010009c00001f2a0000813d0000002001100039000000400010043f000000000001042d00000b1d01000041000000000010043f0000004101000039000000040010043f00000a750100004100002955000104300000001f0220003900000b4d022001970000000001120019000000000021004b0000000002000039000000010200403900000a590010009c00001f3c0000213d000000010020019000001f3c0000c13d000000400010043f000000000001042d00000b1d01000041000000000010043f0000004101000039000000040010043f00000a750100004100002955000104300000000b05000039000000000405041a000000010640019000000001024002700000007f0220618f0000001f0020008c00000000010000390000000101002039000000000016004b00001f700000c13d000000400100043d0000000003210436000000000006004b00001f5d0000613d000000000050043f000000000002004b00001f630000613d00000a6e0500004100000000040000190000000006430019000000000705041a000000000076043500000001055000390000002004400039000000000024004b00001f550000413d00001f640000013d00000b4e044001970000000000430435000000000002004b0000002004000039000000000400603900001f640000013d00000000040000190000003f0240003900000b4d032001970000000002130019000000000032004b0000000003000039000000010300403900000a590020009c00001f760000213d000000010030019000001f760000c13d000000400020043f000000000001042d00000b1d01000041000000000010043f0000002201000039000000040010043f00000a7501000041000029550001043000000b1d01000041000000000010043f0000004101000039000000040010043f00000a750100004100002955000104300003000000000002000000000201041a000000010320019000000001062002700000007f0660618f0000001f0060008c00000000040000390000000104002039000000000043004b00001fbb0000c13d000000400500043d0000000004650436000000000003004b00001fa60000613d000100000004001d000300000006001d000200000005001d000000000010043f000000000100041400000a560010009c00000a5601008041000000c00110021000000b2c011001c700008010020000392953294e0000040f000000010020019000001fc70000613d0000000306000029000000000006004b00001fac0000613d000000000201043b0000000001000019000000020500002900000001070000290000000003170019000000000402041a000000000043043500000001022000390000002001100039000000000061004b00001f9e0000413d00001fae0000013d00000b4e012001970000000000140435000000000006004b0000002001000039000000000100603900001fae0000013d000000000100001900000002050000290000003f0110003900000b4d021001970000000001520019000000000021004b0000000002000039000000010200403900000a590010009c00001fc10000213d000000010020019000001fc10000c13d000000400010043f0000000001050019000000000001042d00000b1d01000041000000000010043f0000002201000039000000040010043f00000a7501000041000029550001043000000b1d01000041000000000010043f0000004101000039000000040010043f00000a750100004100002955000104300000000001000019000029550001043000000000430104340000000001320436000000000003004b00001fd50000613d000000000200001900000000051200190000000006240019000000000606043300000000006504350000002002200039000000000032004b00001fce0000413d000000000213001900000000000204350000001f0230003900000b4d022001970000000001210019000000000001042d00000020030000390000000004310436000000003202043400000000002404350000004001100039000000000002004b00001fea0000613d000000000400001900000000051400190000000006430019000000000606043300000000006504350000002004400039000000000024004b00001fe30000413d000000000312001900000000000304350000001f0220003900000b4d022001970000000001120019000000000001042d00000afa0010009c000020000000213d000000630010008c000020000000a13d00000001030003670000000401300370000000000101043b00000a5b0010009c000020000000213d0000002402300370000000000202043b00000a5b0020009c000020000000213d0000004403300370000000000303043b000000000001042d0000000001000019000029550001043000000b500020009c000020320000813d00000000040100190000001f0120003900000b4d011001970000003f0110003900000b4d05100197000000400100043d0000000005510019000000000015004b0000000007000039000000010700403900000a590050009c000020320000213d0000000100700190000020320000c13d000000400050043f00000000052104360000000007420019000000000037004b000020380000213d00000b4d062001980000001f0720018f00000001044003670000000003650019000020220000613d000000000804034f0000000009050019000000008a08043c0000000009a90436000000000039004b0000201e0000c13d000000000007004b0000202f0000613d000000000464034f0000000306700210000000000703043300000000076701cf000000000767022f000000000404043b0000010006600089000000000464022f00000000046401cf000000000474019f000000000043043500000000022500190000000000020435000000000001042d00000b1d01000041000000000010043f0000004101000039000000040010043f00000a750100004100002955000104300000000001000019000029550001043000000a5b02200197000000000020043f000000200010043f000000000100041400000a560010009c00000a5601008041000000c00110021000000b0e011001c700008010020000392953294e0000040f0000000100200190000020480000613d000000000101043b000000000001042d00000000010000190000295500010430000000000100041100000a5b01100198000020610000613d000000000010043f0000000601000039000000200010043f000000000100041400000a560010009c00000a5601008041000000c00110021000000b0e011001c700008010020000392953294e0000040f0000000100200190000020650000613d0000000e02000039000000000202041a000000000101043b000000000101041a00000a5901100197000000000112004b0000000001004019000000000001042d00000b3801000041000000000010043f00000afc010000410000295500010430000000000100001900002955000104300007000000000002000300000002001d000500000001001d000600000003001d000000000003004b000021730000613d0000000601000029000000000010043f0000000501000039000000200010043f000000000100041400000a560010009c00000a5601008041000000c00110021000000b0e011001c700008010020000392953294e0000040f0000000100200190000021710000613d000000000101043b000000000101041a000000000001004b000020960000c13d0000000101000039000000000101041a000000060010006c000021730000a13d0000000602000029000000010220008a000700000002001d000000000020043f0000000501000039000000200010043f000000000100041400000a560010009c00000a5601008041000000c00110021000000b0e011001c700008010020000392953294e0000040f0000000100200190000021710000613d000000000101043b000000000101041a000000000001004b0000000702000029000020830000613d00000b0f00100198000021730000c13d000000050200002900000a5b02200197000400000001001d00000a5b01100197000700000002001d000000000021004b000021770000c13d0000000601000029000000000010043f0000000701000039000000200010043f000000000100041400000a560010009c00000a5601008041000000c00110021000000b0e011001c700008010020000392953294e0000040f0000000100200190000021710000613d000000000101043b000200000001001d000000000201041a000000000300041100000a5b04300197000000070040006c000500000002001d000020d70000613d000000000024004b000020d70000613d000100000004001d0000000701000029000000000010043f0000000801000039000000200010043f000000000100041400000a560010009c00000a5601008041000000c00110021000000b0e011001c700008010020000392953294e0000040f0000000100200190000021710000613d000000000101043b0000000102000029000000000020043f000000200010043f000000000100041400000a560010009c00000a5601008041000000c00110021000000b0e011001c700008010020000392953294e0000040f0000000100200190000021710000613d000000000101043b000000000101041a000000ff0010019000000005020000290000000003000411000021a50000613d000000070000006b000020ee0000613d0000001e01000039000000000101041a000000ff00100190000021800000613d000000000030043f0000001f01000039000000200010043f000000000100041400000a560010009c00000a5601008041000000c00110021000000b0e011001c700008010020000392953294e0000040f0000000100200190000021710000613d000000000101043b000000000101041a000000ff001001900000000502000029000021910000c13d000000000002004b000020f20000613d0000000201000029000000000001041b0000000701000029000000000010043f0000000601000039000000200010043f000000000100041400000a560010009c00000a5601008041000000c00110021000000b0e011001c700008010020000392953294e0000040f0000000100200190000021710000613d000000000101043b000000000201041a000000010220008a000000000021041b000000030100002900000a5b01100197000500000001001d000000000010043f0000000601000039000000200010043f000000000100041400000a560010009c00000a5601008041000000c00110021000000b0e011001c700008010020000392953294e0000040f0000000100200190000021710000613d000000000101043b000000000201041a0000000102200039000000000021041b00000a65010000410000000000100443000000000100041400000a560010009c00000a5601008041000000c00110021000000a66011001c70000800b020000392953294e0000040f00000001002001900000217b0000613d000000000101043b000300000001001d0000000601000029000000000010043f0000000501000039000000200010043f000000000100041400000a560010009c00000a5601008041000000c00110021000000b0e011001c700008010020000392953294e0000040f0000000100200190000021710000613d0000000302000029000000a00220021000000005022001af00000b1e022001c7000000000101043b000000000021041b000000040100002900000b1e00100198000021600000c13d00000006010000290000000101100039000300000001001d000000000010043f0000000501000039000000200010043f000000000100041400000a560010009c00000a5601008041000000c00110021000000b0e011001c700008010020000392953294e0000040f0000000100200190000021710000613d000000000101043b000000000101041a000000000001004b000021600000c13d0000000101000039000000000101041a000000030010006b000021600000613d0000000301000029000000000010043f0000000501000039000000200010043f000000000100041400000a560010009c00000a5601008041000000c00110021000000b0e011001c700008010020000392953294e0000040f0000000100200190000021710000613d000000000101043b0000000402000029000000000021041b000000000100041400000a560010009c00000a5601008041000000c00110021000000a5d011001c70000800d02000039000000040300003900000b2004000041000000070500002900000005060000290000000607000029295329490000040f0000000100200190000021710000613d000000050000006b0000217c0000613d000000000001042d0000000001000019000029550001043000000b4601000041000000000010043f00000afc01000041000029550001043000000b5101000041000000000010043f00000afc010000410000295500010430000000000001042f00000b5601000041000000000010043f00000afc010000410000295500010430000000400100043d000000440210003900000b5503000041000000000032043500000024021000390000001303000039000000000032043500000a7202000041000000000021043500000004021000390000002003000039000000000032043500000a560010009c00000a5601008041000000400110021000000a73011001c70000295500010430000000400100043d000000640210003900000b53030000410000000000320435000000440210003900000b5403000041000000000032043500000024021000390000002a03000039000000000032043500000a7202000041000000000021043500000004021000390000002003000039000000000032043500000a560010009c00000a5601008041000000400110021000000af3011001c7000029550001043000000b5201000041000000000010043f00000afc010000410000295500010430000000000301001900000000011200a9000000000003004b000021b00000613d00000000033100d9000000000023004b000021b10000c13d000000000001042d00000b1d01000041000000000010043f0000001101000039000000040010043f00000a75010000410000295500010430000000000001004b000021ba0000613d000000000001042d000000400100043d000000440210003900000b3f03000041000000000032043500000024021000390000001d03000039000000000032043500000a7202000041000000000021043500000004021000390000002003000039000000000032043500000a560010009c00000a5601008041000000400110021000000a73011001c7000029550001043000000a5b01100198000021dd0000613d000000000010043f0000000601000039000000200010043f000000000100041400000a560010009c00000a5601008041000000c00110021000000b0e011001c700008010020000392953294e0000040f0000000100200190000021e10000613d000000000101043b000000000101041a00000a5901100197000000000001042d00000b3801000041000000000010043f00000afc01000041000029550001043000000000010000190000295500010430000000000001004b000021e60000613d000000000001042d000000400100043d000000440210003900000b4203000041000000000032043500000024021000390000001303000039000000000032043500000a7202000041000000000021043500000004021000390000002003000039000000000032043500000a560010009c00000a5601008041000000400110021000000a73011001c700002955000104300007000000000002000500000001001d000000400400043d0000001b01000039000000000101041a000700000001001d00000a5b02100198000023820000613d0000001a01000039000000000101041a000600000001001d00020a5b0010019c000023820000613d00000b30010000410000000001140436000300000001001d00000a560040009c00000a560100004100000000010440190000004001100210000000000300041400000a560030009c00000a5603008041000000c003300210000000000113019f00000afc011001c7000400000004001d2953294e0000040f000000040b000029000000600310027000000a5603300197000000a00030008c000000a00400003900000000040340190000001f0640018f000000e00740019000000000057b0019000022230000613d000000000801034f00000000090b0019000000008a08043c0000000009a90436000000000059004b0000221f0000c13d000000000006004b000022300000613d000000000771034f0000000306600210000000000805043300000000086801cf000000000868022f000000000707043b0000010006600089000000000767022f00000000066701cf000000000686019f00000000006504350000000100200190000023920000613d0000001f01400039000001e00210018f0000000001b20019000000000021004b0000000002000039000000010200403900000a590010009c0000236a0000213d00000001002001900000236a0000c13d000000400010043f000000a00030008c000023680000413d00000000020b043300000b310020009c000023680000213d0000008002b00039000000000202043300000b310020009c000023680000213d0000000302000029000000000202043300000afa0020009c000023720000213d000000000002004b000023720000613d000100000002001d0000006001b000390000000001010433000400000001001d00000a65010000410000000000100443000000000100041400000a560010009c00000a5601008041000000c00110021000000a66011001c70000800b020000392953294e0000040f0000000100200190000023700000613d0000000702000029000000a00220027000000a5602200197000000000101043b000000000121004b000023620000413d000000040010006b000023710000a13d000000400200043d000400000002001d00000b30010000410000000001120436000300000001001d00000a560020009c00000a560100004100000000010240190000004001100210000000000200041400000a560020009c00000a5602008041000000c002200210000000000112019f00000afc011001c700000002020000292953294e0000040f000000040b000029000000600310027000000a5603300197000000a00030008c000000a00400003900000000040340190000001f0640018f000000e00740019000000000057b0019000022840000613d000000000801034f00000000090b0019000000008a08043c0000000009a90436000000000059004b000022800000c13d000000000006004b000022910000613d000000000771034f0000000306600210000000000805043300000000086801cf000000000868022f000000000707043b0000010006600089000000000767022f00000000066701cf000000000686019f000000000065043500000001002001900000239e0000613d0000001f01400039000001e00210018f0000000001b20019000000000021004b0000000002000039000000010200403900000a590010009c0000236a0000213d00000001002001900000236a0000c13d000000400010043f000000a00030008c000023680000413d00000000020b043300000b310020009c000023680000213d0000008002b00039000000000202043300000b310020009c000023680000213d0000000302000029000000000202043300000afa0020009c000023720000213d000000000002004b000023720000613d000300000002001d0000006001b000390000000001010433000400000001001d00000a65010000410000000000100443000000000100041400000a560010009c00000a5601008041000000c00110021000000a66011001c70000800b020000392953294e0000040f0000000100200190000023700000613d0000000602000029000000a00220027000000a5602200197000000000101043b000000000121004b000023620000413d000000040010006b000023710000a13d0000000701000029000000c00110027000000a5602100197000000120120008900000a560010009c000023620000213d000000120020008c000022ce0000c13d0000000102000039000022d70000013d0000000a030000390000000102000039000000010010019000000000043300a9000000010300603900000000022300a900000001011002720000000003040019000022d00000c13d00000001052000b900000001015000fa000000000021004b000023620000c13d0000000601000029000000c00110027000000a5602100197000000120120008900000a560010009c000023620000213d000000120020008c000022e50000c13d0000000102000039000022ee0000013d0000000a030000390000000102000039000000010010019000000000043300a9000000010300603900000000022300a900000001011002720000000003040019000022e70000c13d00000003012000b9000400000001001d00000003011000fa000000000021004b000023620000c13d000600000005001d0000001701000039000000000201041a000000400300043d000700000003001d00000b0301000041000000000013043500000a560030009c00000a560100004100000000010340190000004001100210000000000300041400000a560030009c00000a5603008041000000c003300210000000000113019f00000afc011001c700000a5b022001972953294e0000040f000000070b000029000000600310027000000a5603300197000000200030008c000000200400003900000000040340190000001f0640018f000000200740019000000000057b0019000023160000613d000000000801034f00000000090b0019000000008a08043c0000000009a90436000000000059004b000023120000c13d000000000006004b000023230000613d000000000771034f0000000306600210000000000805043300000000086801cf000000000868022f000000000707043b0000010006600089000000000767022f00000000066701cf000000000686019f00000000006504350000000100200190000023aa0000613d0000001f01400039000000600210018f0000000001b20019000000000021004b0000000002000039000000010200403900000a590010009c00000006050000290000236a0000213d00000001002001900000236a0000c13d000000400010043f000000200030008c000023680000413d00000000020b0433000000ff0020008c000023680000213d0000001201200089000000ff0010008c000023620000213d000000120020008c0000233d0000c13d0000000102000039000023460000013d0000000a030000390000000102000039000000010010019000000000043300a9000000010300603900000000022300a9000000010110027200000000030400190000233f0000c13d00000005015000b9000000050000006b0000234c0000613d00000005031000fa000000000053004b000023620000c13d0000000403000029000000000003004b000023c80000613d000000000002004b000023c80000613d00000000033100d900000000012300d90000001904000039000000000404041a000000000004004b000023610000613d000027100040008c000023620000213d000027100540008900000000041500a9000000000032004b000023600000213d00000000011400d9000000000051004b000023620000c13d000027100140011a000000000001042d00000b1d01000041000000000010043f0000001101000039000000040010043f00000a750100004100002955000104300000000001000019000029550001043000000b1d01000041000000000010043f0000004101000039000000040010043f00000a75010000410000295500010430000000000001042f000000400100043d000000440210003900000b3203000041000000000032043500000024021000390000001c03000039000000000032043500000a7202000041000000000021043500000004021000390000002003000039000000000032043500000a560010009c00000a5601008041000000400110021000000a73011001c70000295500010430000000440140003900000b3e02000041000000000021043500000024014000390000001c02000039000000000021043500000a7201000041000000000014043500000004014000390000002002000039000000000021043500000a560040009c00000a5604008041000000400140021000000a73011001c700002955000104300000001f0530018f00000a5806300198000000400200043d0000000004620019000023b50000613d000000000701034f0000000008020019000000007907043c0000000008980436000000000048004b000023990000c13d000023b50000013d0000001f0530018f00000a5806300198000000400200043d0000000004620019000023b50000613d000000000701034f0000000008020019000000007907043c0000000008980436000000000048004b000023a50000c13d000023b50000013d0000001f0530018f00000a5806300198000000400200043d0000000004620019000023b50000613d000000000701034f0000000008020019000000007907043c0000000008980436000000000048004b000023b10000c13d000000000005004b000023c20000613d000000000161034f0000000305500210000000000604043300000000065601cf000000000656022f000000000101043b0000010005500089000000000151022f00000000015101cf000000000161019f0000000000140435000000600130021000000a560020009c00000a56020080410000004002200210000000000112019f000029550001043000000b1d01000041000000000010043f0000001201000039000000040010043f00000a75010000410000295500010430000000000001004b000023d10000613d000000000001042d000000400100043d000000440210003900000b3b03000041000000000032043500000024021000390000001703000039000000000032043500000a7202000041000000000021043500000004021000390000002003000039000000000032043500000a560010009c00000a5601008041000000400110021000000a73011001c700002955000104300008000000000002000100000004001d000500000002001d000600000001001d000700000003001d000000000003004b000025650000613d0000000701000029000000000010043f0000000501000039000000200010043f000000000100041400000a560010009c00000a5601008041000000c00110021000000b0e011001c700008010020000392953294e0000040f0000000100200190000025630000613d000000000101043b000000000101041a000000000001004b000024120000c13d0000000101000039000000000101041a000000070010006c000025650000a13d0000000702000029000000010220008a000800000002001d000000000020043f0000000501000039000000200010043f000000000100041400000a560010009c00000a5601008041000000c00110021000000b0e011001c700008010020000392953294e0000040f0000000100200190000025630000613d000000000101043b000000000101041a000000000001004b0000000802000029000023ff0000613d00000b0f00100198000025650000c13d000000060200002900000a5b02200197000300000001001d00000a5b01100197000800000002001d000000000021004b0000256a0000c13d0000000701000029000000000010043f0000000701000039000000200010043f000000000100041400000a560010009c00000a5601008041000000c00110021000000b0e011001c700008010020000392953294e0000040f0000000100200190000025630000613d000000000101043b000200000001001d000000000301041a000000000400041100000a5b02400197000400000002001d000000080020006c000600000003001d000024530000613d000000040030006b000024530000613d0000000801000029000000000010043f0000000801000039000000200010043f000000000100041400000a560010009c00000a5601008041000000c00110021000000b0e011001c700008010020000392953294e0000040f0000000100200190000025630000613d000000000101043b0000000402000029000000000020043f000000200010043f000000000100041400000a560010009c00000a5601008041000000c00110021000000b0e011001c700008010020000392953294e0000040f0000000100200190000025630000613d000000000101043b000000000101041a000000ff00100190000000060300002900000000040004110000259b0000613d000000080000006b0000246a0000613d0000001e01000039000000000101041a000000ff00100190000025720000613d000000000040043f0000001f01000039000000200010043f000000000100041400000a560010009c00000a5601008041000000c00110021000000b0e011001c700008010020000392953294e0000040f0000000100200190000025630000613d000000000101043b000000000101041a000000ff001001900000000603000029000025830000c13d000000000003004b0000246e0000613d0000000201000029000000000001041b0000000801000029000000000010043f0000000601000039000000200010043f000000000100041400000a560010009c00000a5601008041000000c00110021000000b0e011001c700008010020000392953294e0000040f0000000100200190000025630000613d000000000101043b000000000201041a000000010220008a000000000021041b000000050100002900000a5b01100197000600000001001d000000000010043f0000000601000039000000200010043f000000000100041400000a560010009c00000a5601008041000000c00110021000000b0e011001c700008010020000392953294e0000040f0000000100200190000025630000613d000000000101043b000000000201041a0000000102200039000000000021041b00000a65010000410000000000100443000000000100041400000a560010009c00000a5601008041000000c00110021000000a66011001c70000800b020000392953294e0000040f0000000100200190000025690000613d000000000101043b000200000001001d0000000701000029000000000010043f0000000501000039000000200010043f000000000100041400000a560010009c00000a5601008041000000c00110021000000b0e011001c700008010020000392953294e0000040f0000000100200190000025630000613d0000000202000029000000a00220021000000006022001af00000b1e022001c7000000000101043b000000000021041b000000030100002900000b1e00100198000024dc0000c13d00000007010000290000000101100039000200000001001d000000000010043f0000000501000039000000200010043f000000000100041400000a560010009c00000a5601008041000000c00110021000000b0e011001c700008010020000392953294e0000040f0000000100200190000025630000613d000000000101043b000000000101041a000000000001004b000024dc0000c13d0000000101000039000000000101041a000000020010006b000024dc0000613d0000000201000029000000000010043f0000000501000039000000200010043f000000000100041400000a560010009c00000a5601008041000000c00110021000000b0e011001c700008010020000392953294e0000040f0000000100200190000025630000613d000000000101043b0000000302000029000000000021041b000000000100041400000a560010009c00000a5601008041000000c00110021000000a5d011001c70000800d02000039000000040300003900000b2004000041000000080500002900000006060000290000000707000029295329490000040f0000000100200190000025630000613d000000060000006b0000256e0000613d00000af801000041000000000010044300000005010000290000000400100443000000000100041400000a560010009c00000a5601008041000000c00110021000000af9011001c700008002020000392953294e0000040f0000000100200190000025690000613d000000000101043b000000000001004b000025620000613d000000400700043d00000064017000390000008002000039000500000002001d000000000021043500000044017000390000000702000029000000000021043500000024017000390000000802000029000000000021043500000b210100004100000000001704350000000401700039000000040200002900000000002104350000008402700039000000010100002900000000310104340000000000120435000000a402700039000000000001004b0000251b0000613d000000000400001900000000052400190000000006430019000000000606043300000000006504350000002004400039000000000014004b000025140000413d0000001f0310003900000b4d0330019700000000012100190000000000010435000000a40130003900000a560010009c00000a5601008041000000600110021000000a560070009c00000a560200004100000000020740190000004002200210000000000121019f000000000200041400000a560020009c00000a5602008041000000c002200210000000000112019f0000000602000029000800000007001d295329490000040f000000080b000029000000600310027000000a5603300197000000200030008c000000200400003900000000040340190000001f0640018f000000200740019000000000057b0019000025400000613d000000000801034f00000000090b0019000000008a08043c0000000009a90436000000000059004b0000253c0000c13d000000000006004b0000254d0000613d000000000771034f0000000306600210000000000805043300000000086801cf000000000868022f000000000707043b0000010006600089000000000767022f00000000066701cf000000000686019f00000000006504350000000100200190000025970000613d0000001f01400039000000600210018f0000000001b20019000000000021004b0000000002000039000000010200403900000a590010009c000025cd0000213d0000000100200190000025cd0000c13d000000400010043f000000200030008c000025630000413d00000000010b043300000b2200100198000025630000c13d00000b230110019700000b210010009c000025c90000c13d000000000001042d0000000001000019000029550001043000000b4601000041000000000010043f00000afc010000410000295500010430000000000001042f00000b5101000041000000000010043f00000afc01000041000029550001043000000b5601000041000000000010043f00000afc010000410000295500010430000000400100043d000000440210003900000b5503000041000000000032043500000024021000390000001303000039000000000032043500000a7202000041000000000021043500000004021000390000002003000039000000000032043500000a560010009c00000a5601008041000000400110021000000a73011001c70000295500010430000000400100043d000000640210003900000b53030000410000000000320435000000440210003900000b5403000041000000000032043500000024021000390000002a03000039000000000032043500000a7202000041000000000021043500000004021000390000002003000039000000000032043500000a560010009c00000a5601008041000000400110021000000af3011001c70000295500010430000000000003004b0000259f0000c13d0000006002000039000025c60000013d00000b5201000041000000000010043f00000afc0100004100002955000104300000001f0230003900000a57022001970000003f0220003900000af704200197000000400200043d0000000004420019000000000024004b0000000005000039000000010500403900000a590040009c000025cd0000213d0000000100500190000025cd0000c13d000000400040043f0000001f0430018f000000000632043600000a5805300198000500000006001d0000000003560019000025b90000613d000000000601034f0000000507000029000000006806043c0000000007870436000000000037004b000025b50000c13d000000000004004b000025c60000613d000000000151034f0000000304400210000000000503043300000000054501cf000000000545022f000000000101043b0000010004400089000000000141022f00000000014101cf000000000151019f00000000001304350000000001020433000000000001004b000025d30000c13d00000b2401000041000000000010043f00000afc01000041000029550001043000000b1d01000041000000000010043f0000004101000039000000040010043f00000a75010000410000295500010430000000050200002900000a560020009c00000a5602008041000000400220021000000a560010009c00000a56010080410000006001100210000000000121019f0000295500010430000000000001004b000025df0000613d000000000001042d000000400100043d000000440210003900000b2703000041000000000032043500000024021000390000001503000039000000000032043500000a7202000041000000000021043500000004021000390000002003000039000000000032043500000a560010009c00000a5601008041000000400110021000000a73011001c70000295500010430000000000100041a00000a5b021001970000000001000411000000000012004b000025f60000c13d000000000001042d000000400200043d00000b010300004100000000003204350000000403200039000000000013043500000a560020009c00000a5602008041000000400120021000000a75011001c700002955000104300001000000000002000000000001004b000026310000613d000100000001001d000000000010043f0000000501000039000000200010043f000000000100041400000a560010009c00000a5601008041000000c00110021000000b0e011001c700008010020000392953294e0000040f00000001002001900000262f0000613d000000000101043b000000000101041a000000000001004b0000262c0000c13d0000000101000039000000000101041a0000000102000029000000000021004b000026310000a13d000000010220008a000100000002001d000000000020043f0000000501000039000000200010043f000000000100041400000a560010009c00000a5601008041000000c00110021000000b0e011001c700008010020000392953294e0000040f00000001002001900000262f0000613d000000000101043b000000000101041a000000000001004b0000000102000029000026190000613d00000b0f00100198000026310000c13d000000000001042d0000000001000019000029550001043000000b4601000041000000000010043f00000afc0100004100002955000104300003000000000002000000400400043d00000044054000390000000000350435000000200340003900000af505000041000000000053043500000a5b02200197000000240540003900000000002504350000004402000039000000000024043500000b570040009c000026ad0000813d0000008002400039000000400020043f00000a560030009c00000a56030080410000004002300210000000000304043300000a560030009c00000a56030080410000006003300210000000000223019f000000000300041400000a560030009c00000a5603008041000000c003300210000000000332019f00000a5b021001970000000001030019000300000002001d295329490000040f000000600310027000000a5603300198000026800000613d0000001f0430003900000a57044001970000003f0440003900000af704400197000000400a00043d00000000044a00190000000000a4004b0000000005000039000000010500403900000a590040009c000026ad0000213d0000000100500190000026ad0000c13d000000400040043f0000001f0430018f00000000093a043600000a58053001980000000003590019000026720000613d000000000601034f0000000007090019000000006806043c0000000007870436000000000037004b0000266e0000c13d000000000004004b000026820000613d000000000151034f0000000304400210000000000503043300000000054501cf000000000545022f000000000101043b0000010004400089000000000141022f00000000014101cf000000000151019f0000000000130435000026820000013d000000600a000039000000800900003900000000010a04330000000100200190000026b30000613d000000000001004b0000269e0000c13d00020000000a001d000100000009001d00000af801000041000000000010044300000003010000290000000400100443000000000100041400000a560010009c00000a5601008041000000c00110021000000af9011001c700008002020000392953294e0000040f0000000100200190000026c80000613d000000000101043b000000000001004b0000000201000029000026c90000613d0000000001010433000000000001004b0000000109000029000026aa0000613d00000afa0010009c000026ab0000213d000000200010008c000026ab0000413d0000000001090433000000000001004b0000000002000039000000010200c039000000000021004b000026ab0000c13d000000000001004b000026bd0000613d000000000001042d0000000001000019000029550001043000000b1d01000041000000000010043f0000004101000039000000040010043f00000a75010000410000295500010430000000000001004b000026c00000c13d000000400100043d00000afb02000041000000000021043500000a560010009c00000a5601008041000000400110021000000afc011001c70000295500010430000000400100043d00000afd02000041000026cb0000013d00000a560090009c00000a5609008041000000400290021000000a560010009c00000a56010080410000006001100210000000000121019f0000295500010430000000000001042f000000400100043d00000afe02000041000000000021043500000004021000390000000303000029000000000032043500000a560010009c00000a5601008041000000400110021000000a75011001c700002955000104300003000000000002000000400500043d0000006406500039000000000046043500000a5b0330019700000044045000390000000000340435000000200350003900000b5804000041000000000043043500000a5b02200197000000240450003900000000002404350000006402000039000000000025043500000b590050009c0000274f0000813d000000a002500039000000400020043f00000a560030009c00000a56030080410000004002300210000000000305043300000a560030009c00000a56030080410000006003300210000000000223019f000000000300041400000a560030009c00000a5603008041000000c003300210000000000332019f00000a5b021001970000000001030019000300000002001d295329490000040f000000600310027000000a5603300198000027220000613d0000001f0430003900000a57044001970000003f0440003900000af704400197000000400a00043d00000000044a00190000000000a4004b0000000005000039000000010500403900000a590040009c0000274f0000213d00000001005001900000274f0000c13d000000400040043f0000001f0430018f00000000093a043600000a58053001980000000003590019000027140000613d000000000601034f0000000007090019000000006806043c0000000007870436000000000037004b000027100000c13d000000000004004b000027240000613d000000000151034f0000000304400210000000000503043300000000054501cf000000000545022f000000000101043b0000010004400089000000000141022f00000000014101cf000000000151019f0000000000130435000027240000013d000000600a000039000000800900003900000000010a04330000000100200190000027550000613d000000000001004b000027400000c13d00020000000a001d000100000009001d00000af801000041000000000010044300000003010000290000000400100443000000000100041400000a560010009c00000a5601008041000000c00110021000000af9011001c700008002020000392953294e0000040f00000001002001900000276a0000613d000000000101043b000000000001004b00000002010000290000276b0000613d0000000001010433000000000001004b00000001090000290000274c0000613d00000afa0010009c0000274d0000213d000000200010008c0000274d0000413d0000000001090433000000000001004b0000000002000039000000010200c039000000000021004b0000274d0000c13d000000000001004b0000275f0000613d000000000001042d0000000001000019000029550001043000000b1d01000041000000000010043f0000004101000039000000040010043f00000a75010000410000295500010430000000000001004b000027620000c13d000000400100043d00000afb02000041000000000021043500000a560010009c00000a5601008041000000400110021000000afc011001c70000295500010430000000400100043d00000afd020000410000276d0000013d00000a560090009c00000a5609008041000000400290021000000a560010009c00000a56010080410000006001100210000000000121019f0000295500010430000000000001042f000000400100043d00000afe02000041000000000021043500000004021000390000000303000029000000000032043500000a560010009c00000a5601008041000000400110021000000a75011001c700002955000104300008000000000002000300000001001d000000400100043d000500000001001d00000b4f0010009c000028980000813d00000005010000290000002003100039000400000003001d000000400030043f0000000000010435000100000002001d000000000002004b000028a80000613d0000000101000039000000000101041a000800000001001d00000a65010000410000000000100443000000000100041400000a560010009c00000a5601008041000000c00110021000000a66011001c70000800b020000392953294e0000040f0000000100200190000028a70000613d000000000101043b000700000001001d0000000801000029000000000010043f0000000501000039000000200010043f000000000100041400000a560010009c00000a5601008041000000c00110021000000b0e011001c700008010020000392953294e0000040f0000000100200190000028640000613d000000030200002900000a5b042001970000000702000029000000a0022002100000000103000029000000010030008c000000000300001900000b1e03006041000000000223019f000000000242019f000000000101043b000000000021041b000700000004001d000000000040043f0000000601000039000000200010043f000000000100041400000a560010009c00000a5601008041000000c00110021000000b0e011001c700008010020000392953294e0000040f0000000100200190000028640000613d000000010400002900000b1f024000d1000000000101043b000000000301041a0000000002230019000000000021041b000000070000006b000028ac0000613d000600080040002d0000000001000019000027d50000013d0000000807000029000000000100041400000a560010009c00000a5601008041000000c00110021000000a5d011001c70000800d02000039000000040300003900000b200400004100000000050000190000000706000029000800000007001d295329490000040f00000001002001900000000101000039000028640000613d0000000100100190000027c50000613d00000008070000290000000107700039000000060070006c000027c60000c13d00000001010000390000000602000029000000000021041b00000af801000041000000000010044300000003010000290000000400100443000000000100041400000a560010009c00000a5601008041000000c00110021000000af9011001c700008002020000392953294e0000040f0000000100200190000028a70000613d000000000101043b000000000001004b0000000406000029000028630000613d000000400700043d000000000100041100020a5b0010019b0000000101000039000000000201041a000300000002001d000000010320006a000100800000003d00000064017000390000008002000039000000000021043500000b210100004100000000001704350000000401700039000000020200002900000000002104350000004401700039000600000003001d0000000000310435000000240170003900000000000104350000000501000029000000000101043300000084027000390000000000120435000000a402700039000000000001004b000028130000613d000000000300001900000000042300190000000005630019000000000505043300000000005404350000002003300039000000000013004b0000280c0000413d0000001f0310003900000b4d0330019700000000012100190000000000010435000000a40130003900000a560010009c00000a5601008041000000600110021000000a560070009c00000a560200004100000000020740190000004002200210000000000121019f000000000200041400000a560020009c00000a5602008041000000c002200210000000000112019f0000000702000029000800000007001d295329490000040f000000080a000029000000600310027000000a5603300197000000200030008c00000020040000390000000004034019000000200640019000000000056a0019000028370000613d000000000701034f00000000080a0019000000007907043c0000000008980436000000000058004b000028330000c13d0000001f07400190000028440000613d000000000661034f0000000307700210000000000805043300000000087801cf000000000878022f000000000606043b0000010007700089000000000676022f00000000067601cf000000000686019f00000000006504350000000100200190000028660000613d0000001f01400039000000600210018f0000000001a20019000000000021004b0000000002000039000000010200403900000a590010009c0000000406000029000028980000213d0000000100200190000028980000c13d000000400010043f000000200030008c000028640000413d00000000020a043300000b2200200198000028640000c13d00000b230220019700000b210020009c0000000603000029000028940000c13d0000000103300039000000030030006c0000000007010019000027f70000413d0000000101000039000000000101041a000000030010006c000028640000c13d000000000001042d00000000010000190000295500010430000000000003004b0000286a0000c13d0000006002000039000028910000013d0000001f0230003900000a57022001970000003f0220003900000af704200197000000400200043d0000000004420019000000000024004b0000000005000039000000010500403900000a590040009c000028980000213d0000000100500190000028980000c13d000000400040043f0000001f0430018f000000000632043600000a5805300198000100000006001d0000000003560019000028840000613d000000000601034f0000000107000029000000006806043c0000000007870436000000000037004b000028800000c13d000000000004004b000028910000613d000000000151034f0000000304400210000000000503043300000000054501cf000000000545022f000000000101043b0000010004400089000000000141022f00000000014101cf000000000151019f00000000001304350000000001020433000000000001004b0000289e0000c13d00000b2401000041000000000010043f00000afc01000041000029550001043000000b1d01000041000000000010043f0000004101000039000000040010043f00000a75010000410000295500010430000000010200002900000a560020009c00000a5602008041000000400220021000000a560010009c00000a56010080410000006001100210000000000121019f0000295500010430000000000001042f00000b2601000041000000000010043f00000afc01000041000029550001043000000b2501000041000000000010043f00000afc01000041000029550001043000020000000000020000000202000039000000000302041a00000b4c033001670000000102000039000000000202041a0000000002320019000000000012001a0000290f0000413d000100000001001d00000000011200190000000c02000039000000000202041a000000000021004b000028e90000213d0000000f01000039000000000101041a000200000001001d00000a65010000410000000000100443000000000100041400000a560010009c00000a5601008041000000c00110021000000a66011001c70000800b020000392953294e0000040f0000000100200190000028f00000613d000000000101043b000000020010006c000028f10000413d000000000100041100000a5b01100198000028f80000613d000000000010043f0000000601000039000000200010043f000000000100041400000a560010009c00000a5601008041000000c00110021000000b0e011001c700008010020000392953294e0000040f0000000100200190000028fc0000613d0000000e02000039000000000202041a000000000101043b000000000101041a00000a5901100197000000000112004b0000000001004019000000010010006c000028fe0000413d000000000001042d000000400100043d000000440210003900000b2703000041000000000032043500000024021000390000001503000039000029040000013d000000000001042f000000400100043d000000440210003900000b3903000041000000000032043500000024021000390000001603000039000029040000013d00000b3801000041000000000010043f00000afc01000041000029550001043000000000010000190000295500010430000000400100043d000000440210003900000b3703000041000000000032043500000024021000390000001b03000039000000000032043500000a7202000041000000000021043500000004021000390000002003000039000000000032043500000a560010009c00000a5601008041000000400110021000000a73011001c7000029550001043000000b1d01000041000000000010043f0000001101000039000000040010043f00000a75010000410000295500010430000000000001042f00000a560010009c00000a5601008041000000400110021000000a560020009c00000a56020080410000006002200210000000000112019f000000000200041400000a560020009c00000a5602008041000000c002200210000000000112019f00000a5d011001c700008010020000392953294e0000040f0000000100200190000029290000613d000000000101043b000000000001042d0000000001000019000029550001043000000000050100190000000000200443000000050030008c000029390000413d000000040100003900000000020000190000000506200210000000000664001900000005066002700000000006060031000000000161043a0000000102200039000000000031004b000029310000413d00000a560030009c00000a56030080410000006001300210000000000200041400000a560020009c00000a5602008041000000c002200210000000000112019f00000b5a011001c700000000020500192953294e0000040f0000000100200190000029480000613d000000000101043b000000000001042d000000000001042f0000294c002104210000000102000039000000000001042d0000000002000019000000000001042d00002951002104230000000102000039000000000001042d0000000002000019000000000001042d0000295300000432000029540001042e00002955000104300000000000000000000000000000000000000000000000000000000000000000000000000000000000000000ffffffff00000000000000000000000000000000000000000000000000000001ffffffe000000000000000000000000000000000000000000000000000000000ffffffe0000000000000000000000000000000000000000000000000ffffffffffffffff8000000000000000000000000000000000000000000000000000000000000000000000000000000000000000ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff000000000000000000000000000000000000000002000000000000000000000000000000000000000000000000000000000000008be0079c531659141344cd1fd0a4f28419497f9722a3daafe3b4186f6b6457e03da8a5f161a6c3ff06a60736d0ed24d7963cc6a5c4fafd2fa1dae9bb908e07a5c2575a0e9e593c00f959f8c92f12db2869c3395a3b0502d05e2516446f71f85b3da8a5f161a6c3ff06a60736d0ed24d7963cc6a5c4fafd2fa1dae9bb908e07a475ca53043ea007e5c65182cbb028f60d7179ff4b55739a3949b401801c942e658a35acfbc15ff81a39ae7d344fd709f28e8600b4aa8c65c6b64bfe7fe36bd19b75ca53043ea007e5c65182cbb028f60d7179ff4b55739a3949b401801c942e64796b89b91644bc98cd93958e4c9038275d622183e25ac5af08cc6b5d955391320200000200000000000000000000000000000004000000000000000000000000319284ad7d4265c99e51f9e0112e2425b1ad54f8c4e06d7a4191eaa263c72b14ce6d7b5282bd9a3661ae061feed1dbda4e52ab073b1f9285be6e155d9c38d4ec319284ad7d4265c99e51f9e0112e2425b1ad54f8c4e06d7a4191eaa263c72b1339a5844729cae3e308f36a5ce933956d7c6367997d26743ca06a70b77c062d58c65a7bb8d6351c1cf70c95a316cc6a92839c986682d98bc35f958f4883f9d2a839a5844729cae3e308f36a5ce933956d7c6367997d26743ca06a70b77c062d57fe8a4859c7bd88fc0f24184464406785daae8e84cb1860cc4a4eff72e05fe2470175b7a638427703f0dbe7bb9bbf987a2551717b34e79f33b5b1008d1fa01db9fe8a4859c7bd88fc0f24184464406785daae8e84cb1860cc4a4eff72e05fe2460000000200000000000000000000000000000180000001000000000000000000496e76616c696420636f6d6d697373696f6e2070657263656e7461676500000008c379a00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000640000000000000000000000001e4fbdf700000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000002400000000000000000000000000000000000000000000000000000000000000000000000000000000c7dd451700000000000000000000000000000000000000000000000000000000848769dd00000000000000000000000000000000000000000000000000000000c87b56dc00000000000000000000000000000000000000000000000000000000e985e9c400000000000000000000000000000000000000000000000000000000f2c4ce1d00000000000000000000000000000000000000000000000000000000f4f3b1ff00000000000000000000000000000000000000000000000000000000f4f3b20000000000000000000000000000000000000000000000000000000000f799bed400000000000000000000000000000000000000000000000000000000f9f92be400000000000000000000000000000000000000000000000000000000f2c4ce1e00000000000000000000000000000000000000000000000000000000f2fde38b00000000000000000000000000000000000000000000000000000000f45a2cd700000000000000000000000000000000000000000000000000000000eb7c571c00000000000000000000000000000000000000000000000000000000eb7c571d00000000000000000000000000000000000000000000000000000000ee5c2e6600000000000000000000000000000000000000000000000000000000f0bc91c000000000000000000000000000000000000000000000000000000000e985e9c500000000000000000000000000000000000000000000000000000000eae483c500000000000000000000000000000000000000000000000000000000d8cde1c500000000000000000000000000000000000000000000000000000000e26127e700000000000000000000000000000000000000000000000000000000e26127e800000000000000000000000000000000000000000000000000000000e58306f900000000000000000000000000000000000000000000000000000000e8a3d48500000000000000000000000000000000000000000000000000000000d8cde1c600000000000000000000000000000000000000000000000000000000db2e21bc00000000000000000000000000000000000000000000000000000000dfef66f500000000000000000000000000000000000000000000000000000000cce54e4300000000000000000000000000000000000000000000000000000000cce54e4400000000000000000000000000000000000000000000000000000000cd05de6d00000000000000000000000000000000000000000000000000000000d5abeb0100000000000000000000000000000000000000000000000000000000c87b56dd00000000000000000000000000000000000000000000000000000000cc1acb9d00000000000000000000000000000000000000000000000000000000a22cb46400000000000000000000000000000000000000000000000000000000b88d4fdd00000000000000000000000000000000000000000000000000000000c2e5ec0300000000000000000000000000000000000000000000000000000000c2e5ec0400000000000000000000000000000000000000000000000000000000c4be5b5900000000000000000000000000000000000000000000000000000000c627525500000000000000000000000000000000000000000000000000000000b88d4fde00000000000000000000000000000000000000000000000000000000bfac4d0f00000000000000000000000000000000000000000000000000000000c01b59db00000000000000000000000000000000000000000000000000000000ac3adb1700000000000000000000000000000000000000000000000000000000ac3adb1800000000000000000000000000000000000000000000000000000000acaa27fc00000000000000000000000000000000000000000000000000000000ad9e5e3e00000000000000000000000000000000000000000000000000000000a22cb46500000000000000000000000000000000000000000000000000000000a945bf8000000000000000000000000000000000000000000000000000000000925f789500000000000000000000000000000000000000000000000000000000944964ce00000000000000000000000000000000000000000000000000000000944964cf0000000000000000000000000000000000000000000000000000000095d89b4100000000000000000000000000000000000000000000000000000000a0712d6800000000000000000000000000000000000000000000000000000000925f789600000000000000000000000000000000000000000000000000000000938e3d7b000000000000000000000000000000000000000000000000000000008add9f05000000000000000000000000000000000000000000000000000000008add9f06000000000000000000000000000000000000000000000000000000008d56ef1d000000000000000000000000000000000000000000000000000000008da5cb5b00000000000000000000000000000000000000000000000000000000848769de00000000000000000000000000000000000000000000000000000000849355dc0000000000000000000000000000000000000000000000000000000043d3e4680000000000000000000000000000000000000000000000000000000062b974e1000000000000000000000000000000000000000000000000000000006f8b44af000000000000000000000000000000000000000000000000000000007225037f0000000000000000000000000000000000000000000000000000000072250380000000000000000000000000000000000000000000000000000000008034ed0600000000000000000000000000000000000000000000000000000000838aed02000000000000000000000000000000000000000000000000000000006f8b44b00000000000000000000000000000000000000000000000000000000070a0823100000000000000000000000000000000000000000000000000000000715018a6000000000000000000000000000000000000000000000000000000006bac9e8f000000000000000000000000000000000000000000000000000000006bac9e90000000000000000000000000000000000000000000000000000000006bb7b1d9000000000000000000000000000000000000000000000000000000006d5d40c60000000000000000000000000000000000000000000000000000000062b974e2000000000000000000000000000000000000000000000000000000006352211e00000000000000000000000000000000000000000000000000000000535ba42e0000000000000000000000000000000000000000000000000000000055f804b20000000000000000000000000000000000000000000000000000000055f804b3000000000000000000000000000000000000000000000000000000005b8ad42900000000000000000000000000000000000000000000000000000000600dd5ea00000000000000000000000000000000000000000000000000000000535ba42f00000000000000000000000000000000000000000000000000000000537df3b60000000000000000000000000000000000000000000000000000000054214f69000000000000000000000000000000000000000000000000000000004831302800000000000000000000000000000000000000000000000000000000483130290000000000000000000000000000000000000000000000000000000049b05fab000000000000000000000000000000000000000000000000000000004ada218b0000000000000000000000000000000000000000000000000000000043d3e4690000000000000000000000000000000000000000000000000000000044337ea10000000000000000000000000000000000000000000000000000000011131f1e000000000000000000000000000000000000000000000000000000002ccf7144000000000000000000000000000000000000000000000000000000003e5fe94e000000000000000000000000000000000000000000000000000000003e5fe94f0000000000000000000000000000000000000000000000000000000040c442de0000000000000000000000000000000000000000000000000000000042842e0e000000000000000000000000000000000000000000000000000000002ccf7145000000000000000000000000000000000000000000000000000000002f285709000000000000000000000000000000000000000000000000000000003ccfd60b000000000000000000000000000000000000000000000000000000001aef99b6000000000000000000000000000000000000000000000000000000001aef99b70000000000000000000000000000000000000000000000000000000023b872dd000000000000000000000000000000000000000000000000000000002a55205a0000000000000000000000000000000000000000000000000000000011131f1f0000000000000000000000000000000000000000000000000000000018160ddd00000000000000000000000000000000000000000000000000000000083d40c9000000000000000000000000000000000000000000000000000000000b0bfbe8000000000000000000000000000000000000000000000000000000000b0bfbe9000000000000000000000000000000000000000000000000000000000c894cfe000000000000000000000000000000000000000000000000000000000ee196f500000000000000000000000000000000000000000000000000000000083d40ca00000000000000000000000000000000000000000000000000000000095ea7b300000000000000000000000000000000000000000000000000000000039af9ea00000000000000000000000000000000000000000000000000000000039af9eb0000000000000000000000000000000000000000000000000000000006fdde0300000000000000000000000000000000000000000000000000000000081812fc00000000000000000000000000000000000000000000000000000000011013c60000000000000000000000000000000000000000000000000000000001ffc9a7000000000000000000000000000000000000002000000080000000000000000000000000000000000000000000000000ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff00000000000000000000000000000000310ab089e4439a4c15d089f94afb7896ff553aecb10793d0ab882de59d99a32e02000002000000000000000000000000000000440000000000000000000000006e742063616e20776974686472617700000000000000000000000000000000004f6e6c79206f776e6572206f7220636f6d6d697373696f6e2072656369706965000000000000000000000000000000000000008400000000000000000000000070a0823100000000000000000000000000000000000000000000000000000000a9059cbb00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000ffffffffffffff7f00000000000000000000000000000000000000000000000000000003ffffffe01806aa1896bbf26568e884a7374b41e002500962caba6a15023a8d90e8508b8302000002000000000000000000000000000000240000000000000000000000007fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff1425ea420000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000040000000000000000000000005274afe7000000000000000000000000000000000000000000000000000000009996b315000000000000000000000000000000000000000000000000000000009cc7f708afc65944829bd487b90b72536b1951864fbfc14e125fc972a6507f395472616e73666572206661696c65640000000000000000000000000000000000118cdaa7000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000024000000800000000000000000313ce56700000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000400000080000000000000000000000000000000ff000000000000000000000000000000000000000000000000ffffffff0000000000000000ffffffffffffffffffffffffffffffffffffffff0000000000000000ffffffff000000000000000000000000000000000000000000000000000000000000000000000000000000200000000000000000000000000000000000000000000000000000000000000080000000800000000000000000fffffffffffffffffffffffffffffffffffffffffffffffffffffffffefd9a00fffffffffffffffffffffffffffffffffffffffffffffffffffffffffefd99ff546f6f206561726c7920746f20656d657267656e6379207769746864726177003a6357012c1a3ae0a17d304c9920310382d968ebcc4b1771f41c6b304205b57002000000000000000000000000000000000000400000000000000000000000000000000100000000000000000000000000000000000000000000000000000000546f6b656e20646f6573206e6f742065786973740000000000000000000000000000000000184f03e93ff9f4daa797ed6e38ed64bf6a1f01000000000000000000000000000000000000000000000000000004ee2d6d415b85acef810000000000000000000000000000000000000000000004ee2d6d415b85acef80ffffffff000000000000000000000000000000000000000000000000002386f26fc100000000000000000000000000000000000000000000000000000000000005f5e10000ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff30313233343536373839616263646566000000000000000000000000000000007f000000000000000000000000000000000000000000000000000000000000002e6a736f6e000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000ffffffffffffffdf6e6578697374656e7420746f6b656e00000000000000000000000000000000004552433732314d657461646174613a2055524920717565727920666f72206e6f4e487b710000000000000000000000000000000000000000000000000000000000000002000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000010000000000000001ddf252ad1be2c89b69c2b068fc378daa952ba7f163c4a11628f55a4df523b3ef150b7a020000000000000000000000000000000000000000000000000000000000000000ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff00000000000000000000000000000000000000000000000000000000d1a57ed6000000000000000000000000000000000000000000000000000000002e07630000000000000000000000000000000000000000000000000000000000b562e8dd00000000000000000000000000000000000000000000000000000000546f74616c20737570706c792065786365656465640000000000000000000000000000000000000000000000000000000000000000000000ffffffffffffffbf7fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe0496e73756666696369656e742066756e647320666f72206d696e740000000000546965722073616c65206e6f74207374617274656400000000000000000000000200000000000000000000000000000000000020000000000000000000000000c59ca8fed3e5c51f5e82cfb366dfcefc7d26971433b4e88e0be394cfbdfa4a90496e76616c696420646973636f756e742070657263656e7461676500000000000000000000000000000000000000000000000064000000800000000000000000feaf968c0000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000ffffffffffffffffffff496e76616c69642070726963652066726f6d20436861696e6c696e6b00000000000000000000000000000000000000000000006000000000000000000000000017307eab39ab6107e8899845ad3d59bd9653f200f220920489ca2b5937696c31656e740000000000000000000000000000000000000000000000000000000000436f737420697320686967686572207468616e2074686520616d6f756e742073496e76616c696420616d6f756e7420746f206265206d696e74656400000000008f4eb604000000000000000000000000000000000000000000000000000000005075626c69632073616c65206e6f742061637469766500000000000000000000a5d4097edda6d87cb9329af83fb3712ef77eeb13738ffe43cc35a4ce305ad96250726963652070657220746f6b656e206e6f742073657400000000000000000072000000000000000000000000000000000000000000000000000000000000004e6f7420696e2070726573616c65206c69737420666f722074686973207469655072696365206665656420616464726573736573206e6f7420736574000000005061796d656e7420746f6b656e2061646472657373206e6f742073657400000045786365656473206d617820737570706c790000000000000000000000000000457863656564732074696572206d6178206d696e7420616d6f756e74000000005469657220646f6573206e6f74206578697374000000000000000000000000000000000000000000000000000000000000000040000000000000000000000000cfb3b942000000000000000000000000000000000000000000000000000000008c5be1e5ebec7d5bd14f71427d1e84f3dd0314c0f7b2291e5b200ac8c7c3b925df2d9b4200000000000000000000000000000000000000000000000000000000cf4700e4000000000000000000000000000000000000000000000000000000002a55205a0000000000000000000000000000000000000000000000000000000080ac58cd0000000000000000000000000000000000000000000000000000000001ffc9a7000000000000000000000000000000000000000000000000000000005b5e139f00000000000000000000000000000000000000000000000000000000ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffe0ffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff00000000000000000000000000000000000000000000000000ffffffffffffffe00000000000000000000000000000000000000000000000010000000000000000a11481000000000000000000000000000000000000000000000000000000000059c896be000000000000000000000000000000000000000000000000000000007574652074726164657300000000000000000000000000000000000000000000426c61636b6c697374656420656e7469746965732063616e6e6f74206578656354726164696e672069732064697361626c656400000000000000000000000000ea553b3400000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000ffffffffffffff8023b872dd00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000ffffffffffffff6002000002000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000ff13ee9e190ee2ae721380181cb65fe3453ebba2c268e351c4aec3142ed0f38b
Constructor Arguments (ABI-Encoded and is the last bytes of the Contract Creation Code above)
00000000000000000000000000000000000000000000000000000000000001a000000000000000000000000000000000000000000000000000000000000001e0000000000000000000000000000000000000000000000000000000000000022000000000000000000000000000000000000000000000000000000000000027100000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000028000000000000000000000000000000000000000000000000000000000000002a0000000000000000000000000c4abd8d2315f51c6eb95052d64ef52ba9767e373000000000000000000000000c4abd8d2315f51c6eb95052d64ef52ba9767e37300000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000c4abd8d2315f51c6eb95052d64ef52ba9767e3730000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000001441627374726163742078207a6b4d61726b65747300000000000000000000000000000000000000000000000000000000000000000000000000000000000000066162737a6b6d00000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000035697066733a2f2f516d62786f5731595662646e776845686a394a3576644d4a4e37686179536e44514579695445696350794e445a77000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000000035697066733a2f2f516d564e70426d6f353253317432786b4d575675457151663837415452434142484d623861787479703954746b430000000000000000000000
-----Decoded View---------------
Arg [0] : _name (string): Abstract x zkMarkets
Arg [1] : _symbol (string): abszkm
Arg [2] : _contractURI (string): ipfs://QmbxoW1YVbdnwhEhj9J5vdMJN7haySnDQEyiTEicPyNDZw
Arg [3] : _maxSupply (uint256): 10000
Arg [4] : _publicPrice (uint256): 0
Arg [5] : _defaultBaseURI (string):
Arg [6] : _notRevealedURI (string): ipfs://QmVNpBmo52S1t2xkMWVuEqQf87ATRCABHMb8axtyp9TtkC
Arg [7] : _withdrawalRecipientAddress (address): 0xc4ABd8d2315F51C6Eb95052d64eF52Ba9767E373
Arg [8] : _commissionRecipientAddress (address): 0xc4ABd8d2315F51C6Eb95052d64eF52Ba9767E373
Arg [9] : _fixedCommisionThreshold (uint256): 0
Arg [10] : _commissionPercentageIn10000 (uint256): 0
Arg [11] : _defaultRoyaltyRecipient (address): 0xc4ABd8d2315F51C6Eb95052d64eF52Ba9767E373
Arg [12] : _defaultRoyaltyPercentageIn10000 (uint256): 0
-----Encoded View---------------
24 Constructor Arguments found :
Arg [0] : 00000000000000000000000000000000000000000000000000000000000001a0
Arg [1] : 00000000000000000000000000000000000000000000000000000000000001e0
Arg [2] : 0000000000000000000000000000000000000000000000000000000000000220
Arg [3] : 0000000000000000000000000000000000000000000000000000000000002710
Arg [4] : 0000000000000000000000000000000000000000000000000000000000000000
Arg [5] : 0000000000000000000000000000000000000000000000000000000000000280
Arg [6] : 00000000000000000000000000000000000000000000000000000000000002a0
Arg [7] : 000000000000000000000000c4abd8d2315f51c6eb95052d64ef52ba9767e373
Arg [8] : 000000000000000000000000c4abd8d2315f51c6eb95052d64ef52ba9767e373
Arg [9] : 0000000000000000000000000000000000000000000000000000000000000000
Arg [10] : 0000000000000000000000000000000000000000000000000000000000000000
Arg [11] : 000000000000000000000000c4abd8d2315f51c6eb95052d64ef52ba9767e373
Arg [12] : 0000000000000000000000000000000000000000000000000000000000000000
Arg [13] : 0000000000000000000000000000000000000000000000000000000000000014
Arg [14] : 41627374726163742078207a6b4d61726b657473000000000000000000000000
Arg [15] : 0000000000000000000000000000000000000000000000000000000000000006
Arg [16] : 6162737a6b6d0000000000000000000000000000000000000000000000000000
Arg [17] : 0000000000000000000000000000000000000000000000000000000000000035
Arg [18] : 697066733a2f2f516d62786f5731595662646e776845686a394a3576644d4a4e
Arg [19] : 37686179536e44514579695445696350794e445a770000000000000000000000
Arg [20] : 0000000000000000000000000000000000000000000000000000000000000000
Arg [21] : 0000000000000000000000000000000000000000000000000000000000000035
Arg [22] : 697066733a2f2f516d564e70426d6f353253317432786b4d5756754571516638
Arg [23] : 37415452434142484d623861787479703954746b430000000000000000000000
[ Download: CSV Export ]
[ Download: CSV Export ]
A token is a representation of an on-chain or off-chain asset. The token page shows information such as price, total supply, holders, transfers and social links. Learn more about this page in our Knowledge Base.