// SPDX-License-Identifier: MITpragmasolidity ^0.8.0;/**
* @dev Collection of functions related to the address type
*/libraryAddress{
/**
* @dev Returns true if `account` is a contract.
*
* [IMPORTANT]
* ====
* It is unsafe to assume that an address for which this function returns
* false is an externally-owned account (EOA) and not a contract.
*
* Among others, `isContract` will return false for the following
* types of addresses:
*
* - an externally-owned account
* - a contract in construction
* - an address where a contract will be created
* - an address where a contract lived, but was destroyed
* ====
*/functionisContract(address account) internalviewreturns (bool) {
// This method relies on extcodesize, which returns 0 for contracts in// construction, since the code is only stored at the end of the// constructor execution.uint256 size;
assembly {
size :=extcodesize(account)
}
return size >0;
}
/**
* @dev Replacement for Solidity's `transfer`: sends `amount` wei to
* `recipient`, forwarding all available gas and reverting on errors.
*
* https://eips.ethereum.org/EIPS/eip-1884[EIP1884] increases the gas cost
* of certain opcodes, possibly making contracts go over the 2300 gas limit
* imposed by `transfer`, making them unable to receive funds via
* `transfer`. {sendValue} removes this limitation.
*
* https://diligence.consensys.net/posts/2019/09/stop-using-soliditys-transfer-now/[Learn more].
*
* IMPORTANT: because control is transferred to `recipient`, care must be
* taken to not create reentrancy vulnerabilities. Consider using
* {ReentrancyGuard} or the
* https://solidity.readthedocs.io/en/v0.5.11/security-considerations.html#use-the-checks-effects-interactions-pattern[checks-effects-interactions pattern].
*/functionsendValue(addresspayable recipient, uint256 amount) internal{
require(address(this).balance>= amount, "Address: insufficient balance");
(bool success, ) = recipient.call{value: amount}("");
require(success, "Address: unable to send value, recipient may have reverted");
}
/**
* @dev Performs a Solidity function call using a low level `call`. A
* plain `call` is an unsafe replacement for a function call: use this
* function instead.
*
* If `target` reverts with a revert reason, it is bubbled up by this
* function (like regular Solidity function calls).
*
* Returns the raw returned data. To convert to the expected return value,
* use https://solidity.readthedocs.io/en/latest/units-and-global-variables.html?highlight=abi.decode#abi-encoding-and-decoding-functions[`abi.decode`].
*
* Requirements:
*
* - `target` must be a contract.
* - calling `target` with `data` must not revert.
*
* _Available since v3.1._
*/functionfunctionCall(address target, bytesmemory data) internalreturns (bytesmemory) {
return functionCall(target, data, "Address: low-level call failed");
}
/**
* @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`], but with
* `errorMessage` as a fallback revert reason when `target` reverts.
*
* _Available since v3.1._
*/functionfunctionCall(address target,
bytesmemory data,
stringmemory errorMessage
) internalreturns (bytesmemory) {
return functionCallWithValue(target, data, 0, errorMessage);
}
/**
* @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
* but also transferring `value` wei to `target`.
*
* Requirements:
*
* - the calling contract must have an ETH balance of at least `value`.
* - the called Solidity function must be `payable`.
*
* _Available since v3.1._
*/functionfunctionCallWithValue(address target,
bytesmemory data,
uint256 value
) internalreturns (bytesmemory) {
return functionCallWithValue(target, data, value, "Address: low-level call with value failed");
}
/**
* @dev Same as {xref-Address-functionCallWithValue-address-bytes-uint256-}[`functionCallWithValue`], but
* with `errorMessage` as a fallback revert reason when `target` reverts.
*
* _Available since v3.1._
*/functionfunctionCallWithValue(address target,
bytesmemory data,
uint256 value,
stringmemory errorMessage
) internalreturns (bytesmemory) {
require(address(this).balance>= value, "Address: insufficient balance for call");
require(isContract(target), "Address: call to non-contract");
(bool success, bytesmemory returndata) = target.call{value: value}(data);
return verifyCallResult(success, returndata, errorMessage);
}
/**
* @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
* but performing a static call.
*
* _Available since v3.3._
*/functionfunctionStaticCall(address target, bytesmemory data) internalviewreturns (bytesmemory) {
return functionStaticCall(target, data, "Address: low-level static call failed");
}
/**
* @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`],
* but performing a static call.
*
* _Available since v3.3._
*/functionfunctionStaticCall(address target,
bytesmemory data,
stringmemory errorMessage
) internalviewreturns (bytesmemory) {
require(isContract(target), "Address: static call to non-contract");
(bool success, bytesmemory returndata) = target.staticcall(data);
return verifyCallResult(success, returndata, errorMessage);
}
/**
* @dev Same as {xref-Address-functionCall-address-bytes-}[`functionCall`],
* but performing a delegate call.
*
* _Available since v3.4._
*/functionfunctionDelegateCall(address target, bytesmemory data) internalreturns (bytesmemory) {
return functionDelegateCall(target, data, "Address: low-level delegate call failed");
}
/**
* @dev Same as {xref-Address-functionCall-address-bytes-string-}[`functionCall`],
* but performing a delegate call.
*
* _Available since v3.4._
*/functionfunctionDelegateCall(address target,
bytesmemory data,
stringmemory errorMessage
) internalreturns (bytesmemory) {
require(isContract(target), "Address: delegate call to non-contract");
(bool success, bytesmemory returndata) = target.delegatecall(data);
return verifyCallResult(success, returndata, errorMessage);
}
/**
* @dev Tool to verifies that a low level call was successful, and revert if it wasn't, either by bubbling the
* revert reason using the provided one.
*
* _Available since v4.3._
*/functionverifyCallResult(bool success,
bytesmemory returndata,
stringmemory errorMessage
) internalpurereturns (bytesmemory) {
if (success) {
return returndata;
} else {
// Look for revert reason and bubble it up if presentif (returndata.length>0) {
// The easiest way to bubble the revert reason is using memory via assemblyassembly {
let returndata_size :=mload(returndata)
revert(add(32, returndata), returndata_size)
}
} else {
revert(errorMessage);
}
}
}
}
Contract Source Code
File 2 of 13: Context.sol
// SPDX-License-Identifier: MITpragmasolidity ^0.8.0;/**
* @dev Provides information about the current execution context, including the
* sender of the transaction and its data. While these are generally available
* via msg.sender and msg.data, they should not be accessed in such a direct
* manner, since when dealing with meta-transactions the account sending and
* paying for execution may not be the actual sender (as far as an application
* is concerned).
*
* This contract is only required for intermediate, library-like contracts.
*/abstractcontractContext{
function_msgSender() internalviewvirtualreturns (address) {
returnmsg.sender;
}
function_msgData() internalviewvirtualreturns (bytescalldata) {
returnmsg.data;
}
}
Contract Source Code
File 3 of 13: ERC165.sol
// SPDX-License-Identifier: MITpragmasolidity ^0.8.0;import"./IERC165.sol";
/**
* @dev Implementation of the {IERC165} interface.
*
* Contracts that want to implement ERC165 should inherit from this contract and override {supportsInterface} to check
* for the additional interface id that will be supported. For example:
*
* ```solidity
* function supportsInterface(bytes4 interfaceId) public view virtual override returns (bool) {
* return interfaceId == type(MyInterface).interfaceId || super.supportsInterface(interfaceId);
* }
* ```
*
* Alternatively, {ERC165Storage} provides an easier to use but more expensive implementation.
*/abstractcontractERC165isIERC165{
/**
* @dev See {IERC165-supportsInterface}.
*/functionsupportsInterface(bytes4 interfaceId) publicviewvirtualoverridereturns (bool) {
return interfaceId ==type(IERC165).interfaceId;
}
}
Contract Source Code
File 4 of 13: ERC721.sol
/*
* This file is part of the contracts written for artèQ Investment Fund (https://github.com/billionbuild/arteq-contracts).
* Copyright (c) 2021 BillionBuild (2B) Team.
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, version 3.
*
* This program is distributed in the hope that it will be useful, but
* WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
*/// SPDX-License-Identifier: GNU General Public License v3.0pragmasolidity 0.8.0;import"@openzeppelin/contracts/interfaces/IERC721.sol";
import"@openzeppelin/contracts/interfaces/IERC721Receiver.sol";
import"@openzeppelin/contracts/token/ERC721/extensions/IERC721Metadata.sol";
import"@openzeppelin/contracts/utils/Address.sol";
import"@openzeppelin/contracts/utils/Context.sol";
import"@openzeppelin/contracts/utils/Strings.sol";
import"@openzeppelin/contracts/utils/introspection/ERC165.sol";
/**
* @author Modified by Kam Amini <kam@arteq.io> <kam@2b.team> <kam.cpp@gmail.com>
*
* @notice Use at your own risk
*
* Note: 2B has modified the original code to cover its needs as
* part of artèQ Investment Fund ecosystem
*
* @dev Implementation of https://eips.ethereum.org/EIPS/eip-721[ERC721] Non-Fungible Token Standard, including
* the Metadata extension, but not including the Enumerable extension, which is available separately as
* {ERC721Enumerable}.
*/contractERC721isContext, ERC165, IERC721, IERC721Metadata{
usingAddressforaddress;
usingStringsforuint256;
// Token namestringprivate _name;
// Token symbolstringprivate _symbol;
// Mapping from token ID to owner addressmapping(uint256=>address) private _owners;
// Mapping owner address to token countmapping(address=>uint256) private _balances;
// Mapping from token ID to approved addressmapping(uint256=>address) private _tokenApprovals;
// Mapping from owner to operator approvalsmapping(address=>mapping(address=>bool)) private _operatorApprovals;
/**
* @dev Initializes the contract by setting a `name` and a `symbol` to the token collection.
*/constructor(stringmemory name_, stringmemory symbol_) {
_name = name_;
_symbol = symbol_;
}
/**
* @dev See {IERC165-supportsInterface}.
*/functionsupportsInterface(bytes4 interfaceId) publicviewvirtualoverride(ERC165, IERC165) returns (bool) {
return
interfaceId ==type(IERC721).interfaceId||
interfaceId ==type(IERC721Metadata).interfaceId||super.supportsInterface(interfaceId);
}
/**
* @dev See {IERC721-balanceOf}.
*/functionbalanceOf(address owner) publicviewvirtualoverridereturns (uint256) {
require(owner !=address(0), "ERC721: balance query for the zero address");
return _balances[owner];
}
/**
* @dev See {IERC721-ownerOf}.
*/functionownerOf(uint256 tokenId) publicviewvirtualoverridereturns (address) {
address owner = _owners[tokenId];
require(owner !=address(0), "ERC721: owner query for nonexistent token");
return owner;
}
/**
* @dev See {IERC721Metadata-name}.
*/functionname() publicviewvirtualoverridereturns (stringmemory) {
return _name;
}
/**
* @dev See {IERC721Metadata-symbol}.
*/functionsymbol() publicviewvirtualoverridereturns (stringmemory) {
return _symbol;
}
/**
* @dev See {IERC721Metadata-tokenURI}.
*/functiontokenURI(uint256 tokenId) publicviewvirtualoverridereturns (stringmemory) {
require(_exists(tokenId), "ERC721Metadata: URI query for nonexistent token");
stringmemory baseURI = _baseURI();
returnbytes(baseURI).length>0 ? string(abi.encodePacked(baseURI, tokenId.toString())) : "";
}
/**
* @dev Base URI for computing {tokenURI}. If set, the resulting URI for each
* token will be the concatenation of the `baseURI` and the `tokenId`. Empty
* by default, can be overriden in child contracts.
*/function_baseURI() internalviewvirtualreturns (stringmemory) {
return"";
}
/**
* @dev See {IERC721-approve}.
*/functionapprove(address to, uint256 tokenId) publicvirtualoverride{
address owner = ERC721.ownerOf(tokenId);
require(to != owner, "ERC721: approval to current owner");
require(
_msgSender() == owner || isApprovedForAll(owner, _msgSender()),
"ERC721: approve caller is not owner nor approved for all"
);
_approve(to, tokenId);
}
/**
* @dev See {IERC721-getApproved}.
*/functiongetApproved(uint256 tokenId) publicviewvirtualoverridereturns (address) {
require(_exists(tokenId), "ERC721: approved query for nonexistent token");
return _tokenApprovals[tokenId];
}
/**
* @dev See {IERC721-setApprovalForAll}.
*/functionsetApprovalForAll(address operator, bool approved) publicvirtualoverride{
require(operator != _msgSender(), "ERC721: approve to caller");
_operatorApprovals[_msgSender()][operator] = approved;
emit ApprovalForAll(_msgSender(), operator, approved);
}
/**
* @dev See {IERC721-isApprovedForAll}.
*/functionisApprovedForAll(address owner, address operator) publicviewvirtualoverridereturns (bool) {
return _operatorApprovals[owner][operator];
}
/**
* @dev See {IERC721-transferFrom}.
*/functiontransferFrom(addressfrom,
address to,
uint256 tokenId
) publicvirtualoverride{
//solhint-disable-next-line max-line-lengthrequire(_isApprovedOrOwner(_msgSender(), tokenId), "ERC721: transfer caller is not owner nor approved");
_transfer(from, to, tokenId);
}
/**
* @dev See {IERC721-safeTransferFrom}.
*/functionsafeTransferFrom(addressfrom,
address to,
uint256 tokenId
) publicvirtualoverride{
safeTransferFrom(from, to, tokenId, "");
}
/**
* @dev See {IERC721-safeTransferFrom}.
*/functionsafeTransferFrom(addressfrom,
address to,
uint256 tokenId,
bytesmemory _data
) publicvirtualoverride{
require(_isApprovedOrOwner(_msgSender(), tokenId), "ERC721: transfer caller is not owner nor approved");
_safeTransfer(from, to, tokenId, _data);
}
/**
* @dev Safely transfers `tokenId` token from `from` to `to`, checking first that contract recipients
* are aware of the ERC721 protocol to prevent tokens from being forever locked.
*
* `_data` is additional data, it has no specified format and it is sent in call to `to`.
*
* This internal function is equivalent to {safeTransferFrom}, and can be used to e.g.
* implement alternative mechanisms to perform token transfer, such as signature-based.
*
* Requirements:
*
* - `from` cannot be the zero address.
* - `to` cannot be the zero address.
* - `tokenId` token must exist and be owned by `from`.
* - If `to` refers to a smart contract, it must implement {IERC721Receiver-onERC721Received}, which is called upon a safe transfer.
*
* Emits a {Transfer} event.
*/function_safeTransfer(addressfrom,
address to,
uint256 tokenId,
bytesmemory _data
) internalvirtual{
_transfer(from, to, tokenId);
require(_checkOnERC721Received(from, to, tokenId, _data), "ERC721: transfer to non ERC721Receiver implementer");
}
/**
* @dev Returns whether `tokenId` exists.
*
* Tokens can be managed by their owner or approved accounts via {approve} or {setApprovalForAll}.
*
* Tokens start existing when they are minted (`_mint`),
* and stop existing when they are burned (`_burn`).
*/function_exists(uint256 tokenId) internalviewvirtualreturns (bool) {
return _owners[tokenId] !=address(0);
}
/**
* @dev Returns whether `spender` is allowed to manage `tokenId`.
*
* Requirements:
*
* - `tokenId` must exist.
*/function_isApprovedOrOwner(address spender, uint256 tokenId) internalviewvirtualreturns (bool) {
require(_exists(tokenId), "ERC721: operator query for nonexistent token");
address owner = ERC721.ownerOf(tokenId);
return (spender == owner || getApproved(tokenId) == spender || isApprovedForAll(owner, spender));
}
/**
* @dev Safely mints `tokenId` and transfers it to `to`.
*
* Requirements:
*
* - `tokenId` must not exist.
* - If `to` refers to a smart contract, it must implement {IERC721Receiver-onERC721Received}, which is called upon a safe transfer.
*
* Emits a {Transfer} event.
*/function_safeMint(address to, uint256 tokenId) internalvirtual{
_safeMint(to, tokenId, "");
}
/**
* @dev Same as {xref-ERC721-_safeMint-address-uint256-}[`_safeMint`], with an additional `data` parameter which is
* forwarded in {IERC721Receiver-onERC721Received} to contract recipients.
*/function_safeMint(address to,
uint256 tokenId,
bytesmemory _data
) internalvirtual{
_mint(address(0), to, tokenId);
require(
_checkOnERC721Received(address(0), to, tokenId, _data),
"ERC721: transfer to non ERC721Receiver implementer"
);
}
/**
* @dev Mints `tokenId` and transfers it to `to`.
*
* WARNING: Usage of this method is discouraged, use {_safeMint} whenever possible
*
* Requirements:
*
* - `tokenId` must not exist.
* - `to` cannot be the zero address.
*
* Emits a {Transfer} event.
*/function_mint(addressfrom, address to, uint256 tokenId) internalvirtual{
require(to !=address(0), "ERC721: mint to the zero address");
require(!_exists(tokenId), "ERC721: token already minted");
_beforeTokenTransfer(from, to, tokenId);
_balances[to] +=1;
_owners[tokenId] = to;
emit Transfer(from, to, tokenId);
}
/**
* @dev Destroys `tokenId`.
* The approval is cleared when the token is burned.
*
* Requirements:
*
* - `tokenId` must exist.
*
* Emits a {Transfer} event.
*/function_burn(uint256 tokenId) internalvirtual{
address owner = ERC721.ownerOf(tokenId);
_beforeTokenTransfer(owner, address(0), tokenId);
// Clear approvals
_approve(address(0), tokenId);
_balances[owner] -=1;
delete _owners[tokenId];
emit Transfer(owner, address(0), tokenId);
}
/**
* @dev Transfers `tokenId` from `from` to `to`.
* As opposed to {transferFrom}, this imposes no restrictions on msg.sender.
*
* Requirements:
*
* - `to` cannot be the zero address.
* - `tokenId` token must be owned by `from`.
*
* Emits a {Transfer} event.
*/function_transfer(addressfrom,
address to,
uint256 tokenId
) internalvirtual{
require(ERC721.ownerOf(tokenId) ==from, "ERC721: transfer of token that is not own");
require(to !=address(0), "ERC721: transfer to the zero address");
_beforeTokenTransfer(from, to, tokenId);
// Clear approvals from the previous owner
_approve(address(0), tokenId);
_balances[from] -=1;
_balances[to] +=1;
_owners[tokenId] = to;
emit Transfer(from, to, tokenId);
}
/**
* @dev Approve `to` to operate on `tokenId`
*
* Emits a {Approval} event.
*/function_approve(address to, uint256 tokenId) internalvirtual{
_tokenApprovals[tokenId] = to;
emit Approval(ERC721.ownerOf(tokenId), to, tokenId);
}
/**
* @dev Internal function to invoke {IERC721Receiver-onERC721Received} on a target address.
* The call is not executed if the target address is not a contract.
*
* @param from address representing the previous owner of the given token ID
* @param to target address that will receive the tokens
* @param tokenId uint256 ID of the token to be transferred
* @param _data bytes optional data to send along with the call
* @return bool whether the call correctly returned the expected magic value
*/function_checkOnERC721Received(addressfrom,
address to,
uint256 tokenId,
bytesmemory _data
) privatereturns (bool) {
if (to.isContract()) {
try IERC721Receiver(to).onERC721Received(_msgSender(), from, tokenId, _data) returns (bytes4 retval) {
return retval == IERC721Receiver.onERC721Received.selector;
} catch (bytesmemory reason) {
if (reason.length==0) {
revert("ERC721: transfer to non ERC721Receiver implementer");
} else {
assembly {
revert(add(32, reason), mload(reason))
}
}
}
} else {
returntrue;
}
}
/**
* @dev Hook that is called before any token transfer. This includes minting
* and burning.
*
* Calling conditions:
*
* - When `from` and `to` are both non-zero, ``from``'s `tokenId` will be
* transferred to `to`.
* - When `from` is zero, `tokenId` will be minted for `to`.
* - When `to` is zero, ``from``'s `tokenId` will be burned.
* - `from` and `to` are never both zero.
*
* To learn more about hooks, head to xref:ROOT:extending-contracts.adoc#using-hooks[Using Hooks].
*/function_beforeTokenTransfer(addressfrom,
address to,
uint256 tokenId
) internalvirtual{}
}
Contract Source Code
File 5 of 13: ERC721URIStorage.sol
/*
* This file is part of the contracts written for artèQ Investment Fund (https://github.com/billionbuild/arteq-contracts).
* Copyright (c) 2021 BillionBuild (2B) Team.
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, version 3.
*
* This program is distributed in the hope that it will be useful, but
* WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
*/// SPDX-License-Identifier: GNU General Public License v3.0// Based on OpenZeppelin Contracts v4.3.2 (token/ERC721/extensions/ERC721URIStorage.sol)pragmasolidity 0.8.0;import"./ERC721.sol";
/**
* @author Modified by Kam Amini <kam@arteq.io> <kam@2b.team> <kam.cpp@gmail.com>
*
* @notice Use at your own risk
*
* Note: 2B has modified the original code to cover its needs as
* part of artèQ Investment Fund ecosystem
*
* @dev ERC721 token with storage based token URI management.
*/abstractcontractERC721URIStorageisERC721{
usingStringsforuint256;
// Optional mapping for token URIsmapping(uint256=>string) private _tokenURIs;
/**
* @dev See {IERC721Metadata-tokenURI}.
*/functiontokenURI(uint256 tokenId) publicviewvirtualoverridereturns (stringmemory) {
require(_exists(tokenId), "ERC721URIStorage: URI query for nonexistent token");
stringmemory _tokenURI = _tokenURIs[tokenId];
stringmemory base = _baseURI();
// If there is no base URI, return the token URI.if (bytes(base).length==0) {
return _tokenURI;
}
// If both are set, concatenate the baseURI and tokenURI (via abi.encodePacked).if (bytes(_tokenURI).length>0) {
returnstring(abi.encodePacked(base, _tokenURI));
}
returnsuper.tokenURI(tokenId);
}
/**
* @dev Sets `_tokenURI` as the tokenURI of `tokenId`.
*
* Requirements:
*
* - `tokenId` must exist.
*/function_setTokenURI(uint256 tokenId, stringmemory _tokenURI) internalvirtual{
require(_exists(tokenId), "ERC721URIStorage: URI set of nonexistent token");
_tokenURIs[tokenId] = _tokenURI;
}
/**
* @dev Destroys `tokenId`.
* The approval is cleared when the token is burned.
*
* Requirements:
*
* - `tokenId` must exist.
*
* Emits a {Transfer} event.
*/function_burn(uint256 tokenId) internalvirtualoverride{
super._burn(tokenId);
if (bytes(_tokenURIs[tokenId]).length!=0) {
delete _tokenURIs[tokenId];
}
}
}
// SPDX-License-Identifier: MITpragmasolidity ^0.8.0;import"./IERC165.sol";
/**
* @dev Interface for the NFT Royalty Standard
*/interfaceIERC2981isIERC165{
/**
* @dev Called with the sale price to determine how much royalty is owed and to whom.
* @param tokenId - the NFT asset queried for royalty information
* @param salePrice - the sale price of the NFT asset specified by `tokenId`
* @return receiver - address of who should be sent the royalty payment
* @return royaltyAmount - the royalty payment amount for `salePrice`
*/functionroyaltyInfo(uint256 tokenId, uint256 salePrice)
externalviewreturns (address receiver, uint256 royaltyAmount);
}
/*
* This file is part of the contracts written for artèQ Investment Fund (https://github.com/billionbuild/arteq-contracts).
* Copyright (c) 2021 BillionBuild (2B) Team.
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, version 3.
*
* This program is distributed in the hope that it will be useful, but
* WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
*/// SPDX-License-Identifier: GNU General Public License v3.0pragmasolidity 0.8.0;/// @author Kam Amini <kam@arteq.io> <kam@2b.team> <kam.cpp@gmail.com>/// @title The interface for finalizing tasks. Mainly used by artèQ contracts to/// perform administrative tasks in conjuction with admin contract.interfaceIarteQTaskFinalizer{
eventTaskFinalized(address finalizer, address origin, uint256 taskId);
functionfinalizeTask(address origin, uint256 taskId) external;
}
Contract Source Code
File 12 of 13: Strings.sol
// SPDX-License-Identifier: MITpragmasolidity ^0.8.0;/**
* @dev String operations.
*/libraryStrings{
bytes16privateconstant _HEX_SYMBOLS ="0123456789abcdef";
/**
* @dev Converts a `uint256` to its ASCII `string` decimal representation.
*/functiontoString(uint256 value) internalpurereturns (stringmemory) {
// Inspired by OraclizeAPI's implementation - MIT licence// https://github.com/oraclize/ethereum-api/blob/b42146b063c7d6ee1358846c198246239e9360e8/oraclizeAPI_0.4.25.solif (value ==0) {
return"0";
}
uint256 temp = value;
uint256 digits;
while (temp !=0) {
digits++;
temp /=10;
}
bytesmemory buffer =newbytes(digits);
while (value !=0) {
digits -=1;
buffer[digits] =bytes1(uint8(48+uint256(value %10)));
value /=10;
}
returnstring(buffer);
}
/**
* @dev Converts a `uint256` to its ASCII `string` hexadecimal representation.
*/functiontoHexString(uint256 value) internalpurereturns (stringmemory) {
if (value ==0) {
return"0x00";
}
uint256 temp = value;
uint256 length =0;
while (temp !=0) {
length++;
temp >>=8;
}
return toHexString(value, length);
}
/**
* @dev Converts a `uint256` to its ASCII `string` hexadecimal representation with fixed length.
*/functiontoHexString(uint256 value, uint256 length) internalpurereturns (stringmemory) {
bytesmemory buffer =newbytes(2* length +2);
buffer[0] ="0";
buffer[1] ="x";
for (uint256 i =2* length +1; i >1; --i) {
buffer[i] = _HEX_SYMBOLS[value &0xf];
value >>=4;
}
require(value ==0, "Strings: hex length insufficient");
returnstring(buffer);
}
}
Contract Source Code
File 13 of 13: arteQArtDrop.sol
/*
* This file is part of the contracts written for artèQ Investment Fund (https://arteq.io).
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, version 3.
*
* This program is distributed in the hope that it will be useful, but
* WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
*/// SPDX-License-Identifier: GNU General Public License v3.0pragmasolidity 0.8.0;import"@openzeppelin/contracts/interfaces/IERC165.sol";
import"@openzeppelin/contracts/interfaces/IERC2981.sol";
import"./ERC721URIStorage.sol";
import"./ERC721.sol";
import"./IarteQTaskFinalizer.sol";
/// @author Kam Amini <kam@arteq.io> <kam.cpp@gmail.com>////// @notice Use at your own riskcontractarteQArtDropisERC721URIStorage, IERC2981{
stringprivateconstant DEFAULT_TOKEN_URI ="DEFAULT_TOKEN_URI";
uint256publicconstant MAX_NR_TOKENS_PER_ACCOUNT =5;
uint256publicconstant MAX_RESERVATIONS_COUNT =10000;
int256publicconstant LOCKED_STAGE =0;
int256publicconstant WHITELISTING_STAGE =2;
int256publicconstant RESERVATION_STAGE =3;
int256publicconstant DISTRIBUTION_STAGE =4;
// Counter for token IDsuint256private _tokenIdCounter;
// Counter for pre-minted token IDsuint256private _preMintedTokenIdCounter;
// number of tokens owned by the contractuint256 _contractBalance;
addressprivate _adminContract;
// in weiuint256private _pricePerToken;
// in weiuint256private _serviceFee;
stringprivate _defaultTokenURI;
address _royaltyWallet;
uint256 _royaltyPercentage;
// The current art drop stage. It can have the following values://// 0: Locked / Read-only mode// 2: Selection of the registered wallets (whitelisting)// 3: Reservation / Purchase stage// 4: Distribution of the tokens / Drop stage//// * 1 is missing from the above list. That's to keep the off-chain// and on-chain states in sync.// * Only admin accounts with a quorum of votes can change the// current stage.// * Some functions only work in certain stages.// * When the 4th stage (last stage) is finished, the contract// will be put back into locked mode (stage 0).// * Admins can only advance/retreat the current stage by movements of +1 or -1.int256private _stage;
// A mapping from the whitelisted addresses to the maximum number of tokens they can obtainmapping(address=>uint256) _whitelistedAccounts;
// Counts the number of whitelisted accountsuint256 _whitelistedAccountsCounter;
// Counts the number of reserved tokensuint256 _reservedTokensCounter;
// Enabled reservations without a need to be whitelistedbool _canReserveWithoutBeingWhitelisted;
// An operator which is allowed to perform certain operations such as adding whitelisted// accounts, removing them, or doing the token reservation for credit card payments. These// accounts can only be defined by a quorom of votes among admins.mapping(address=>uint256) _operators;
eventWhitelistedAccountAdded(address doer, address account, uint256 maxNrOfTokensToObtain);
eventWhitelistedAccountRemoved(address doer, address account);
eventPricePerTokenChanged(address doer, uint256 adminTaskId, uint256 oldValue, uint256 newValue);
eventServiceFeeChanged(address doer, uint256 adminTaskId, uint256 oldValue, uint256 newValue);
eventStageChanged(address doer, uint256 adminTaskId, int256 oldValue, int256 newValue);
eventOperatorAdded(address doer, uint256 adminTaskId, address toBeOperatorAccount);
eventOperatorRemoved(address doer, uint256 adminTaskId, address toBeRemovedOperatorAccount);
eventDefaultTokenURIChanged(address doer, uint256 adminTaskId, string newValue);
eventTokensReserved(address doer, address target, uint256 nrOfTokensToReserve);
eventDeposited(address doer, uint256 priceOfTokens, uint256 serviceFee, uint256 totalValue);
eventReturned(address doer, address target, uint256 returnedValue);
eventWithdrawn(address doer, address target, uint256 amount);
eventTokenURIChanged(address doer, uint256 tokenId, string newValue);
eventGenesisTokenURIChanged(address doer, uint256 adminTaskId, string newValue);
eventRoyaltyWalletChanged(address doer, uint256 adminTaskId, address newRoyaltyWallet);
eventRoyaltyPercentageChanged(address doer, uint256 adminTaskId, uint256 newRoyaltyPercentage);
eventCanReserveWithoutBeingWhitelistedChanged(address doer, uint256 adminTaskId, bool newValue);
modifieradminApprovalRequired(uint256 adminTaskId) {
_;
// This must succeed otherwise the tx gets reverted
IarteQTaskFinalizer(_adminContract).finalizeTask(msg.sender, adminTaskId);
}
modifieronlyLockedStage() {
require(_stage == LOCKED_STAGE, "arteQArtDrop: only callable in locked stage");
_;
}
modifieronlyWhitelistingStage() {
require(_stage == WHITELISTING_STAGE, "arteQArtDrop: only callable in whitelisting stage");
_;
}
modifieronlyReservationStage() {
require(_stage == RESERVATION_STAGE, "arteQArtDrop: only callable in reservation stage");
_;
}
modifieronlyReservationAndDistributionStages() {
require(_stage == RESERVATION_STAGE || _stage == DISTRIBUTION_STAGE, "arteQArtDrop: only callable in reservation and distribution stage");
_;
}
modifieronlyDistributionStage() {
require(_stage == DISTRIBUTION_STAGE, "arteQArtDrop: only callable in distribution stage");
_;
}
modifieronlyWhenNotLocked() {
require(_stage >1, "arteQArtDrop: only callable in not-locked stages");
_;
}
modifieronlyWhenNotInReservationStage() {
require(_stage != RESERVATION_STAGE, "arteQArtDrop: only callable in a non-reservation stage");
_;
}
modifieronlyOperator() {
require(_operators[msg.sender] >0, "arteQArtDrop: not an operator account");
_;
}
functionsupportsInterface(bytes4 interfaceId) publicviewvirtualoverride(ERC721, IERC165) returns (bool) {
return interfaceId ==type(IERC2981).interfaceId||super.supportsInterface(interfaceId);
}
constructor(address adminContract,
stringmemory name,
stringmemory symbol,
uint256 initialPricePerToken,
uint256 initialServiceFee,
stringmemory initialDefaultTokenURI,
stringmemory initialGenesisTokenURI
) ERC721(name, symbol) {
require(adminContract !=address(0), "arteQArtDrop: admin contract cannot be zero");
require(adminContract.code.length>0, "arteQArtDrop: non-contract account for admin contract");
require(initialPricePerToken >0, "arteQArtDrop: zero initial price per token");
require(bytes(initialDefaultTokenURI).length>0, "arteQArtDrop: invalid default token uri");
require(bytes(initialGenesisTokenURI).length>0, "arteQArtDrop: invalid genesis token uri");
_adminContract = adminContract;
_pricePerToken = initialPricePerToken;
emit PricePerTokenChanged(msg.sender, 0, 0, _pricePerToken);
_serviceFee = initialServiceFee;
emit ServiceFeeChanged(msg.sender, 0, 0, _serviceFee);
_defaultTokenURI = initialDefaultTokenURI;
emit DefaultTokenURIChanged(msg.sender, 0, _defaultTokenURI);
_tokenIdCounter =1;
_preMintedTokenIdCounter =1;
_contractBalance =0;
_whitelistedAccountsCounter =0;
_reservedTokensCounter =0;
// Contract is locked/read-only by default.
_stage =0;
emit StageChanged(msg.sender, 0, 0, _stage);
// Mint genesis token. Contract will be the eternal owner of the genesis token.
_mint(address(0), address(this), 0);
_setTokenURI(0, initialGenesisTokenURI);
_contractBalance +=1;
emit GenesisTokenURIChanged(msg.sender, 0, initialGenesisTokenURI);
_royaltyWallet =address(this);
emit RoyaltyWalletChanged(msg.sender, 0, _royaltyWallet);
_royaltyPercentage =10;
emit RoyaltyPercentageChanged(msg.sender, 0, _royaltyPercentage);
_canReserveWithoutBeingWhitelisted =false;
emit CanReserveWithoutBeingWhitelistedChanged(msg.sender, 0, _canReserveWithoutBeingWhitelisted);
}
functiontokenURI(uint256 tokenId) publicviewvirtualoverridereturns (stringmemory) {
if (_exists(tokenId)) {
stringmemory tokenURIValue =super.tokenURI(tokenId);
if (keccak256(bytes(tokenURIValue)) ==keccak256(bytes(DEFAULT_TOKEN_URI))) {
return _defaultTokenURI;
}
return tokenURIValue;
}
if (tokenId >=1&& tokenId < _preMintedTokenIdCounter) {
return _defaultTokenURI;
}
revert("arteQArtDrop: token id does not exist");
}
functionownerOf(uint256 tokenId) publicviewvirtualoverridereturns (address) {
if (_exists(tokenId)) {
returnsuper.ownerOf(tokenId);
}
if (tokenId >=1&& tokenId < _preMintedTokenIdCounter) {
returnaddress(this);
}
revert("arteQArtDrop: token is does not exist");
}
functionbalanceOf(address owner) publicviewvirtualoverridereturns (uint256) {
if (owner ==address(this)) {
return _contractBalance;
}
returnsuper.balanceOf(owner);
}
functionpreMint(uint256 nr) externalonlyOperator{
for (uint256 i =0; i < nr; i++) {
require(_preMintedTokenIdCounter <= MAX_RESERVATIONS_COUNT, "arteQArtDrop: cannot pre-mint more");
emit Transfer(address(0), address(this), _preMintedTokenIdCounter);
_preMintedTokenIdCounter +=1;
}
_contractBalance += nr;
}
functionpricePerToken() externalviewreturns (uint256) {
return _pricePerToken;
}
functionserviceFee() externalviewreturns (uint256) {
return _serviceFee;
}
functiondefaultTokenURI() externalviewreturns (stringmemory) {
return _defaultTokenURI;
}
functionnrPreMintedTokens() externalviewreturns (uint256) {
return _preMintedTokenIdCounter -1;
}
functionstage() externalviewreturns (int256) {
return _stage;
}
functionroyaltyPercentage() externalviewreturns (uint256) {
return _royaltyPercentage;
}
functionroyaltyWallet() externalviewreturns (address) {
return _royaltyWallet;
}
functionnrOfWhitelistedAccounts() externalviewreturns (uint256) {
return _whitelistedAccountsCounter;
}
functionnrOfReservedTokens() externalviewreturns (uint256) {
return _reservedTokensCounter;
}
functioncanReserveWithoutBeingWhitelisted() externalviewreturns (bool) {
return _canReserveWithoutBeingWhitelisted;
}
functionsetPricePerToken(uint256 adminTaskId, uint256 newValue) externalonlyWhenNotLockedonlyWhenNotInReservationStageadminApprovalRequired(adminTaskId) {
require(newValue >0, "arteQArtDrop: new price cannot be zero");
uint256 oldValue = _pricePerToken;
_pricePerToken = newValue;
emit PricePerTokenChanged(msg.sender, adminTaskId, oldValue, _pricePerToken);
}
functionsetServiceFee(uint256 adminTaskId, uint256 newValue) externalonlyWhenNotLockedonlyWhenNotInReservationStageadminApprovalRequired(adminTaskId) {
require(newValue >0, "arteQArtDrop: new price cannot be zero");
uint256 oldValue = _serviceFee;
_serviceFee = newValue;
emit ServiceFeeChanged(msg.sender, adminTaskId, oldValue, _serviceFee);
}
functionsetDefaultTokenURI(uint256 adminTaskId, stringmemory newValue) externalonlyWhenNotLockedonlyWhenNotInReservationStageadminApprovalRequired(adminTaskId) {
require(bytes(newValue).length>0, "arteQArtDrop: empty string");
_defaultTokenURI = newValue;
emit DefaultTokenURIChanged(msg.sender, adminTaskId, _defaultTokenURI);
}
functionsetGenesisTokenURI(uint256 adminTaskId, stringmemory newValue) externalonlyLockedStageadminApprovalRequired(adminTaskId) {
require(bytes(newValue).length>0, "arteQArtDrop: empty string");
_setTokenURI(0, newValue);
emit GenesisTokenURIChanged(msg.sender, adminTaskId, newValue);
}
functionsetRoyaltyWallet(uint256 adminTaskId, address newRoyaltyWallet) externaladminApprovalRequired(adminTaskId) {
require(newRoyaltyWallet !=address(0), "arteQArtDrop: invalid royalty wallet");
_royaltyWallet = newRoyaltyWallet;
emit RoyaltyWalletChanged(msg.sender, adminTaskId, newRoyaltyWallet);
}
functionsetRoyaltyPercentage(uint256 adminTaskId, uint256 newRoyaltyPercentage) externaladminApprovalRequired(adminTaskId) {
require(newRoyaltyPercentage >=0&& newRoyaltyPercentage <=75, "arteQArtDrop: invalid royalty percentage");
_royaltyPercentage = newRoyaltyPercentage;
emit RoyaltyPercentageChanged(msg.sender, adminTaskId, newRoyaltyPercentage);
}
functionsetCanReserveWithoutBeingWhitelisted(uint256 adminTaskId, bool newValue) externaladminApprovalRequired(adminTaskId) {
_canReserveWithoutBeingWhitelisted = newValue;
emit CanReserveWithoutBeingWhitelistedChanged(msg.sender, adminTaskId, newValue);
}
functionretreatStage(uint256 adminTaskId) externaladminApprovalRequired(adminTaskId) {
int256 oldStage = _stage;
_stage -=1;
if (_stage ==-1) {
_stage =4;
} elseif (_stage ==1) {
_stage =0;
}
emit StageChanged(msg.sender, adminTaskId, oldStage, _stage);
}
functionadvanceStage(uint256 adminTaskId) externaladminApprovalRequired(adminTaskId) {
int256 oldStage = _stage;
_stage +=1;
if (_stage ==5) {
_stage =0;
} elseif (_stage ==1) {
_stage =2;
}
emit StageChanged(msg.sender, adminTaskId, oldStage, _stage);
}
functionaddOperator(uint256 adminTaskId, address toBeOperatorAccount) externaladminApprovalRequired(adminTaskId) {
require(toBeOperatorAccount !=address(0), "arteQArtDrop: cannot set zero as operator");
require(_operators[toBeOperatorAccount] ==0, "arteQArtDrop: already an operator");
_operators[toBeOperatorAccount] =1;
emit OperatorAdded(msg.sender, adminTaskId, toBeOperatorAccount);
}
functionremoveOperator(uint256 adminTaskId, address toBeRemovedOperatorAccount) externaladminApprovalRequired(adminTaskId) {
require(toBeRemovedOperatorAccount !=address(0), "arteQArtDrop: cannot remove zero as operator");
require(_operators[toBeRemovedOperatorAccount] ==1, "arteQArtDrop: not an operator");
_operators[toBeRemovedOperatorAccount] =0;
emit OperatorRemoved(msg.sender, adminTaskId, toBeRemovedOperatorAccount);
}
functionisOperator(address account) externalviewreturns(bool) {
return _operators[account] ==1;
}
functionaddToWhitelistedAccounts(address[] memory accounts,
uint[] memory listOfMaxNrOfTokensToObtain
) externalonlyOperatoronlyWhitelistingStage{
require(accounts.length>0, "arteQArtDrop: zero length");
require(listOfMaxNrOfTokensToObtain.length>0, "arteQArtDrop: zero length");
require(accounts.length== listOfMaxNrOfTokensToObtain.length, "arteQArtDrop: different lengths");
for (uint256 i =0; i < accounts.length; i++) {
address account = accounts[i];
uint256 maxNrOfTokensToObtain = listOfMaxNrOfTokensToObtain[i];
require(account !=address(0), "arteQArtDrop: cannot whitelist zero address");
require(maxNrOfTokensToObtain >=1&& maxNrOfTokensToObtain <= MAX_NR_TOKENS_PER_ACCOUNT,
"arteQArtDrop: invalid nr of tokens to obtain");
require(account.code.length==0, "arteQArtDrop: cannot whitelist a contract");
require(_whitelistedAccounts[account] ==0, "arteQArtDrop: already whitelisted");
_whitelistedAccounts[account] = maxNrOfTokensToObtain;
_whitelistedAccountsCounter +=1;
emit WhitelistedAccountAdded(msg.sender, account, maxNrOfTokensToObtain);
}
}
functionremoveFromWhitelistedAccounts(address[] memory accounts) externalonlyOperatoronlyWhitelistingStage{
require(accounts.length>0, "arteQArtDrop: zero length");
for (uint256 i =0; i < accounts.length; i++) {
address account = accounts[i];
require(account !=address(0), "arteQArtDrop: cannot remove zero address");
require(_whitelistedAccounts[account] >0, "arteQArtDrop: account is not whitelisted");
_whitelistedAccounts[account] =0;
_whitelistedAccountsCounter -=1;
emit WhitelistedAccountRemoved(msg.sender, account);
}
}
functionwhitelistedNrOfTokens(address account) externalviewreturns (uint256) {
if (!_canReserveWithoutBeingWhitelisted) {
return _whitelistedAccounts[account];
}
if (_whitelistedAccounts[account] ==0) {
return MAX_NR_TOKENS_PER_ACCOUNT;
}
return _whitelistedAccounts[account];
}
// Only callable by a whitelisted account//// * Account must have sent enough ETH to cover the price of all tokens + service fee// * Account cannot reserve more than what has been whitelisted forfunctionreserveTokens(uint256 nrOfTokensToReserve) externalpayableonlyReservationAndDistributionStages{
require(msg.value>0, "arteQArtDrop: zero funds");
require(nrOfTokensToReserve >0, "arteQArtDrop: zero tokens to reserve");
if (_canReserveWithoutBeingWhitelisted && _whitelistedAccounts[msg.sender] ==0) {
_whitelistedAccounts[msg.sender] =5;
}
require(_whitelistedAccounts[msg.sender] >0, "arteQArtDrop: not a whitelisted account");
require(nrOfTokensToReserve <= _whitelistedAccounts[msg.sender],
"arteQArtDrop: exceeding the reservation allowance");
require((_reservedTokensCounter + nrOfTokensToReserve) <= MAX_RESERVATIONS_COUNT,
"arteQArtDrop: exceeding max number of reservations");
// Handle paymentsuint256 priceOfTokens = nrOfTokensToReserve * _pricePerToken;
uint256 priceToPay = priceOfTokens + _serviceFee;
require(msg.value>= priceToPay, "arteQArtDrop: insufficient funds");
uint256 remainder =msg.value- priceToPay;
if (remainder >0) {
(bool success, ) =msg.sender.call{value: remainder}(newbytes(0));
require(success, "arteQArtDrop: failed to send the remainder");
emit Returned(msg.sender, msg.sender, remainder);
}
emit Deposited(msg.sender, priceOfTokens, _serviceFee, priceToPay);
_reserveTokens(msg.sender, nrOfTokensToReserve);
}
// This method is called by an operator to complete the reservation of fiat payments// such as credit card, iDeal, etc.functionreserveTokensForAccounts(address[] memory accounts,
uint256[] memory listOfNrOfTokensToReserve
) externalonlyOperatoronlyReservationAndDistributionStages{
require(accounts.length>0, "arteQArtDrop: zero length");
require(listOfNrOfTokensToReserve.length>0, "arteQArtDrop: zero length");
require(accounts.length== listOfNrOfTokensToReserve.length, "arteQArtDrop: different lengths");
for (uint256 i =0; i < accounts.length; i++) {
address account = accounts[i];
uint256 nrOfTokensToReserve = listOfNrOfTokensToReserve[i];
require(account !=address(0), "arteQArtDrop: cannot be zero address");
if (_canReserveWithoutBeingWhitelisted && _whitelistedAccounts[account] ==0) {
_whitelistedAccounts[account] =5;
}
require(_whitelistedAccounts[account] >0, "arteQArtDrop: not a whitelisted account");
require(nrOfTokensToReserve <= _whitelistedAccounts[account],
"arteQArtDrop: exceeding the reservation allowance");
_reserveTokens(account, nrOfTokensToReserve);
}
}
functionupdateTokenURIs(uint256[] memory tokenIds, string[] memory newTokenURIs) externalonlyOperatoronlyDistributionStage{
require(tokenIds.length>0, "arteQArtDrop: zero length");
require(newTokenURIs.length>0, "arteQArtDrop: zero length");
require(tokenIds.length== newTokenURIs.length, "arteQArtDrop: different lengths");
for (uint256 i =0; i < tokenIds.length; i++) {
uint256 tokenId = tokenIds[i];
stringmemory newTokenURI = newTokenURIs[i];
require(tokenId >0, "arteQArtDrop: cannot alter genesis token");
require(bytes(newTokenURI).length>0, "arteQArtDrop: empty string");
_setTokenURI(tokenId, newTokenURI);
emit TokenURIChanged(msg.sender, tokenId, newTokenURI);
}
}
functiontransferTo(address target, uint256 amount) externalonlyOperator{
require(target !=address(0), "arteQArtDrop: target cannot be zero");
require(amount >0, "arteQArtDrop: cannot transfer zero");
require(amount <=address(this).balance, "arteQArtDrop: transfer more than balance");
(bool success, ) = target.call{value: amount}(newbytes(0));
require(success, "arteQArtDrop: failed to transfer");
emit Withdrawn(msg.sender, target, amount);
}
functionroyaltyInfo(uint256, uint256 salePrice) externalviewvirtualoverridereturns (address, uint256) {
uint256 royalty = (salePrice * _royaltyPercentage) /100;
return (_royaltyWallet, royalty);
}
function_reserveTokens(address target, uint256 nrOfTokensToReserve) internal{
for (uint256 i =1; i <= nrOfTokensToReserve; i++) {
uint256 newTokenId = _tokenIdCounter;
_mint(address(this), target, newTokenId);
_setTokenURI(newTokenId, DEFAULT_TOKEN_URI);
_tokenIdCounter +=1;
require(_reservedTokensCounter <= MAX_RESERVATIONS_COUNT,
"arteQArtDrop: exceeding max number of reservations");
_reservedTokensCounter +=1;
}
if ((_contractBalance -1) > nrOfTokensToReserve) {
_contractBalance -= nrOfTokensToReserve;
} else {
_contractBalance =1; // eventually, the contract must only own the genesis token
}
require(_contractBalance >=1, "arteQArtDrop: contract balance went below 1");
_whitelistedAccounts[target] -= nrOfTokensToReserve;
require(_whitelistedAccounts[target] >=0, "arteQArtDrop: should not happen");
emit TokensReserved(msg.sender, target, nrOfTokensToReserve);
}
receive() externalpayable{
revert("arteQArtDrop: cannot accept ether");
}
fallback() externalpayable{
revert("arteQArtDrop: cannot accept ether");
}
}