diff --git a/contracts/OriumMarketplace.sol b/contracts/OriumMarketplace.sol deleted file mode 100644 index ed89ad8..0000000 --- a/contracts/OriumMarketplace.sol +++ /dev/null @@ -1,823 +0,0 @@ -// SPDX-License-Identifier: CC0-1.0 - -pragma solidity 0.8.9; - -import { IERC721 } from "@openzeppelin/contracts/token/ERC721/IERC721.sol"; -import { IERC20 } from "@openzeppelin/contracts/token/ERC20/IERC20.sol"; -import { IERC7432, RoleAssignment } from "./interfaces/IERC7432.sol"; -import { OwnableUpgradeable } from "@openzeppelin/contracts-upgradeable/access/OwnableUpgradeable.sol"; -import { Initializable } from "@openzeppelin/contracts-upgradeable/proxy/utils/Initializable.sol"; -import { PausableUpgradeable } from "@openzeppelin/contracts-upgradeable/security/PausableUpgradeable.sol"; - -/** - * @title Orium Marketplace - Marketplace for renting NFTs - * @dev This contract is used to manage NFTs rentals, powered by ERC-7432 Non-Fungible Token Roles - * @author Orium Network Team - developers@orium.network - */ -contract OriumMarketplace is Initializable, OwnableUpgradeable, PausableUpgradeable { - /** ######### Constants ########### **/ - - /// @dev 100 ether is 100% - uint256 public constant MAX_PERCENTAGE = 100 ether; - /// @dev 2.5 ether is 2.5% - uint256 public constant DEFAULT_FEE_PERCENTAGE = 2.5 ether; - - /** ######### Global Variables ########### **/ - - /// @dev rolesRegistry is a ERC-7432 contract - address public defaultRolesRegistry; - - /// @dev tokenAddress => rolesRegistry - mapping(address => address) public tokenRolesRegistry; - - /// @dev deadline is set in seconds - uint256 public maxDeadline; - - /// @dev tokenAddress => feePercentageInWei - mapping(address => FeeInfo) public feeInfo; - - /// @dev tokenAddress => royaltyInfo - mapping(address => RoyaltyInfo) public royaltyInfo; - - /// @dev hashedOffer => bool - mapping(bytes32 => bool) public isCreated; - - /// @dev lender => nonce => deadline - mapping(address => mapping(uint256 => uint64)) public nonceDeadline; - - /// @dev hashedOffer => Rental - mapping(bytes32 => Rental) public rentals; - - /** ######### Structs ########### **/ - - struct Rental { - address borrower; - uint64 expirationDate; - } - - /// @dev Royalty info. Used to charge fees for the creator. - struct RoyaltyInfo { - address creator; - uint256 royaltyPercentageInWei; - address treasury; - } - - /// @dev Marketplace fee info. - struct FeeInfo { - uint256 feePercentageInWei; - bool isCustomFee; - } - - /// @dev Rental offer info. - struct RentalOffer { - address lender; - address borrower; - address tokenAddress; - uint256 tokenId; - address feeTokenAddress; - uint256 feeAmountPerSecond; - uint256 nonce; - uint64 deadline; - bytes32[] roles; - bytes[] rolesData; - } - - /// @dev Direct rental info. - struct DirectRental { - address tokenAddress; - uint256 tokenId; - address lender; - address borrower; - uint64 duration; - bytes32[] roles; - bytes[] rolesData; - } - - /** ######### Events ########### **/ - - /** - * @param tokenAddress The NFT address. - * @param feePercentageInWei The fee percentage in wei. - * @param isCustomFee If the fee is custom or not. Used to allow collections with no fee. - */ - event MarketplaceFeeSet(address indexed tokenAddress, uint256 feePercentageInWei, bool isCustomFee); - /** - * @param tokenAddress The NFT address. - * @param creator The address of the creator. - * @param royaltyPercentageInWei The royalty percentage in wei. - * @param treasury The address where the fees will be sent. If the treasury is address(0), the fees will be burned. - */ - event CreatorRoyaltySet( - address indexed tokenAddress, - address indexed creator, - uint256 royaltyPercentageInWei, - address treasury - ); - /** - * @param nonce The nonce of the rental offer - * @param tokenAddress The address of the contract of the NFT to rent - * @param tokenId The tokenId of the NFT to rent - * @param lender The address of the user lending the NFT - * @param borrower The address of the user renting the NFT - * @param feeTokenAddress The address of the ERC20 token for rental fees - * @param feeAmountPerSecond The amount of fee per second - * @param deadline The deadline until when the rental offer is valid - * @param roles The array of roles to be assigned to the borrower - * @param rolesData The array of data for each role - */ - event RentalOfferCreated( - uint256 indexed nonce, - address indexed tokenAddress, - uint256 indexed tokenId, - address lender, - address borrower, - address feeTokenAddress, - uint256 feeAmountPerSecond, - uint256 deadline, - bytes32[] roles, - bytes[] rolesData - ); - - /** - * @param nonce The nonce of the rental offer - * @param tokenAddress The address of the contract of the NFT rented - * @param tokenId The tokenId of the rented NFT - * @param lender The address of the lender - * @param borrower The address of the borrower - * @param expirationDate The expiration date of the rental - */ - event RentalStarted( - uint256 indexed nonce, - address indexed tokenAddress, - uint256 indexed tokenId, - address lender, - address borrower, - uint64 expirationDate - ); - - /** - * @param nonce The nonce of the rental offer - * @param lender The address of the user lending the NFT - */ - event RentalOfferCancelled(uint256 indexed nonce, address indexed lender); - - /** - * @param nonce The nonce of the rental offer - * @param tokenAddress The address of the contract of the NFT rented - * @param tokenId The tokenId of the rented NFT - * @param lender The address of the lender - * @param borrower The address of the borrower - */ - event RentalEnded( - uint256 indexed nonce, - address indexed tokenAddress, - uint256 indexed tokenId, - address lender, - address borrower - ); - - /** - * @param tokenAddress The NFT address. - * @param rolesRegistry The address of the roles registry. - */ - event RolesRegistrySet(address indexed tokenAddress, address indexed rolesRegistry); - - /** - * @param directRentalHash The hash of the direct rental - * @param tokenAddress The address of the contract of the NFT rented - * @param tokenId The tokenId of the rented NFT - * @param lender The address of the lender - * @param borrower The address of the borrower - * @param duration The duration of the rental - * @param roles The array of roles to be assigned to the borrower - * @param rolesData The array of data for each role - */ - event DirectRentalStarted( - bytes32 indexed directRentalHash, - address indexed tokenAddress, - uint256 indexed tokenId, - address lender, - address borrower, - uint256 duration, - bytes32[] roles, - bytes[] rolesData - ); - - /** - * @param directRentalHash The hash of the direct rental - * @param lender The address of the user lending the NFT - */ - event DirectRentalEnded(bytes32 indexed directRentalHash, address indexed lender); - - /** ######### Modifiers ########### **/ - - /** - * @notice Checks the ownership of the token. - * @dev Throws if the caller is not the owner of the token. - * @param _tokenAddress The NFT address. - * @param _tokenId The id of the token. - */ - modifier onlyTokenOwner(address _tokenAddress, uint256 _tokenId) { - require( - msg.sender == IERC721(_tokenAddress).ownerOf(_tokenId), - "OriumMarketplace: only token owner can call this function" - ); - _; - } - - /** ######### Initializer ########### **/ - /** - * @notice Initializes the contract. - * @dev The owner of the contract will be the owner of the protocol. - * @param _owner The owner of the protocol. - * @param _defaultRolesRegistry The address of the roles registry. - * @param _maxDeadline The maximum deadline. - */ - function initialize(address _owner, address _defaultRolesRegistry, uint256 _maxDeadline) public initializer { - __Pausable_init(); - __Ownable_init(); - - defaultRolesRegistry = _defaultRolesRegistry; - maxDeadline = _maxDeadline; - - transferOwnership(_owner); - } - - /** ============================ Rental Functions ================================== **/ - - /** ######### Setters ########### **/ - /** - * @notice Creates a rental offer. - * @dev To optimize for gas, only the offer hash is stored on-chain - * @param _offer The rental offer struct. - */ - function createRentalOffer( - RentalOffer calldata _offer - ) external onlyTokenOwner(_offer.tokenAddress, _offer.tokenId) { - _validateCreateRentalOffer(_offer); - - bytes32 _offerHash = hashRentalOffer(_offer); - - nonceDeadline[msg.sender][_offer.nonce] = _offer.deadline; - isCreated[_offerHash] = true; - - emit RentalOfferCreated( - _offer.nonce, - _offer.tokenAddress, - _offer.tokenId, - _offer.lender, - _offer.borrower, - _offer.feeTokenAddress, - _offer.feeAmountPerSecond, - _offer.deadline, - _offer.roles, - _offer.rolesData - ); - } - - /** - * @dev Validates the create rental offer. - * @param _offer The rental offer struct. - */ - function _validateCreateRentalOffer(RentalOffer calldata _offer) internal view { - require(_offer.nonce != 0, "OriumMarketplace: Nonce cannot be 0"); - require(msg.sender == _offer.lender, "OriumMarketplace: Sender and Lender mismatch"); - require(_offer.roles.length > 0, "OriumMarketplace: roles should not be empty"); - require( - _offer.roles.length == _offer.rolesData.length, - "OriumMarketplace: roles and rolesData should have the same length" - ); - require( - _offer.deadline <= block.timestamp + maxDeadline && _offer.deadline > block.timestamp, - "OriumMarketplace: Invalid deadline" - ); - require(nonceDeadline[_offer.lender][_offer.nonce] == 0, "OriumMarketplace: nonce already used"); - } - - function cancelRentalOffer(uint256 nonce) external { - require(nonceDeadline[msg.sender][nonce] > block.timestamp, "OriumMarketplace: Nonce expired or not used yet"); - - nonceDeadline[msg.sender][nonce] = uint64(block.timestamp); - emit RentalOfferCancelled(nonce, msg.sender); - } - - /** - * @notice Accepts a rental offer. - * @dev The borrower can be address(0) to allow anyone to rent the NFT. - * @param _offer The rental offer struct. It should be the same as the one used to create the offer. - * @param _duration The duration of the rental. - */ - function acceptRentalOffer(RentalOffer calldata _offer, uint64 _duration) external { - uint64 _expirationDate = uint64(block.timestamp + _duration); - - _validateAcceptRentalOffer(_offer, _expirationDate); - - _transferFees(_offer.tokenAddress, _offer.feeTokenAddress, _offer.feeAmountPerSecond, _duration, _offer.lender); - - _batchGrantRole( - _offer.roles, - _offer.rolesData, - _offer.tokenAddress, - _offer.tokenId, - _offer.lender, - msg.sender, - _expirationDate, - false - ); - - rentals[hashRentalOffer(_offer)] = Rental({ borrower: msg.sender, expirationDate: _expirationDate }); - - emit RentalStarted( - _offer.nonce, - _offer.tokenAddress, - _offer.tokenId, - _offer.lender, - msg.sender, - _expirationDate - ); - } - - /** - * @dev Validates the accept rental offer. - * @param _offer The rental offer struct. It should be the same as the one used to create the offer. - * @param _expirationDate The period of time the NFT will be rented. - */ - function _validateAcceptRentalOffer(RentalOffer calldata _offer, uint64 _expirationDate) internal view { - bytes32 _offerHash = hashRentalOffer(_offer); - require(rentals[_offerHash].expirationDate <= block.timestamp, "OriumMarketplace: Rental already started"); - require(isCreated[_offerHash], "OriumMarketplace: Offer not created"); - require( - address(0) == _offer.borrower || msg.sender == _offer.borrower, - "OriumMarketplace: Sender is not allowed to rent this NFT" - ); - require( - nonceDeadline[_offer.lender][_offer.nonce] > _expirationDate, - "OriumMarketplace: expiration date is greater than offer deadline" - ); - } - - /** - * @dev Transfers the fees to the marketplace, the creator and the lender. - * @param _feeTokenAddress The address of the ERC20 token for rental fees. - * @param _feeAmountPerSecond The amount of fee per second. - * @param _duration The duration of the rental. - * @param _lenderAddress The address of the lender. - */ - function _transferFees( - address _tokenAddress, - address _feeTokenAddress, - uint256 _feeAmountPerSecond, - uint64 _duration, - address _lenderAddress - ) internal { - uint256 _feeAmount = _feeAmountPerSecond * _duration; - if (_feeAmount == 0) return; - - uint256 _marketplaceFeeAmount = _getAmountFromPercentage(_feeAmount, marketplaceFeeOf(_tokenAddress)); - if (_marketplaceFeeAmount > 0) { - require( - IERC20(_feeTokenAddress).transferFrom(msg.sender, owner(), _marketplaceFeeAmount), - "OriumMarketplace: Transfer failed" - ); - } - - uint256 _royaltyAmount = _getAmountFromPercentage( - _feeAmount, - royaltyInfo[_tokenAddress].royaltyPercentageInWei - ); - if (_royaltyAmount > 0) { - require( - IERC20(_feeTokenAddress).transferFrom(msg.sender, royaltyInfo[_tokenAddress].treasury, _royaltyAmount), - "OriumMarketplace: Transfer failed" - ); - } - - uint256 _lenderAmount = _feeAmount - _royaltyAmount - _marketplaceFeeAmount; - require( - IERC20(_feeTokenAddress).transferFrom(msg.sender, _lenderAddress, _lenderAmount), - "OriumMarketplace: Transfer failed" - ); // TODO: Change to vesting contract address later - } - - /** - * @dev All values needs to be in wei. - * @param _amount The amount to calculate the percentage from. - * @param _percentage The percentage to calculate. - */ - function _getAmountFromPercentage(uint256 _amount, uint256 _percentage) internal pure returns (uint256) { - return (_amount * _percentage) / MAX_PERCENTAGE; - } - - /** - * @dev Grants the roles to the borrower. - * @param _roles The array of roles to be assigned to the borrower - * @param _rolesData The array of data for each role - * @param _tokenAddress The address of the contract of the NFT to rent - * @param _tokenId The tokenId of the NFT to rent - * @param _grantor The address of the user lending the NFT - * @param _grantee The address of the user renting the NFT - * @param _expirationDate The deadline until when the rental offer is valid - */ - function _batchGrantRole( - bytes32[] memory _roles, - bytes[] memory _rolesData, - address _tokenAddress, - uint256 _tokenId, - address _grantor, - address _grantee, - uint64 _expirationDate, - bool _revocable - ) internal { - address _rolesRegistry = rolesRegistryOf(_tokenAddress); - for (uint256 i = 0; i < _roles.length; i++) { - _grantUniqueRoleChecked( // Needed to avoid stack too deep error - _roles[i], - _tokenAddress, - _tokenId, - _grantor, - _grantee, - _expirationDate, - _rolesData[i], - _rolesRegistry, - _revocable - ); - } - } - - /** - * @dev Grants the role to the borrower. - * @param _role The role to be granted - * @param _tokenAddress The address of the contract of the NFT to rent - * @param _tokenId The tokenId of the NFT to rent - * @param _grantor The address of the user lending the NFT - * @param _grantee The address of the user renting the NFT - * @param _expirationDate The deadline until when the rental offer is valid - * @param _data The data for the role - */ - function _grantUniqueRoleChecked( - bytes32 _role, - address _tokenAddress, - uint256 _tokenId, - address _grantor, - address _grantee, - uint64 _expirationDate, - bytes memory _data, - address _rolesRegistry, - bool _revocable - ) internal { - RoleAssignment memory _roleAssignment = RoleAssignment({ - role: _role, - tokenAddress: _tokenAddress, - tokenId: _tokenId, - grantor: _grantor, - grantee: _grantee, - expirationDate: _expirationDate, - data: _data - }); - if (_revocable) { - IERC7432(_rolesRegistry).grantRevocableRoleFrom(_roleAssignment); - } else { - IERC7432(_rolesRegistry).grantRoleFrom(_roleAssignment); - } - } - - /** - * @notice Ends the rental. - * @dev Can only be called by the borrower. - * @dev Borrower needs to approve marketplace to revoke the roles. - * @param _offer The rental offer struct. It should be the same as the one used to create the offer. - */ - function endRental(RentalOffer memory _offer) external { - bytes32 _offerHash = hashRentalOffer(_offer); - - _validateEndRental(_offer, _offerHash); - - _batchRevokeRole( - _offer.roles, - _offer.tokenAddress, - _offer.tokenId, - _offer.lender, - rentals[_offerHash].borrower - ); - - rentals[_offerHash].expirationDate = uint64(block.timestamp); - - emit RentalEnded( - _offer.nonce, - _offer.tokenAddress, - _offer.tokenId, - _offer.lender, - rentals[_offerHash].borrower - ); - } - - /** - * @dev Validates the end rental. - * @param _offer The rental offer struct. It should be the same as the one used to create the offer. - * @param _offerHash The hash of the rental offer struct. - */ - function _validateEndRental(RentalOffer memory _offer, bytes32 _offerHash) internal view { - require(isCreated[_offerHash], "OriumMarketplace: Offer not created"); - require(msg.sender == rentals[_offerHash].borrower, "OriumMarketplace: Only borrower can end a rental"); - require(nonceDeadline[_offer.lender][_offer.nonce] > block.timestamp, "OriumMarketplace: Rental Offer expired"); - require(rentals[_offerHash].expirationDate > block.timestamp, "OriumMarketplace: Rental ended"); - } - - /** - * @dev Revokes the roles from the borrower. - * @param _roles The array of roles to be revoked from the borrower - * @param _tokenAddress The address of the contract of the NFT to rent - * @param _tokenId The tokenId of the NFT to rent - * @param _grantor The address of the user lending the NFT - * @param _grantee The address of the user renting the NFT - */ - function _batchRevokeRole( - bytes32[] memory _roles, - address _tokenAddress, - uint256 _tokenId, - address _grantor, - address _grantee - ) internal { - address _rolesRegistry = rolesRegistryOf(_tokenAddress); - for (uint256 i = 0; i < _roles.length; i++) { - IERC7432(_rolesRegistry).revokeRoleFrom(_roles[i], _tokenAddress, _tokenId, _grantor, _grantee); - } - } - - /** - * @notice Creates a direct rental. - * @dev The lender needs to approve marketplace to grant the roles. - * @param _directRental The direct rental struct. - */ - function createDirectRental( - DirectRental memory _directRental - ) external onlyTokenOwner(_directRental.tokenAddress, _directRental.tokenId) { - _validateCreateDirectRental(_directRental); - - bytes32 _hashedDirectRental = hashDirectRental(_directRental); - uint64 _expirationDate = uint64(block.timestamp + _directRental.duration); - isCreated[_hashedDirectRental] = true; - rentals[_hashedDirectRental] = Rental({ borrower: _directRental.borrower, expirationDate: _expirationDate }); - - _batchGrantRole( - _directRental.roles, - _directRental.rolesData, - _directRental.tokenAddress, - _directRental.tokenId, - _directRental.lender, - _directRental.borrower, - _expirationDate, - true - ); - - emit DirectRentalStarted( - _hashedDirectRental, - _directRental.tokenAddress, - _directRental.tokenId, - _directRental.lender, - _directRental.borrower, - _directRental.duration, - _directRental.roles, - _directRental.rolesData - ); - } - - /** - * @dev Validates the create direct rental. - * @param _directRental The direct rental struct. - */ - function _validateCreateDirectRental(DirectRental memory _directRental) internal view { - require(_directRental.duration <= maxDeadline, "OriumMarketplace: Duration is greater than max deadline"); - require(msg.sender == _directRental.lender, "OriumMarketplace: Sender and Lender mismatch"); - require(_directRental.roles.length > 0, "OriumMarketplace: roles should not be empty"); - require( - _directRental.roles.length == _directRental.rolesData.length, - "OriumMarketplace: roles and rolesData should have the same length" - ); - } - - /** - * @notice Cancels a direct rental. - * @dev The lender needs to approve marketplace to revoke the roles. - * @param _directRental The direct rental struct. - */ - function cancelDirectRental(DirectRental memory _directRental) external { - bytes32 _hashedDirectRental = hashDirectRental(_directRental); - - _validateCancelDirectRental(_directRental, _hashedDirectRental); - - rentals[_hashedDirectRental].expirationDate = uint64(block.timestamp); - - _batchRevokeRole( - _directRental.roles, - _directRental.tokenAddress, - _directRental.tokenId, - _directRental.lender, - _directRental.borrower - ); - - emit DirectRentalEnded(_hashedDirectRental, _directRental.lender); - } - - function _validateCancelDirectRental(DirectRental memory _directRental, bytes32 _hashedDirectRental) internal view { - require(isCreated[_hashedDirectRental], "OriumMarketplace: Direct rental not created"); - require( - rentals[_hashedDirectRental].expirationDate > block.timestamp, - "OriumMarketplace: Direct rental expired" - ); - require( - msg.sender == _directRental.lender || msg.sender == _directRental.borrower, - "OriumMarketplace: Sender and Lender/Borrower mismatch" - ); - } - - /** ######### Getters ########### **/ - - /** - * @notice Gets the rental offer hash. - * @param _offer The rental offer struct to be hashed. - */ - function hashRentalOffer(RentalOffer memory _offer) public pure returns (bytes32) { - return - keccak256( - abi.encode( - _offer.lender, - _offer.borrower, - _offer.tokenAddress, - _offer.tokenId, - _offer.feeTokenAddress, - _offer.feeAmountPerSecond, - _offer.nonce, - _offer.deadline, - _offer.roles, - _offer.rolesData - ) - ); - } - - /** - * @notice Gets the direct rental hash. - * @param _directRental The direct rental struct to be hashed. - */ - function hashDirectRental(DirectRental memory _directRental) public pure returns (bytes32) { - return - keccak256( - abi.encode( - _directRental.tokenAddress, - _directRental.tokenId, - _directRental.lender, - _directRental.borrower, - _directRental.duration, - _directRental.roles, - _directRental.rolesData - ) - ); - } - - /** ============================ Core Functions ================================== **/ - - /** ######### Setters ########### **/ - - /** - * @notice Sets the roles registry. - * @dev Only owner can set the roles registry. - */ - function pause() external onlyOwner { - _pause(); - } - - /** - * @notice Unpauses the contract. - * @dev Only owner can unpause the contract. - */ - function unpause() external onlyOwner { - _unpause(); - } - - /** - * @notice Sets the marketplace fee for a collection. - * @dev If no fee is set, the default fee will be used. - * @param _tokenAddress The NFT address. - * @param _feePercentageInWei The fee percentage in wei. - * @param _isCustomFee If the fee is custom or not. - */ - function setMarketplaceFeeForCollection( - address _tokenAddress, - uint256 _feePercentageInWei, - bool _isCustomFee - ) external onlyOwner { - uint256 _royaltyPercentage = royaltyInfo[_tokenAddress].royaltyPercentageInWei; - require( - _royaltyPercentage + _feePercentageInWei < MAX_PERCENTAGE, - "OriumMarketplace: Royalty percentage + marketplace fee cannot be greater than 100%" - ); - - feeInfo[_tokenAddress] = FeeInfo({ feePercentageInWei: _feePercentageInWei, isCustomFee: _isCustomFee }); - - emit MarketplaceFeeSet(_tokenAddress, _feePercentageInWei, _isCustomFee); - } - - /** - * @notice Sets the royalty info. - * @dev Only owner can associate a collection with a creator. - * @param _tokenAddress The NFT address. - * @param _creator The address of the creator. - */ - function setCreator(address _tokenAddress, address _creator) external onlyOwner { - _setRoyalty(_creator, _tokenAddress, 0, address(0)); - } - - /** - * @notice Sets the royalty info. - * @param _tokenAddress The NFT address. - * @param _royaltyPercentageInWei The royalty percentage in wei. - * @param _treasury The address where the fees will be sent. If the treasury is address(0), the fees will be burned. - */ - function setRoyaltyInfo(address _tokenAddress, uint256 _royaltyPercentageInWei, address _treasury) external { - require( - msg.sender == royaltyInfo[_tokenAddress].creator, - "OriumMarketplace: Only creator can set royalty info" - ); - - _setRoyalty(msg.sender, _tokenAddress, _royaltyPercentageInWei, _treasury); - } - - /** - * @notice Sets the royalty info. - * @dev Only owner can associate a collection with a creator. - * @param _creator The address of the creator. - * @param _tokenAddress The NFT address. - * @param _royaltyPercentageInWei The royalty percentage in wei. - * @param _treasury The address where the fees will be sent. If the treasury is address(0), the fees will be burned. - */ - function _setRoyalty( - address _creator, - address _tokenAddress, - uint256 _royaltyPercentageInWei, - address _treasury - ) internal { - require( - _royaltyPercentageInWei + marketplaceFeeOf(_tokenAddress) < MAX_PERCENTAGE, - "OriumMarketplace: Royalty percentage + marketplace fee cannot be greater than 100%" - ); - - royaltyInfo[_tokenAddress] = RoyaltyInfo({ - creator: _creator, - royaltyPercentageInWei: _royaltyPercentageInWei, - treasury: _treasury - }); - - emit CreatorRoyaltySet(_tokenAddress, _creator, _royaltyPercentageInWei, _treasury); - } - - /** - * @notice Sets the maximum deadline. - * @dev Only owner can set the maximum deadline. - * @param _maxDeadline The maximum deadline. - */ - function setMaxDeadline(uint256 _maxDeadline) external onlyOwner { - require(_maxDeadline > 0, "OriumMarketplace: Max deadline should be greater than 0"); - maxDeadline = _maxDeadline; - } - - /** - * @notice Sets the roles registry for a collection. - * @dev Only owner can set the roles registry for a collection. - * @param _tokenAddress The NFT address. - * @param _rolesRegistry The roles registry address. - */ - function setRolesRegistry(address _tokenAddress, address _rolesRegistry) external onlyOwner { - tokenRolesRegistry[_tokenAddress] = _rolesRegistry; - emit RolesRegistrySet(_tokenAddress, _rolesRegistry); - } - - /** - * @notice Sets the default roles registry. - * @dev Only owner can set the default roles registry. - * @param _rolesRegistry The roles registry address. - */ - function setDefaultRolesRegistry(address _rolesRegistry) external onlyOwner { - defaultRolesRegistry = _rolesRegistry; - } - - /** ######### Getters ########### **/ - - /** - * @notice Gets the marketplace fee for a collection. - * @dev If no custom fee is set, the default fee will be used. - * @param _tokenAddress The NFT address. - */ - function marketplaceFeeOf(address _tokenAddress) public view returns (uint256) { - return feeInfo[_tokenAddress].isCustomFee ? feeInfo[_tokenAddress].feePercentageInWei : DEFAULT_FEE_PERCENTAGE; - } - - /** - * @notice Gets the roles registry for a collection. - * @dev If no custom roles registry is set, the default roles registry will be used. - * @param _tokenAddress The NFT address. - */ - function rolesRegistryOf(address _tokenAddress) public view returns (address) { - return - tokenRolesRegistry[_tokenAddress] == address(0) ? defaultRolesRegistry : tokenRolesRegistry[_tokenAddress]; - } -} diff --git a/contracts/OriumNftMarketplace.sol b/contracts/OriumNftMarketplace.sol index a602172..f20af42 100644 --- a/contracts/OriumNftMarketplace.sol +++ b/contracts/OriumNftMarketplace.sol @@ -3,10 +3,12 @@ pragma solidity 0.8.9; import { IERC721 } from '@openzeppelin/contracts/token/ERC721/IERC721.sol'; +import { IERC20 } from '@openzeppelin/contracts/token/ERC20/IERC20.sol'; import { IOriumMarketplaceRoyalties } from './interfaces/IOriumMarketplaceRoyalties.sol'; import { OwnableUpgradeable } from '@openzeppelin/contracts-upgradeable/access/OwnableUpgradeable.sol'; import { Initializable } from '@openzeppelin/contracts-upgradeable/proxy/utils/Initializable.sol'; import { PausableUpgradeable } from '@openzeppelin/contracts-upgradeable/security/PausableUpgradeable.sol'; +import { LibOriumNftMarketplace, RentalOffer, Rental } from './libraries/LibOriumNftMarketplace.sol'; /** * @title Orium NFT Marketplace - Marketplace for renting NFTs @@ -28,22 +30,8 @@ contract OriumNftMarketplace is Initializable, OwnableUpgradeable, PausableUpgra /// @dev role => tokenAddress => tokenId => deadline mapping(bytes32 => mapping(address => mapping(uint256 => uint64))) public roleDeadline; - /** ######### Structs ########### **/ - - /// @dev Rental offer info. - struct RentalOffer { - address lender; - address borrower; - address tokenAddress; - uint256 tokenId; - address feeTokenAddress; - uint256 feeAmountPerSecond; - uint256 nonce; - uint64 deadline; - uint64 minDuration; - bytes32[] roles; - bytes[] rolesData; - } + /// @dev hashedOffer => Rental + mapping(bytes32 => Rental) public rentals; /** ######### Events ########### **/ @@ -73,6 +61,14 @@ contract OriumNftMarketplace is Initializable, OwnableUpgradeable, PausableUpgra bytes[] rolesData ); + /** + * @param lender The address of the lender + * @param nonce The nonce of the rental offer + * @param borrower The address of the borrower + * @param expirationDate The expiration date of the rental + */ + event RentalStarted(address indexed lender, uint256 indexed nonce, address indexed borrower, uint64 expirationDate); + /** ######### Modifiers ########### **/ /** @@ -116,9 +112,13 @@ contract OriumNftMarketplace is Initializable, OwnableUpgradeable, PausableUpgra function createRentalOffer( RentalOffer calldata _offer ) external onlyTokenOwner(_offer.tokenAddress, _offer.tokenId) { - _validateCreateRentalOffer(_offer); + LibOriumNftMarketplace.validateCreateRentalOfferParams( + oriumMarketplaceRoyalties, + _offer, + nonceDeadline[_offer.lender][_offer.nonce] + ); - bytes32 _offerHash = hashRentalOffer(_offer); + bytes32 _offerHash = LibOriumNftMarketplace.hashRentalOffer(_offer); nonceDeadline[msg.sender][_offer.nonce] = _offer.deadline; isCreated[_offerHash] = true; @@ -145,14 +145,49 @@ contract OriumNftMarketplace is Initializable, OwnableUpgradeable, PausableUpgra ); } - /** ######### Getters ########### **/ - /** - * @notice Gets the rental offer hash. - * @param _offer The rental offer struct to be hashed. + * @notice Accepts a rental offer. + * @dev The borrower can be address(0) to allow anyone to rent the NFT. + * @param _offer The rental offer struct. It should be the same as the one used to create the offer. + * @param _duration The duration of the rental. */ - function hashRentalOffer(RentalOffer memory _offer) public pure returns (bytes32) { - return keccak256(abi.encode(_offer)); + function acceptRentalOffer(RentalOffer calldata _offer, uint64 _duration) external whenNotPaused { + bytes32 _offerHash = LibOriumNftMarketplace.hashRentalOffer(_offer); + uint64 _expirationDate = uint64(block.timestamp + _duration); + + LibOriumNftMarketplace.validateAcceptRentalOfferParams( + _offer.borrower, + _offer.minDuration, + isCreated[_offerHash], + rentals[_offerHash].expirationDate, + _duration, + nonceDeadline[_offer.lender][_offer.nonce], + _expirationDate + ); + + LibOriumNftMarketplace.transferFees( + _offer.feeTokenAddress, + owner(), + _offer.lender, + oriumMarketplaceRoyalties, + _offer.tokenAddress, + _offer.feeAmountPerSecond, + _duration + ); + + LibOriumNftMarketplace.batchGrantRole( + oriumMarketplaceRoyalties, + _offer.tokenAddress, + _offer.tokenId, + _offer.borrower, + _expirationDate, + _offer.roles, + _offer.rolesData + ); + + rentals[_offerHash] = Rental({ borrower: msg.sender, expirationDate: _expirationDate }); + + emit RentalStarted(_offer.lender, _offer.nonce, msg.sender, _expirationDate); } /** ============================ Core Functions ================================== **/ @@ -174,39 +209,4 @@ contract OriumNftMarketplace is Initializable, OwnableUpgradeable, PausableUpgra function unpause() external onlyOwner { _unpause(); } - - /** ######### Internals ########### **/ - - /** - * @dev Validates the create rental offer. - * @param _offer The rental offer struct. - */ - function _validateCreateRentalOffer(RentalOffer calldata _offer) internal view { - require( - IOriumMarketplaceRoyalties(oriumMarketplaceRoyalties).isTrustedFeeTokenAddressForToken( - _offer.tokenAddress, - _offer.feeTokenAddress - ), - 'OriumSftMarketplace: tokenAddress is not trusted' - ); - require( - _offer.deadline <= block.timestamp + IOriumMarketplaceRoyalties(oriumMarketplaceRoyalties).maxDuration() && - _offer.deadline > block.timestamp, - 'OriumNftMarketplace: Invalid deadline' - ); - require(nonceDeadline[_offer.lender][_offer.nonce] == 0, 'OriumNftMarketplace: nonce already used'); - - require(_offer.nonce != 0, 'OriumNftMarketplace: Nonce cannot be 0'); - require(msg.sender == _offer.lender, 'OriumNftMarketplace: Sender and Lender mismatch'); - require(_offer.roles.length > 0, 'OriumNftMarketplace: roles should not be empty'); - require( - _offer.roles.length == _offer.rolesData.length, - 'OriumNftMarketplace: roles and rolesData should have the same length' - ); - require( - _offer.borrower != address(0) || _offer.feeAmountPerSecond > 0, - 'OriumNftMarketplace: feeAmountPerSecond should be greater than 0' - ); - require(_offer.minDuration <= _offer.deadline - block.timestamp, 'OriumNftMarketplace: minDuration is invalid'); - } } diff --git a/contracts/interfaces/IERC7432.sol b/contracts/interfaces/IERC7432.sol index 887cb30..d0f103d 100644 --- a/contracts/interfaces/IERC7432.sol +++ b/contracts/interfaces/IERC7432.sol @@ -2,95 +2,69 @@ pragma solidity 0.8.9; -import { IERC165 } from "@openzeppelin/contracts/utils/introspection/IERC165.sol"; - -struct RoleData { - uint64 expirationDate; - bool revocable; - bytes data; -} - -struct RoleAssignment { - bytes32 role; - address tokenAddress; - uint256 tokenId; - address grantor; - address grantee; - uint64 expirationDate; - bytes data; -} +import { IERC165 } from '@openzeppelin/contracts/utils/introspection/IERC165.sol'; /// @title ERC-7432 Non-Fungible Token Roles /// @dev See https://eips.ethereum.org/EIPS/eip-7432 -/// Note: the ERC-165 identifier for this interface is 0x04984ac8. +/// Note: the ERC-165 identifier for this interface is 0xfecc9ed3. interface IERC7432 is IERC165 { + struct Role { + bytes32 roleId; + address tokenAddress; + uint256 tokenId; + address recipient; + uint64 expirationDate; + bool revocable; + bytes data; + } + /** Events **/ /// @notice Emitted when a role is granted. - /// @param _role The role identifier. /// @param _tokenAddress The token address. /// @param _tokenId The token identifier. - /// @param _grantor The user assigning the role. - /// @param _grantee The user receiving the role. + /// @param _roleId The role identifier. + /// @param _owner The user assigning the role. + /// @param _recipient The user receiving the role. /// @param _expirationDate The expiration date of the role. /// @param _revocable Whether the role is revocable or not. /// @param _data Any additional data about the role. event RoleGranted( - bytes32 indexed _role, address indexed _tokenAddress, uint256 indexed _tokenId, - address _grantor, - address _grantee, + bytes32 indexed _roleId, + address _owner, + address _recipient, uint64 _expirationDate, bool _revocable, bytes _data ); /// @notice Emitted when a role is revoked. - /// @param _role The role identifier. /// @param _tokenAddress The token address. /// @param _tokenId The token identifier. - /// @param _revoker The user revoking the role. - /// @param _grantee The user that receives the role revocation. - event RoleRevoked( - bytes32 indexed _role, - address indexed _tokenAddress, - uint256 indexed _tokenId, - address _revoker, - address _grantee - ); + /// @param _roleId The role identifier. + event RoleRevoked(address indexed _tokenAddress, uint256 indexed _tokenId, bytes32 indexed _roleId); - /// @notice Emitted when a user is approved to manage any role on behalf of another user. + /// @notice Emitted when a user is approved to manage roles on behalf of another user. /// @param _tokenAddress The token address. /// @param _operator The user approved to grant and revoke roles. /// @param _isApproved The approval status. - event RoleApprovalForAll(address indexed _tokenAddress, address indexed _operator, bool _isApproved); + event RoleApprovalForAll(address indexed _tokenAddress, address indexed _operator, bool indexed _isApproved); /** External Functions **/ - /// @notice Grants a role on behalf of a user. - /// @param _roleAssignment The role assignment data. - function grantRoleFrom(RoleAssignment calldata _roleAssignment) external; - - /// @notice Grants a role on behalf of a user. - /// @param _roleAssignment The role assignment data. - function grantRevocableRoleFrom(RoleAssignment calldata _roleAssignment) external; + /// @notice Grants a role to a user. + /// @param _role The role attributes. + function grantRole(Role calldata _role) external; - /// @notice Revokes a role on behalf of a user. - /// @param _role The role identifier. + /// @notice Revokes a role from a user. /// @param _tokenAddress The token address. /// @param _tokenId The token identifier. - /// @param _revoker The user revoking the role. - /// @param _grantee The user that receives the role revocation. - function revokeRoleFrom( - bytes32 _role, - address _tokenAddress, - uint256 _tokenId, - address _revoker, - address _grantee - ) external; + /// @param _roleId The role identifier. + function revokeRole(address _tokenAddress, uint256 _tokenId, bytes32 _roleId) external; - /// @notice Approves operator to grant and revoke any roles on behalf of another user. + /// @notice Approves operator to grant and revoke roles on behalf of another user. /// @param _tokenAddress The token address. /// @param _operator The user approved to grant and revoke roles. /// @param _approved The approval status. @@ -98,81 +72,58 @@ interface IERC7432 is IERC165 { /** View Functions **/ - /// @notice Checks if a user has a role. - /// @param _role The role identifier. + /// @notice Retrieves the recipient of an NFT role. /// @param _tokenAddress The token address. /// @param _tokenId The token identifier. - /// @param _grantor The user that assigned the role. - /// @param _grantee The user that received the role. - function hasNonUniqueRole( - bytes32 _role, + /// @param _roleId The role identifier. + /// @return recipient_ The user that received the role. + function recipientOf( address _tokenAddress, uint256 _tokenId, - address _grantor, - address _grantee - ) external view returns (bool); + bytes32 _roleId + ) external view returns (address recipient_); - /// @notice Checks if a user has a unique role. - /// @param _role The role identifier. + /// @notice Retrieves the custom data of a role assignment. /// @param _tokenAddress The token address. /// @param _tokenId The token identifier. - /// @param _grantor The user that assigned the role. - /// @param _grantee The user that received the role. - function hasRole( - bytes32 _role, + /// @param _roleId The role identifier. + /// @return data_ The custom data of the role. + function roleData( address _tokenAddress, uint256 _tokenId, - address _grantor, - address _grantee - ) external view returns (bool); + bytes32 _roleId + ) external view returns (bytes memory data_); - /// @notice Returns the custom data of a role assignment. - /// @param _role The role identifier. + /// @notice Retrieves the expiration date of a role assignment. /// @param _tokenAddress The token address. /// @param _tokenId The token identifier. - /// @param _grantor The user that assigned the role. - /// @param _grantee The user that received the role. - function roleData( - bytes32 _role, + /// @param _roleId The role identifier. + /// @return expirationDate_ The expiration date of the role. + function roleExpirationDate( address _tokenAddress, uint256 _tokenId, - address _grantor, - address _grantee - ) external view returns (RoleData memory data_); + bytes32 _roleId + ) external view returns (uint64 expirationDate_); - /// @notice Returns the expiration date of a role assignment. - /// @param _role The role identifier. + /// @notice Verifies if the role is revocable. /// @param _tokenAddress The token address. /// @param _tokenId The token identifier. - /// @param _grantor The user that assigned the role. - /// @param _grantee The user that received the role. - function roleExpirationDate( - bytes32 _role, + /// @param _roleId The role identifier. + /// @return revocable_ Whether the role is revocable. + function isRoleRevocable( address _tokenAddress, uint256 _tokenId, - address _grantor, - address _grantee - ) external view returns (uint64 expirationDate_); + bytes32 _roleId + ) external view returns (bool revocable_); - /// @notice Checks if the grantor approved the operator for all NFTs. + /// @notice Verifies if the owner approved the operator. /// @param _tokenAddress The token address. - /// @param _grantor The user that approved the operator. + /// @param _owner The user that approved the operator. /// @param _operator The user that can grant and revoke roles. + /// @return Whether the operator is approved. function isRoleApprovedForAll( address _tokenAddress, - address _grantor, + address _owner, address _operator ) external view returns (bool); - - /// @notice Returns the last grantee of a role. - /// @param _role The role. - /// @param _tokenAddress The token address. - /// @param _tokenId The token ID. - /// @param _grantor The user that granted the role. - function lastGrantee( - bytes32 _role, - address _tokenAddress, - uint256 _tokenId, - address _grantor - ) external view returns (address); } diff --git a/contracts/interfaces/IERC7432VaultExtension.sol b/contracts/interfaces/IERC7432VaultExtension.sol new file mode 100644 index 0000000..0f35335 --- /dev/null +++ b/contracts/interfaces/IERC7432VaultExtension.sol @@ -0,0 +1,24 @@ +// SPDX-License-Identifier: CC0-1.0 + +pragma solidity 0.8.9; + +/// @title ERC-7432 Vault Extension +/// @dev See https://eips.ethereum.org/EIPS/eip-7432 +/// Note: the ERC-165 identifier for this interface is 0xf3fef3a3. +interface IERC7432VaultExtension { + /** Events **/ + + /// @notice Emitted when an NFT is withdrawn. + /// @param _owner The original owner of the NFT. + /// @param _tokenAddress The token address. + /// @param _tokenId The token identifier. + event Withdraw(address indexed _owner, address indexed _tokenAddress, uint256 indexed _tokenId); + + /** External Functions **/ + + /// @notice Withdraw NFT back to original owner. + /// @dev Reverts if sender is not approved or the original owner. + /// @param _tokenAddress The token address. + /// @param _tokenId The token identifier. + function withdraw(address _tokenAddress, uint256 _tokenId) external; +} diff --git a/contracts/libraries/LibOriumNftMarketplace.sol b/contracts/libraries/LibOriumNftMarketplace.sol new file mode 100644 index 0000000..36e81cd --- /dev/null +++ b/contracts/libraries/LibOriumNftMarketplace.sol @@ -0,0 +1,201 @@ +// SPDX-License-Identifier: CC0-1.0 + +pragma solidity 0.8.9; + +import { IERC7589 } from '../interfaces/IERC7589.sol'; +import { IOriumMarketplaceRoyalties } from '../interfaces/IOriumMarketplaceRoyalties.sol'; +import { IERC20 } from '@openzeppelin/contracts/token/ERC20/IERC20.sol'; +import { IERC7432 } from '../interfaces/IERC7432.sol'; +/// @dev Rental offer info. +struct RentalOffer { + address lender; + address borrower; + address tokenAddress; + uint256 tokenId; + address feeTokenAddress; + uint256 feeAmountPerSecond; + uint256 nonce; + uint64 deadline; + uint64 minDuration; + bytes32[] roles; + bytes[] rolesData; +} + +/// @dev Rental info. +struct Rental { + address borrower; + uint64 expirationDate; +} + +library LibOriumNftMarketplace { + /// @dev 100 ether is 100% + uint256 public constant MAX_PERCENTAGE = 100 ether; + + /** + * @notice Gets the rental offer hash. + * @dev This function is used to hash the rental offer struct with retrocompatibility. + * is only used for reading the hash from the storage. + * @param _offer The rental offer struct to be hashed. + */ + function hashRentalOffer(RentalOffer memory _offer) external pure returns (bytes32) { + return keccak256(abi.encode(_offer)); + } + + /** + * @dev All values needs to be in wei. + * @param _amount The amount to calculate the percentage from. + * @param _percentage The percentage to calculate. + */ + function getAmountFromPercentage(uint256 _amount, uint256 _percentage) public pure returns (uint256) { + return (_amount * _percentage) / MAX_PERCENTAGE; + } + + /** + * @notice Validates the rental offer. + * @param _offer The rental offer struct to be validated. + */ + function validateCreateRentalOfferParams( + address _oriumMarketplaceRoyalties, + RentalOffer memory _offer, + uint64 _nonceDeadline + ) external view { + require( + IOriumMarketplaceRoyalties(_oriumMarketplaceRoyalties).isTrustedFeeTokenAddressForToken( + _offer.tokenAddress, + _offer.feeTokenAddress + ), + 'OriumNftMarketplace: tokenAddress is not trusted' + ); + require( + _offer.deadline <= block.timestamp + IOriumMarketplaceRoyalties(_oriumMarketplaceRoyalties).maxDuration() && + _offer.deadline > block.timestamp, + 'OriumNftMarketplace: Invalid deadline' + ); + require(_offer.nonce != 0, 'OriumNftMarketplace: Nonce cannot be 0'); + require(msg.sender == _offer.lender, 'OriumNftMarketplace: Sender and Lender mismatch'); + require(_offer.roles.length > 0, 'OriumNftMarketplace: roles should not be empty'); + require( + _offer.roles.length == _offer.rolesData.length, + 'OriumNftMarketplace: roles and rolesData should have the same length' + ); + require( + _offer.borrower != address(0) || _offer.feeAmountPerSecond > 0, + 'OriumNftMarketplace: feeAmountPerSecond should be greater than 0' + ); + require(_offer.minDuration <= _offer.deadline - block.timestamp, 'OriumNftMarketplace: minDuration is invalid'); + require(_nonceDeadline == 0, 'OriumNftMarketplace: nonce already used'); + } + + /** + * @notice Transfers the fees. + * @dev The fee token address should be approved before calling this function. + * @param _feeTokenAddress The fee token address. + * @param _marketplaceTreasuryAddress The marketplace treasury address. + * @param _lenderAddress The lender address. + * @param _royaltiesAddress The Orium marketplace royalties contract address. + * @param _tokenAddress The token address. + * @param _feeAmountPerSecond The fee amount per second. + * @param _duration The duration of the rental. + */ + function transferFees( + address _feeTokenAddress, + address _marketplaceTreasuryAddress, + address _lenderAddress, + address _royaltiesAddress, + address _tokenAddress, + uint256 _feeAmountPerSecond, + uint64 _duration + ) external { + uint256 _totalAmount = _feeAmountPerSecond * _duration; + if (_totalAmount == 0) return; + + IOriumMarketplaceRoyalties _royalties = IOriumMarketplaceRoyalties(_royaltiesAddress); + uint256 _marketplaceFeePercentageInWei = _royalties.marketplaceFeeOf(_tokenAddress); + IOriumMarketplaceRoyalties.RoyaltyInfo memory _royaltyInfo = _royalties.royaltyInfoOf(_tokenAddress); + + uint256 _marketplaceAmount = getAmountFromPercentage(_totalAmount, _marketplaceFeePercentageInWei); + uint256 _royaltyAmount = getAmountFromPercentage(_totalAmount, _royaltyInfo.royaltyPercentageInWei); + uint256 _lenderAmount = _totalAmount - _royaltyAmount - _marketplaceAmount; + + _transferAmount(_feeTokenAddress, msg.sender, _marketplaceTreasuryAddress, _marketplaceAmount); + _transferAmount(_feeTokenAddress, msg.sender, _royaltyInfo.treasury, _royaltyAmount); + _transferAmount(_feeTokenAddress, msg.sender, _lenderAddress, _lenderAmount); + } + + /** + * @notice Transfers an amount to a receipient. + * @dev This function is used to make an ERC20 transfer. + * @param _tokenAddress The token address. + * @param _from The sender address. + * @param _to The recipient address. + * @param _amount The amount to transfer. + */ + function _transferAmount(address _tokenAddress, address _from, address _to, uint256 _amount) internal { + if (_amount == 0) return; + require(IERC20(_tokenAddress).transferFrom(_from, _to, _amount), 'OriumNftMarketplace: Transfer failed'); + } + + function validateAcceptRentalOfferParams( + address _borrower, + uint64 _minDuration, + bool _isCreated, + uint64 _previousRentalExpirationDate, + uint64 _duration, + uint256 _nonceDeadline, + uint64 _expirationDate + ) external view { + require(_isCreated, 'OriumNftMarketplace: Offer not created'); + require( + _previousRentalExpirationDate <= block.timestamp, + 'OriumNftMarketplace: This offer has an ongoing rental' + ); + require(_duration >= _minDuration, 'OriumNftMarketplace: Duration is less than the offer minimum duration'); + require( + _nonceDeadline > _expirationDate, + 'OriumNftMarketplace: expiration date is greater than offer deadline' + ); + require( + address(0) == _borrower || msg.sender == _borrower, + 'OriumNftMarketplace: Sender is not allowed to rent this NFT' + ); + } + + /** + * @notice Batch grants roles. + * @dev This function is used to batch grant roles. + * @param _oriumMarketplaceRoyalties The Orium marketplace royalties contract address. + * @param _tokenAddress The token address. + * @param _tokenId The token id. + * @param _recipient The recipient address. + * @param _expirationDate The expiration date. + * @param _roleIds The role ids. + * @param _data The data. + */ + function batchGrantRole( + address _oriumMarketplaceRoyalties, + address _tokenAddress, + uint256 _tokenId, + address _recipient, + uint64 _expirationDate, + bytes32[] calldata _roleIds, + bytes[] calldata _data + ) external { + address _rolesRegsitry = IOriumMarketplaceRoyalties(_oriumMarketplaceRoyalties).nftRolesRegistryOf( + _tokenAddress + ); + + for (uint256 i = 0; i < _roleIds.length; i++) { + IERC7432(_rolesRegsitry).grantRole( + IERC7432.Role({ + roleId: _roleIds[i], + tokenAddress: _tokenAddress, + tokenId: _tokenId, + recipient: _recipient, + expirationDate: _expirationDate, + revocable: false, + data: _data[i] + }) + ); + } + } +} diff --git a/contracts/mocks/NftRolesRegistryVault.sol b/contracts/mocks/NftRolesRegistryVault.sol new file mode 100644 index 0000000..f9f30a3 --- /dev/null +++ b/contracts/mocks/NftRolesRegistryVault.sol @@ -0,0 +1,225 @@ +// SPDX-License-Identifier: CC0-1.0 + +pragma solidity 0.8.9; + +import { IERC7432 } from '../interfaces/IERC7432.sol'; +import { IERC7432VaultExtension } from '../interfaces/IERC7432VaultExtension.sol'; +import { IERC721 } from '@openzeppelin/contracts/token/ERC721/IERC721.sol'; + +contract NftRolesRegistryVault is IERC7432, IERC7432VaultExtension { + struct RoleData { + address recipient; + uint64 expirationDate; + bool revocable; + bytes data; + } + + // tokenAddress => tokenId => owner + mapping(address => mapping(uint256 => address)) public originalOwners; + + // tokenAddress => tokenId => roleId => struct(recipient, expirationDate, revocable, data) + mapping(address => mapping(uint256 => mapping(bytes32 => RoleData))) public roles; + + // owner => tokenAddress => operator => isApproved + mapping(address => mapping(address => mapping(address => bool))) public tokenApprovals; + + /** ERC-7432 External Functions **/ + + function grantRole(IERC7432.Role calldata _role) external override { + require(_role.expirationDate > block.timestamp, 'NftRolesRegistryVault: expiration date must be in the future'); + + // deposit NFT if necessary + // reverts if sender is not approved or original owner + address _originalOwner = _depositNft(_role.tokenAddress, _role.tokenId); + + // role must be expired or revocable + RoleData storage _roleData = roles[_role.tokenAddress][_role.tokenId][_role.roleId]; + require( + _roleData.revocable || _roleData.expirationDate < block.timestamp, + 'NftRolesRegistryVault: role must be expired or revocable' + ); + + roles[_role.tokenAddress][_role.tokenId][_role.roleId] = RoleData( + _role.recipient, + _role.expirationDate, + _role.revocable, + _role.data + ); + + emit RoleGranted( + _role.tokenAddress, + _role.tokenId, + _role.roleId, + _originalOwner, + _role.recipient, + _role.expirationDate, + _role.revocable, + _role.data + ); + } + + function revokeRole(address _tokenAddress, uint256 _tokenId, bytes32 _roleId) external override { + address _recipient = roles[_tokenAddress][_tokenId][_roleId].recipient; + address _caller = _getApprovedCaller(_tokenAddress, _tokenId, _recipient); + + // if caller is recipient, the role can be revoked regardless of its state + if (_caller != _recipient) { + // if caller is owner, the role can only be revoked if revocable or expired + require( + roles[_tokenAddress][_tokenId][_roleId].revocable || + roles[_tokenAddress][_tokenId][_roleId].expirationDate < block.timestamp, + 'NftRolesRegistryVault: role is not revocable nor expired' + ); + } + + delete roles[_tokenAddress][_tokenId][_roleId]; + emit RoleRevoked(_tokenAddress, _tokenId, _roleId); + } + + function setRoleApprovalForAll(address _tokenAddress, address _operator, bool _approved) external override { + tokenApprovals[msg.sender][_tokenAddress][_operator] = _approved; + emit RoleApprovalForAll(_tokenAddress, _operator, _approved); + } + + /** ERC-7432 View Functions **/ + + function recipientOf( + address _tokenAddress, + uint256 _tokenId, + bytes32 _roleId + ) external view returns (address recipient_) { + if ( + _isTokenDeposited(_tokenAddress, _tokenId) && + roles[_tokenAddress][_tokenId][_roleId].expirationDate > block.timestamp + ) { + return roles[_tokenAddress][_tokenId][_roleId].recipient; + } + return address(0); + } + + function roleData( + address _tokenAddress, + uint256 _tokenId, + bytes32 _roleId + ) external view returns (bytes memory data_) { + if (!_isTokenDeposited(_tokenAddress, _tokenId)) { + return ''; + } + return roles[_tokenAddress][_tokenId][_roleId].data; + } + + function roleExpirationDate( + address _tokenAddress, + uint256 _tokenId, + bytes32 _roleId + ) external view returns (uint64 expirationDate_) { + if (!_isTokenDeposited(_tokenAddress, _tokenId)) { + return 0; + } + return roles[_tokenAddress][_tokenId][_roleId].expirationDate; + } + + function isRoleRevocable( + address _tokenAddress, + uint256 _tokenId, + bytes32 _roleId + ) external view returns (bool revocable_) { + if (!_isTokenDeposited(_tokenAddress, _tokenId)) { + return false; + } + return roles[_tokenAddress][_tokenId][_roleId].revocable; + } + + function isRoleApprovedForAll(address _tokenAddress, address _owner, address _operator) public view returns (bool) { + return tokenApprovals[_owner][_tokenAddress][_operator]; + } + + /** ERC-7432 Vault Extension Functions **/ + + function withdraw(address _tokenAddress, uint256 _tokenId) external override { + address originalOwner = originalOwners[_tokenAddress][_tokenId]; + + require(_isWithdrawable(_tokenAddress, _tokenId), 'NftRolesRegistryVault: NFT is not withdrawable'); + + require( + originalOwner == msg.sender || isRoleApprovedForAll(_tokenAddress, originalOwner, msg.sender), + 'NftRolesRegistryVault: sender must be owner or approved' + ); + + delete originalOwners[_tokenAddress][_tokenId]; + IERC721(_tokenAddress).transferFrom(address(this), originalOwner, _tokenId); + emit Withdraw(originalOwner, _tokenAddress, _tokenId); + } + + /** ERC-165 Functions **/ + + function supportsInterface(bytes4 interfaceId) external view virtual override returns (bool) { + return interfaceId == type(IERC7432).interfaceId || interfaceId == type(IERC7432VaultExtension).interfaceId; + } + + /** Internal Functions **/ + + /// @notice Updates originalOwner, validates the sender and deposits NFT (if not deposited yet). + /// @param _tokenAddress The token address. + /// @param _tokenId The token identifier. + /// @return originalOwner_ The original owner of the NFT. + function _depositNft(address _tokenAddress, uint256 _tokenId) internal returns (address originalOwner_) { + address _currentOwner = IERC721(_tokenAddress).ownerOf(_tokenId); + + if (_currentOwner == address(this)) { + // if the NFT is already on the contract, check if sender is approved or original owner + originalOwner_ = originalOwners[_tokenAddress][_tokenId]; + require( + originalOwner_ == msg.sender || isRoleApprovedForAll(_tokenAddress, originalOwner_, msg.sender), + 'NftRolesRegistryVault: sender must be owner or approved' + ); + } else { + // if NFT is not in the contract, deposit it and store the original owner + require( + _currentOwner == msg.sender || isRoleApprovedForAll(_tokenAddress, _currentOwner, msg.sender), + 'NftRolesRegistryVault: sender must be owner or approved' + ); + IERC721(_tokenAddress).transferFrom(_currentOwner, address(this), _tokenId); + originalOwners[_tokenAddress][_tokenId] = _currentOwner; + originalOwner_ = _currentOwner; + } + } + + /// @notice Returns the account approved to call the revokeRole function. Reverts otherwise. + /// @param _tokenAddress The token address. + /// @param _tokenId The token identifier. + /// @param _recipient The user that received the role. + /// @return caller_ The approved account. + function _getApprovedCaller( + address _tokenAddress, + uint256 _tokenId, + address _recipient + ) internal view returns (address caller_) { + if (msg.sender == _recipient || isRoleApprovedForAll(_tokenAddress, _recipient, msg.sender)) { + return _recipient; + } + address originalOwner = originalOwners[_tokenAddress][_tokenId]; + if (msg.sender == originalOwner || isRoleApprovedForAll(_tokenAddress, originalOwner, msg.sender)) { + return originalOwner; + } + revert('NftRolesRegistryVault: role does not exist or sender is not approved'); + } + + /// @notice Check if an NFT is withdrawable. + /// @param _tokenAddress The token address. + /// @param _tokenId The token identifier. + /// @return True if the NFT is withdrawable. + function _isWithdrawable(address _tokenAddress, uint256 _tokenId) internal view returns (bool) { + // todo needs to implement a way to track expiration dates to make sure NFTs are withdrawable + // mocked result + return _isTokenDeposited(_tokenAddress, _tokenId); + } + + /// @notice Checks if the NFT is deposited on this contract. + /// @param _tokenAddress The token address. + /// @param _tokenId The token identifier. + /// @return deposited_ Whether the NFT is deposited or not. + function _isTokenDeposited(address _tokenAddress, uint256 _tokenId) internal view returns (bool) { + return originalOwners[_tokenAddress][_tokenId] != address(0); + } +} diff --git a/test/OriumMarketplace.test.ts b/test/OriumMarketplace.test.ts deleted file mode 100644 index f11d7e5..0000000 --- a/test/OriumMarketplace.test.ts +++ /dev/null @@ -1,803 +0,0 @@ -import { ethers } from 'hardhat' -import { loadFixture } from '@nomicfoundation/hardhat-network-helpers' -import { deployMarketplaceContracts } from './fixtures/OriumMarketplaceFixture' -import { expect } from 'chai' -import { toWei } from '../utils/bignumber' -import { DirectRental, FeeInfo, RentalOffer, RoyaltyInfo } from '../utils/types' -import { AddressZero, DIRECT_RENTAL_NONCE, EMPTY_BYTES, ONE_DAY, ONE_HOUR, THREE_MONTHS } from '../utils/constants' -import { randomBytes } from 'crypto' -import { USER_ROLE } from '../utils/roles' -import { hashDirectRental } from '../utils/hash' -import { IERC7432, MockERC20, MockERC721, OriumMarketplace } from '../typechain-types' - -describe('OriumMarketplace', () => { - let marketplace: OriumMarketplace - let rolesRegistry: IERC7432 - let mockERC721: MockERC721 - let mockERC20: MockERC20 - - // We are disabling this rule because hardhat uses first account as deployer by default, and we are separating deployer and operator - // eslint-disable-next-line @typescript-eslint/no-unused-vars - let deployer: Awaited> - let operator: Awaited> - let notOperator: Awaited> - let creator: Awaited> - let creatorTreasury: Awaited> - let lender: Awaited> - let borrower: Awaited> - - // Values to be used across tests - const maxDeadline = THREE_MONTHS - const feeInfo: FeeInfo = { - feePercentageInWei: toWei('5'), - isCustomFee: true, - } - - before(async function () { - // we are disabling this rule so ; may not be added automatically by prettier at the beginning of the line - // prettier-ignore - [deployer, operator, notOperator, creator, creatorTreasury, lender, borrower] = await ethers.getSigners() - }) - - beforeEach(async () => { - // we are disabling this rule so ; may not be added automatically by prettier at the beginning of the line - // prettier-ignore - [marketplace, rolesRegistry, mockERC721, mockERC20] = await loadFixture(deployMarketplaceContracts) - }) - - describe('Main Functions', async () => { - describe('Rental Functions', async () => { - const duration = ONE_HOUR - const tokenId = 1 - - beforeEach(async () => { - await mockERC721.mint(lender.address, tokenId) - await rolesRegistry - .connect(lender) - .setRoleApprovalForAll(await mockERC721.getAddress(), await marketplace.getAddress(), true) - }) - - describe('Rental Offers', async () => { - let rentalOffer: RentalOffer - - beforeEach(async () => { - await marketplace.connect(operator).setCreator(await mockERC721.getAddress(), creator.address) - - const royaltyInfo: RoyaltyInfo = { - creator: creator.address, - royaltyPercentageInWei: toWei('10'), - treasury: creatorTreasury.address, - } - - await marketplace - .connect(creator) - .setRoyaltyInfo(await mockERC721.getAddress(), royaltyInfo.royaltyPercentageInWei, royaltyInfo.treasury) - - const blockTimestamp = Number((await ethers.provider.getBlock('latest'))?.timestamp) - - rentalOffer = { - nonce: `0x${randomBytes(32).toString('hex')}`, - lender: lender.address, - borrower: AddressZero, - tokenAddress: await mockERC721.getAddress(), - tokenId, - feeTokenAddress: await mockERC20.getAddress(), - feeAmountPerSecond: toWei('0'), - deadline: blockTimestamp + ONE_DAY, - roles: [USER_ROLE], - rolesData: [EMPTY_BYTES], - } - }) - describe('When Rental Offer is not created', async () => { - describe('Create Rental Offer', async () => { - it('Should create a rental offer', async () => { - await expect(marketplace.connect(lender).createRentalOffer(rentalOffer)) - .to.emit(marketplace, 'RentalOfferCreated') - .withArgs( - rentalOffer.nonce, - rentalOffer.tokenAddress, - rentalOffer.tokenId, - rentalOffer.lender, - rentalOffer.borrower, - rentalOffer.feeTokenAddress, - rentalOffer.feeAmountPerSecond, - rentalOffer.deadline, - rentalOffer.roles, - rentalOffer.rolesData, - ) - }) - it('Should NOT create a rental offer if caller is not the lender', async () => { - await expect(marketplace.connect(notOperator).createRentalOffer(rentalOffer)).to.be.revertedWith( - 'OriumMarketplace: only token owner can call this function', - ) - }) - it("Should NOT create a rental offer if lender is not the caller's address", async () => { - rentalOffer.lender = creator.address - await expect(marketplace.connect(lender).createRentalOffer(rentalOffer)).to.be.revertedWith( - 'OriumMarketplace: Sender and Lender mismatch', - ) - }) - it("Should NOT create a rental offer if roles and rolesData don't have the same length", async () => { - rentalOffer.roles = [`0x${randomBytes(32).toString('hex')}`] - rentalOffer.rolesData = [`0x${randomBytes(32).toString('hex')}`, `0x${randomBytes(32).toString('hex')}`] - await expect(marketplace.connect(lender).createRentalOffer(rentalOffer)).to.be.revertedWith( - 'OriumMarketplace: roles and rolesData should have the same length', - ) - }) - it('Should NOT create a rental offer if deadline is greater than maxDeadline', async () => { - rentalOffer.deadline = maxDeadline + 1 - await expect(marketplace.connect(lender).createRentalOffer(rentalOffer)).to.be.revertedWith( - 'OriumMarketplace: Invalid deadline', - ) - }) - it("Should NOT create a rental offer if deadline is less than block's timestamp", async () => { - rentalOffer.deadline = Number((await ethers.provider.getBlock('latest'))?.timestamp) - 1 - await expect(marketplace.connect(lender).createRentalOffer(rentalOffer)).to.be.revertedWith( - 'OriumMarketplace: Invalid deadline', - ) - }) - it('Should NOT create the same rental offer twice', async () => { - await marketplace.connect(lender).createRentalOffer(rentalOffer) - await expect(marketplace.connect(lender).createRentalOffer(rentalOffer)).to.be.revertedWith( - 'OriumMarketplace: nonce already used', - ) - }) - it('Should NOT create a rental offer if roles or rolesData are empty', async () => { - rentalOffer.roles = [] - await expect(marketplace.connect(lender).createRentalOffer(rentalOffer)).to.be.revertedWith( - 'OriumMarketplace: roles should not be empty', - ) - }) - it("Should NOT create a rental offer if nonce is the direct rental's nonce", async () => { - rentalOffer.nonce = DIRECT_RENTAL_NONCE.toString() - await expect(marketplace.connect(lender).createRentalOffer(rentalOffer)).to.be.revertedWith( - 'OriumMarketplace: Nonce cannot be 0', - ) - }) - }) - }) - describe('When Rental Offer is created', async () => { - beforeEach(async () => { - await marketplace.connect(lender).createRentalOffer(rentalOffer) - }) - describe('Accept Rental Offer', async () => { - it('Should accept a public rental offer', async () => { - const blockTimestamp = Number((await ethers.provider.getBlock('latest'))?.timestamp) - const expirationDate = blockTimestamp + duration + 1 - await expect(marketplace.connect(borrower).acceptRentalOffer(rentalOffer, duration)) - .to.emit(marketplace, 'RentalStarted') - .withArgs( - rentalOffer.nonce, - rentalOffer.tokenAddress, - rentalOffer.tokenId, - rentalOffer.lender, - borrower.address, - expirationDate, - ) - }) - it('Should accept a private rental offer', async () => { - rentalOffer.borrower = borrower.address - rentalOffer.nonce = `0x${randomBytes(32).toString('hex')}` - await marketplace.connect(lender).createRentalOffer(rentalOffer) - const blockTimestamp = Number((await ethers.provider.getBlock('latest'))?.timestamp) - const expirationDate = blockTimestamp + duration + 1 - await expect(marketplace.connect(borrower).acceptRentalOffer(rentalOffer, duration)) - .to.emit(marketplace, 'RentalStarted') - .withArgs( - rentalOffer.nonce, - rentalOffer.tokenAddress, - rentalOffer.tokenId, - rentalOffer.lender, - borrower.address, - expirationDate, - ) - }) - it('Should accept a rental offer if token has a different registry', async () => { - await marketplace - .connect(operator) - .setRolesRegistry(await mockERC721.getAddress(), await rolesRegistry.getAddress()) - const blockTimestamp = Number((await ethers.provider.getBlock('latest'))?.timestamp) - const expirationDate = blockTimestamp + duration + 1 - await expect(marketplace.connect(borrower).acceptRentalOffer(rentalOffer, duration)) - .to.emit(marketplace, 'RentalStarted') - .withArgs( - rentalOffer.nonce, - rentalOffer.tokenAddress, - rentalOffer.tokenId, - rentalOffer.lender, - borrower.address, - expirationDate, - ) - }) - it('Should accept a rental offer more than once', async () => { - const rentalExpirationDate1 = Number((await ethers.provider.getBlock('latest'))?.timestamp) + duration + 1 - - await expect(marketplace.connect(borrower).acceptRentalOffer(rentalOffer, duration)) - .to.emit(marketplace, 'RentalStarted') - .withArgs( - rentalOffer.nonce, - rentalOffer.tokenAddress, - rentalOffer.tokenId, - rentalOffer.lender, - borrower.address, - rentalExpirationDate1, - ) - - await ethers.provider.send('evm_increaseTime', [duration + 1]) - - await expect(marketplace.connect(borrower).acceptRentalOffer(rentalOffer, duration)) - .to.emit(marketplace, 'RentalStarted') - .withArgs( - rentalOffer.nonce, - rentalOffer.tokenAddress, - rentalOffer.tokenId, - rentalOffer.lender, - borrower.address, - rentalExpirationDate1 + duration + 1, - ) - }) - it('Should accept a rental offer by anyone if borrower is the zero address', async () => { - rentalOffer.borrower = ethers.ZeroAddress - rentalOffer.nonce = `0x${randomBytes(32).toString('hex')}` - await marketplace.connect(lender).createRentalOffer(rentalOffer) - - const blockTimestamp = Number((await ethers.provider.getBlock('latest'))?.timestamp) - await expect(marketplace.connect(notOperator).acceptRentalOffer(rentalOffer, duration)) - .to.emit(marketplace, 'RentalStarted') - .withArgs( - rentalOffer.nonce, - rentalOffer.tokenAddress, - rentalOffer.tokenId, - rentalOffer.lender, - notOperator.address, - blockTimestamp + duration + 1, - ) - }) - it('Should NOT accept a rental offer if caller is not the borrower', async () => { - rentalOffer.nonce = `0x${randomBytes(32).toString('hex')}` - rentalOffer.borrower = borrower.address - await marketplace.connect(lender).createRentalOffer(rentalOffer) - await expect( - marketplace.connect(notOperator).acceptRentalOffer(rentalOffer, duration), - ).to.be.revertedWith('OriumMarketplace: Sender is not allowed to rent this NFT') - }) - it('Should NOT accept a rental offer if offer is expired', async () => { - // move foward in time to expire the offer - const blockTimestamp = Number((await ethers.provider.getBlock('latest'))?.timestamp) - const timeToMove = rentalOffer.deadline - blockTimestamp + 1 - await ethers.provider.send('evm_increaseTime', [timeToMove]) - - await expect(marketplace.connect(borrower).acceptRentalOffer(rentalOffer, duration)).to.be.revertedWith( - 'OriumMarketplace: expiration date is greater than offer deadline', - ) - }) - it('Should NOT accept a rental offer if offer is not created', async () => { - rentalOffer.nonce = `0x${randomBytes(32).toString('hex')}` - await expect(marketplace.connect(borrower).acceptRentalOffer(rentalOffer, duration)).to.be.revertedWith( - 'OriumMarketplace: Offer not created', - ) - }) - it('Should NOT accept a rental offer if expiration date is higher than offer deadline', async () => { - const maxDuration = - rentalOffer.deadline - Number((await ethers.provider.getBlock('latest'))?.timestamp) + 1 - await expect( - marketplace.connect(borrower).acceptRentalOffer(rentalOffer, maxDuration), - ).to.be.revertedWith('OriumMarketplace: expiration date is greater than offer deadline') - }) - it('Should NOT accept a rental offer if expiration date is less than block timestamp', async () => { - await expect(marketplace.connect(borrower).acceptRentalOffer(rentalOffer, 0)).to.be.revertedWith( - 'RolesRegistry: expiration date must be in the future', - ) - }) - describe('Fees', async function () { - const feeAmountPerSecond = toWei('1') - const feeAmount = feeAmountPerSecond * BigInt(duration) - - beforeEach(async () => { - rentalOffer.feeAmountPerSecond = feeAmountPerSecond - rentalOffer.nonce = `0x${randomBytes(32).toString('hex')}` - await marketplace.connect(lender).createRentalOffer(rentalOffer) - await mockERC20.mint(borrower.address, feeAmount * BigInt(2)) - await mockERC20.connect(borrower).approve(await marketplace.getAddress(), feeAmount * BigInt(2)) - }) - - it('Should accept a rental offer with fee', async () => { - const blockTimestamp = Number((await ethers.provider.getBlock('latest'))?.timestamp) - const expirationDate = blockTimestamp + duration + 1 - await expect(marketplace.connect(borrower).acceptRentalOffer(rentalOffer, duration)) - .to.emit(marketplace, 'RentalStarted') - .withArgs( - rentalOffer.nonce, - rentalOffer.tokenAddress, - rentalOffer.tokenId, - rentalOffer.lender, - borrower.address, - expirationDate, - ) - .to.emit(mockERC20, 'Transfer') - }) - it('Should accept a rental offer if marketplace fee is 0', async () => { - await marketplace - .connect(operator) - .setMarketplaceFeeForCollection(await mockERC721.getAddress(), 0, true) - await expect(marketplace.connect(borrower).acceptRentalOffer(rentalOffer, duration)).to.emit( - marketplace, - 'RentalStarted', - ) - }) - it('Should accept a rental offer if royalty fee is 0', async () => { - await marketplace - .connect(creator) - .setRoyaltyInfo(await mockERC721.getAddress(), '0', creatorTreasury.address) - await expect(marketplace.connect(borrower).acceptRentalOffer(rentalOffer, duration)).to.emit( - marketplace, - 'RentalStarted', - ) - }) - it('Should NOT accept a rental offer if marketplace fee transfer fails', async () => { - await mockERC20.transferReverts(true, 0) - await expect(marketplace.connect(borrower).acceptRentalOffer(rentalOffer, duration)).to.be.revertedWith( - 'OriumMarketplace: Transfer failed', - ) - }) - it('Should NOT accept a rental offer if royalty fee transfer fails', async () => { - await mockERC20.transferReverts(true, 1) - await expect(marketplace.connect(borrower).acceptRentalOffer(rentalOffer, duration)).to.be.revertedWith( - 'OriumMarketplace: Transfer failed', - ) - }) - it('Should NOT accept a rental offer if lender fee transfer fails', async () => { - await mockERC20.transferReverts(true, 2) - await expect(marketplace.connect(borrower).acceptRentalOffer(rentalOffer, duration)).to.be.revertedWith( - 'OriumMarketplace: Transfer failed', - ) - }) - it('Should NOT accept a rental offer twice', async () => { - await marketplace.connect(borrower).acceptRentalOffer(rentalOffer, duration) - await expect(marketplace.connect(borrower).acceptRentalOffer(rentalOffer, duration)).to.be.revertedWith( - 'OriumMarketplace: Rental already started', - ) - }) - }) - }) - describe('Cancel Rental Offer', async () => { - it('Should cancel a rental offer', async () => { - await expect(marketplace.connect(lender).cancelRentalOffer(rentalOffer.nonce)) - .to.emit(marketplace, 'RentalOfferCancelled') - .withArgs(rentalOffer.nonce, lender.address) - }) - it('Should NOT cancel a rental offer if nonce not used yet by caller', async () => { - await expect(marketplace.connect(notOperator).cancelRentalOffer(rentalOffer.nonce)).to.be.revertedWith( - 'OriumMarketplace: Nonce expired or not used yet', - ) - }) - it("Should NOT cancel a rental offer after deadline's expiration", async () => { - // move foward in time to expire the offer - const blockTimestamp = Number((await ethers.provider.getBlock('latest'))?.timestamp) - const timeToMove = rentalOffer.deadline - blockTimestamp + 1 - await ethers.provider.send('evm_increaseTime', [timeToMove]) - - await expect(marketplace.connect(lender).cancelRentalOffer(rentalOffer.nonce)).to.be.revertedWith( - 'OriumMarketplace: Nonce expired or not used yet', - ) - }) - }) - }) - describe('When Rental Offer is accepted', async () => { - beforeEach(async () => { - await marketplace.connect(lender).createRentalOffer(rentalOffer) - await marketplace.connect(borrower).acceptRentalOffer(rentalOffer, duration) - await rolesRegistry - .connect(borrower) - .setRoleApprovalForAll(await mockERC721.getAddress(), await marketplace.getAddress(), true) - }) - describe('End Rental', async () => { - it('Should end a rental by the borrower', async () => { - await expect(marketplace.connect(borrower).endRental(rentalOffer)) - .to.emit(marketplace, 'RentalEnded') - .withArgs( - rentalOffer.nonce, - rentalOffer.tokenAddress, - rentalOffer.tokenId, - rentalOffer.lender, - borrower.address, - ) - }) - it('Should NOT end a rental by the lender', async () => { - await expect(marketplace.connect(lender).endRental(rentalOffer)).to.be.revertedWith( - 'OriumMarketplace: Only borrower can end a rental', - ) - }) - it('Should NOT end a rental if caller is not the borrower', async () => { - await expect(marketplace.connect(notOperator).endRental(rentalOffer)).to.be.revertedWith( - 'OriumMarketplace: Only borrower can end a rental', - ) - }) - it('Should NOT end a rental if rental is not started', async () => { - await expect( - marketplace - .connect(borrower) - .endRental({ ...rentalOffer, nonce: `0x${randomBytes(32).toString('hex')}` }), - ).to.be.revertedWith('OriumMarketplace: Offer not created') - }) - it('Should NOT end a rental if rental is expired', async () => { - // move foward in time to expire the offer - const blockTimestamp = Number((await ethers.provider.getBlock('latest'))?.timestamp) - const timeToMove = rentalOffer.deadline - blockTimestamp + 1 - await ethers.provider.send('evm_increaseTime', [timeToMove]) - - await expect(marketplace.connect(borrower).endRental(rentalOffer)).to.be.revertedWith( - 'OriumMarketplace: Rental Offer expired', - ) - }) - it('Should end a rental if the role was revoked by borrower directly in registry', async () => { - await rolesRegistry - .connect(borrower) - .setRoleApprovalForAll(await mockERC721.getAddress(), borrower.address, true) - await rolesRegistry - .connect(borrower) - .revokeRoleFrom( - rentalOffer.roles[0], - rentalOffer.tokenAddress, - rentalOffer.tokenId, - rentalOffer.lender, - borrower.address, - ) - await expect(marketplace.connect(borrower).endRental(rentalOffer)) - .to.emit(marketplace, 'RentalEnded') - .withArgs( - rentalOffer.nonce, - rentalOffer.tokenAddress, - rentalOffer.tokenId, - rentalOffer.lender, - borrower.address, - ) - }) - it('Should NOT end rental twice', async () => { - await marketplace.connect(borrower).endRental(rentalOffer) - await expect(marketplace.connect(borrower).endRental(rentalOffer)).to.be.revertedWith( - 'OriumMarketplace: Rental ended', - ) - }) - }) - }) - }) - - describe('Direct Rentals', async function () { - let directRental: DirectRental - let directRentalHash: string - beforeEach(async () => { - directRental = { - tokenAddress: await mockERC721.getAddress(), - tokenId, - lender: lender.address, - borrower: borrower.address, - duration, - roles: [USER_ROLE], - rolesData: [EMPTY_BYTES], - } - directRentalHash = hashDirectRental(directRental) - }) - describe('Create Direct Rental', async () => { - it("Should create a direct rental if caller is the token's owner", async () => { - await expect(marketplace.connect(lender).createDirectRental(directRental)) - .to.emit(marketplace, 'DirectRentalStarted') - .withArgs( - directRentalHash, - await mockERC721.getAddress(), - tokenId, - lender.address, - borrower.address, - directRental.duration, - directRental.roles, - directRental.rolesData, - ) - }) - it('Should NOT create a direct rental if caller is not the token owner', async () => { - await expect(marketplace.connect(notOperator).createDirectRental(directRental)).to.be.revertedWith( - 'OriumMarketplace: only token owner can call this function', - ) - }) - it("Should NOT create a direct rental if lender address is not the token's owner", async () => { - directRental.lender = creator.address - await expect(marketplace.connect(lender).createDirectRental(directRental)).to.be.revertedWith( - 'OriumMarketplace: Sender and Lender mismatch', - ) - }) - it("Should NOT create a direct rental if roles and rolesData don't have the same length", async () => { - directRental.roles = [`0x${randomBytes(32).toString('hex')}`] - directRental.rolesData = [`0x${randomBytes(32).toString('hex')}`, `0x${randomBytes(32).toString('hex')}`] - await expect(marketplace.connect(lender).createDirectRental(directRental)).to.be.revertedWith( - 'OriumMarketplace: roles and rolesData should have the same length', - ) - }) - it('Should NOT create a direct rental if roles or rolesData are empty', async () => { - directRental.roles = [] - await expect(marketplace.connect(lender).createDirectRental(directRental)).to.be.revertedWith( - 'OriumMarketplace: roles should not be empty', - ) - }) - it('Should NOT create a direct rental if expiration date is greater than maxDeadline', async () => { - directRental.duration += maxDeadline - await expect(marketplace.connect(lender).createDirectRental(directRental)).to.be.revertedWith( - 'OriumMarketplace: Duration is greater than max deadline', - ) - }) - }) - describe('Cancel Direct Rental', async () => { - beforeEach(async () => { - await marketplace.connect(lender).createDirectRental(directRental) - }) - it('Should cancel a direct rental if caller is the lender', async () => { - await expect(marketplace.connect(lender).cancelDirectRental(directRental)) - .to.emit(marketplace, 'DirectRentalEnded') - .withArgs(directRentalHash, lender.address) - }) - it('Should cancel a direct rental if caller is the borrower', async () => { - await rolesRegistry - .connect(borrower) - .setRoleApprovalForAll(await mockERC721.getAddress(), await marketplace.getAddress(), true) - await expect(marketplace.connect(borrower).cancelDirectRental(directRental)) - .to.emit(marketplace, 'DirectRentalEnded') - .withArgs(directRentalHash, lender.address) - }) - it('Should NOT cancel a direct rental if caller is neither borrower or lender', async () => { - await expect(marketplace.connect(notOperator).cancelDirectRental(directRental)).to.be.revertedWith( - 'OriumMarketplace: Sender and Lender/Borrower mismatch', - ) - }) - it('Should NOT cancel a direct rental twice', async () => { - await marketplace.connect(lender).cancelDirectRental(directRental) - await expect(marketplace.connect(lender).cancelDirectRental(directRental)).to.be.revertedWith( - 'OriumMarketplace: Direct rental expired', - ) - }) - it("Should NOT cancel a direct rental if rental doesn't exist", async () => { - await expect( - marketplace.connect(lender).cancelDirectRental({ ...directRental, duration: 0 }), - ).to.be.revertedWith('OriumMarketplace: Direct rental not created') - }) - }) - }) - }) - describe('Core Functions', async () => { - describe('Initialize', async () => { - it("Should NOT initialize the contract if it's already initialized", async () => { - await expect(marketplace.initialize(operator.address, ethers.ZeroAddress, 0)).to.be.revertedWith( - 'Initializable: contract is already initialized', - ) - }) - }) - describe('Pausable', async () => { - describe('Pause', async () => { - it('Should pause the contract', async () => { - await marketplace.connect(operator).pause() - expect(await marketplace.paused()).to.be.true - }) - - it('Should NOT pause the contract if caller is not the operator', async () => { - await expect(marketplace.connect(notOperator).pause()).to.be.revertedWith( - 'Ownable: caller is not the owner', - ) - }) - }) - describe('Unpause', async () => { - it('Should unpause the contract', async () => { - await marketplace.connect(operator).pause() - await marketplace.connect(operator).unpause() - expect(await marketplace.paused()).to.be.false - }) - - it('Should NOT unpause the contract if caller is not the operator', async () => { - await marketplace.connect(operator).pause() - await expect(marketplace.connect(notOperator).unpause()).to.be.revertedWith( - 'Ownable: caller is not the owner', - ) - }) - }) - }) - describe('Marketplace Fee', async () => { - it('Should set the marketplace for a collection', async () => { - await expect( - marketplace - .connect(operator) - .setMarketplaceFeeForCollection( - await mockERC721.getAddress(), - feeInfo.feePercentageInWei, - feeInfo.isCustomFee, - ), - ) - .to.emit(marketplace, 'MarketplaceFeeSet') - .withArgs(await mockERC721.getAddress(), feeInfo.feePercentageInWei, feeInfo.isCustomFee) - expect(await marketplace.feeInfo(await mockERC721.getAddress())).to.be.deep.equal([ - feeInfo.feePercentageInWei, - feeInfo.isCustomFee, - ]) - expect(await marketplace.marketplaceFeeOf(await mockERC721.getAddress())).to.be.equal( - feeInfo.feePercentageInWei, - ) - }) - it('Should NOT set the marketplace fee if caller is not the operator', async () => { - await expect( - marketplace - .connect(notOperator) - .setMarketplaceFeeForCollection( - await mockERC721.getAddress(), - feeInfo.feePercentageInWei, - feeInfo.isCustomFee, - ), - ).to.be.revertedWith('Ownable: caller is not the owner') - }) - it("Should NOT set the marketplace fee if marketplace fee + creator royalty it's greater than 100%", async () => { - await marketplace.connect(operator).setCreator(await mockERC721.getAddress(), creator.address) - - const royaltyInfo: RoyaltyInfo = { - creator: creator.address, - royaltyPercentageInWei: toWei('10'), - treasury: creatorTreasury.address, - } - - await marketplace - .connect(creator) - .setRoyaltyInfo(await mockERC721.getAddress(), royaltyInfo.royaltyPercentageInWei, royaltyInfo.treasury) - - const feeInfo: FeeInfo = { - feePercentageInWei: toWei('95'), - isCustomFee: true, - } - await expect( - marketplace - .connect(operator) - .setMarketplaceFeeForCollection( - await mockERC721.getAddress(), - feeInfo.feePercentageInWei, - feeInfo.isCustomFee, - ), - ).to.be.revertedWith('OriumMarketplace: Royalty percentage + marketplace fee cannot be greater than 100%') - }) - }) - describe('Creator Royalties', async () => { - describe('Operator', async () => { - it('Should set the creator royalties for a collection', async () => { - const royaltyInfo: RoyaltyInfo = { - creator: creator.address, - royaltyPercentageInWei: toWei('0'), - treasury: ethers.ZeroAddress, - } - - await expect(marketplace.connect(operator).setCreator(await mockERC721.getAddress(), creator.address)) - .to.emit(marketplace, 'CreatorRoyaltySet') - .withArgs( - await mockERC721.getAddress(), - creator.address, - royaltyInfo.royaltyPercentageInWei, - royaltyInfo.treasury, - ) - - expect(await marketplace.royaltyInfo(await mockERC721.getAddress())).to.be.deep.equal([ - royaltyInfo.creator, - royaltyInfo.royaltyPercentageInWei, - royaltyInfo.treasury, - ]) - }) - it('Should NOT set the creator royalties if caller is not the operator', async () => { - await expect( - marketplace.connect(notOperator).setCreator(await mockERC721.getAddress(), creator.address), - ).to.be.revertedWith('Ownable: caller is not the owner') - }) - }) - - describe('Creator', async () => { - beforeEach(async () => { - await marketplace.connect(operator).setCreator(await mockERC721.getAddress(), creator.address) - }) - it("Should update the creator royalties for a collection if it's already set", async () => { - const royaltyInfo: RoyaltyInfo = { - creator: creator.address, - royaltyPercentageInWei: toWei('0'), - treasury: creatorTreasury.address, - } - - await expect( - marketplace - .connect(creator) - .setRoyaltyInfo( - await mockERC721.getAddress(), - royaltyInfo.royaltyPercentageInWei, - royaltyInfo.treasury, - ), - ) - .to.emit(marketplace, 'CreatorRoyaltySet') - .withArgs( - await mockERC721.getAddress(), - creator.address, - royaltyInfo.royaltyPercentageInWei, - royaltyInfo.treasury, - ) - }) - it('Should NOT update the creator royalties for a collection if caller is not the creator', async () => { - const royaltyInfo: RoyaltyInfo = { - creator: creator.address, - royaltyPercentageInWei: toWei('0'), - treasury: creatorTreasury.address, - } - - await expect( - marketplace - .connect(notOperator) - .setRoyaltyInfo( - await mockERC721.getAddress(), - royaltyInfo.royaltyPercentageInWei, - royaltyInfo.treasury, - ), - ).to.be.revertedWith('OriumMarketplace: Only creator can set royalty info') - }) - it("Should NOT update the creator royalties for a collection if creator's royalty percentage + marketplace fee is greater than 100%", async () => { - const royaltyInfo: RoyaltyInfo = { - creator: creator.address, - royaltyPercentageInWei: toWei('99'), - treasury: creatorTreasury.address, - } - - await expect( - marketplace - .connect(creator) - .setRoyaltyInfo( - await mockERC721.getAddress(), - royaltyInfo.royaltyPercentageInWei, - royaltyInfo.treasury, - ), - ).to.be.revertedWith('OriumMarketplace: Royalty percentage + marketplace fee cannot be greater than 100%') - }) - }) - }) - describe('Max Deadline', async () => { - it('Should set the max deadline by operator', async () => { - await marketplace.connect(operator).setMaxDeadline(maxDeadline) - expect(await marketplace.maxDeadline()).to.be.equal(maxDeadline) - }) - it('Should NOT set the max deadline if caller is not the operator', async () => { - await expect(marketplace.connect(notOperator).setMaxDeadline(maxDeadline)).to.be.revertedWith( - 'Ownable: caller is not the owner', - ) - }) - it('Should NOT set the max deadline 0', async () => { - await expect(marketplace.connect(operator).setMaxDeadline(0)).to.be.revertedWith( - 'OriumMarketplace: Max deadline should be greater than 0', - ) - }) - }) - - describe('Roles Registry', async () => { - it('Should set the roles registry for a collection', async () => { - await expect( - marketplace - .connect(operator) - .setRolesRegistry(await mockERC721.getAddress(), await rolesRegistry.getAddress()), - ) - .to.emit(marketplace, 'RolesRegistrySet') - .withArgs(await mockERC721.getAddress(), await rolesRegistry.getAddress()) - }) - it('Should NOT set the roles registry if caller is not the operator', async () => { - await expect( - marketplace - .connect(notOperator) - .setRolesRegistry(await mockERC721.getAddress(), await rolesRegistry.getAddress()), - ).to.be.revertedWith('Ownable: caller is not the owner') - }) - }) - - describe('Default Roles Registry', async () => { - it('Should set the default roles registry for a collection', async () => { - await expect(marketplace.connect(operator).setDefaultRolesRegistry(await rolesRegistry.getAddress())).to.not - .be.reverted - }) - it('Should NOT set the default roles registry if caller is not the operator', async () => { - await expect( - marketplace.connect(notOperator).setDefaultRolesRegistry(await rolesRegistry.getAddress()), - ).to.be.revertedWith('Ownable: caller is not the owner') - }) - }) - }) - }) -}) diff --git a/test/OriumNftMarketplace.test.ts b/test/OriumNftMarketplace.test.ts index 7491952..6b9149d 100644 --- a/test/OriumNftMarketplace.test.ts +++ b/test/OriumNftMarketplace.test.ts @@ -1,9 +1,10 @@ +/* eslint-disable no-unexpected-multiline */ import { ethers } from 'hardhat' import { loadFixture, time } from '@nomicfoundation/hardhat-network-helpers' import { expect } from 'chai' import { toWei } from '../utils/bignumber' -import { RentalOffer } from '../utils/types' -import { AddressZero, EMPTY_BYTES, ONE_DAY, THREE_MONTHS } from '../utils/constants' +import { RentalOffer, RoyaltyInfo } from '../utils/types' +import { AddressZero, EMPTY_BYTES, ONE_DAY, ONE_HOUR, THREE_MONTHS } from '../utils/constants' import { randomBytes } from 'crypto' import { USER_ROLE } from '../utils/roles' import { IERC7432, MockERC20, MockERC721, OriumMarketplaceRoyalties, OriumNftMarketplace } from '../typechain-types' @@ -22,16 +23,19 @@ describe('OriumNftMarketplace', () => { let operator: Awaited> let notOperator: Awaited> let creator: Awaited> + let creatorTreasury: Awaited> let lender: Awaited> + let borrower: Awaited> // Values to be used across tests const maxDeadline = THREE_MONTHS const tokenId = 1 + const duration = ONE_HOUR before(async function () { // we are disabling this rule so ; may not be added automatically by prettier at the beginning of the line // prettier-ignore - [deployer, operator, notOperator, creator, lender] = await ethers.getSigners() + [deployer, operator, notOperator, creator, creatorTreasury, lender, borrower] = await ethers.getSigners() }) beforeEach(async () => { @@ -59,7 +63,26 @@ describe('OriumNftMarketplace', () => { let rentalOffer: RentalOffer beforeEach(async () => { - const blockTimestamp = Number((await ethers.provider.getBlock('latest'))?.timestamp) + await marketplaceRoyalties + .connect(operator) + .setRoyaltyInfo(creator.address, await mockERC721.getAddress(), 0, AddressZero) + + const royaltyInfo: RoyaltyInfo = { + creator: creator.address, + royaltyPercentageInWei: toWei('10'), + treasury: creatorTreasury.address, + } + + await marketplaceRoyalties + .connect(creator) + .setRoyaltyInfo( + creator.address, + await mockERC721.getAddress(), + royaltyInfo.royaltyPercentageInWei, + royaltyInfo.treasury, + ) + + const blockTimestamp = Number(await time.latest()) rentalOffer = { nonce: `0x${randomBytes(32).toString('hex')}`, @@ -117,7 +140,7 @@ describe('OriumNftMarketplace', () => { await marketplace.connect(lender).createRentalOffer(rentalOffer) await time.increase(ONE_DAY) rentalOffer.nonce = `0x${randomBytes(32).toString('hex')}` - rentalOffer.deadline = Number((await ethers.provider.getBlock('latest'))?.timestamp) + ONE_DAY + rentalOffer.deadline = Number(await time.latest()) + ONE_DAY await expect(marketplace.connect(lender).createRentalOffer(rentalOffer)) .to.emit(marketplace, 'RentalOfferCreated') .withArgs( @@ -159,7 +182,7 @@ describe('OriumNftMarketplace', () => { ) }) it("Should NOT create a rental offer if deadline is less than block's timestamp", async () => { - rentalOffer.deadline = Number((await ethers.provider.getBlock('latest'))?.timestamp) - 1 + rentalOffer.deadline = Number(await time.latest()) - 1 await expect(marketplace.connect(lender).createRentalOffer(rentalOffer)).to.be.revertedWith( 'OriumNftMarketplace: Invalid deadline', ) @@ -193,7 +216,7 @@ describe('OriumNftMarketplace', () => { .connect(operator) .setTrustedFeeTokenForToken([await mockERC721.getAddress()], [await mockERC20.getAddress()], [false]) await expect(marketplace.connect(lender).createRentalOffer(rentalOffer)).to.be.revertedWith( - 'OriumSftMarketplace: tokenAddress is not trusted', + 'OriumNftMarketplace: tokenAddress is not trusted', ) }) it('Should NOT create a rental offer if deadline is less than minDuration', async () => { @@ -211,6 +234,213 @@ describe('OriumNftMarketplace', () => { }) }) }) + + describe('When Rental Offer is created', async () => { + let totalFeeAmount: bigint + beforeEach(async () => { + totalFeeAmount = rentalOffer.feeAmountPerSecond * BigInt(duration) + await marketplace.connect(lender).createRentalOffer(rentalOffer) + await mockERC20.mint(borrower.address, totalFeeAmount.toString()) + await mockERC20.connect(borrower).approve(await marketplace.getAddress(), totalFeeAmount.toString()) + await mockERC721.connect(lender).setApprovalForAll(await rolesRegistry.getAddress(), true) + }) + describe('Accept Rental Offer', async () => { + it('Should accept a public rental offer', async () => { + const blockTimestamp = await time.latest() + const expirationDate = Number(blockTimestamp) + duration + 1 + await expect(marketplace.connect(borrower).acceptRentalOffer(rentalOffer, duration)) + .to.emit(marketplace, 'RentalStarted') + .withArgs(rentalOffer.lender, rentalOffer.nonce, borrower.address, expirationDate) + }) + it('Should accept a private rental offer', async () => { + await time.increase(ONE_DAY) + rentalOffer.borrower = borrower.address + rentalOffer.nonce = `0x${randomBytes(32).toString('hex')}` + rentalOffer.deadline = Number(await time.latest()) + ONE_DAY + rentalOffer.feeAmountPerSecond = toWei('0') + await marketplace.connect(lender).createRentalOffer(rentalOffer) + + const blockTimestamp = await time.latest() + const expirationDate = Number(blockTimestamp) + duration + 1 + await expect(marketplace.connect(borrower).acceptRentalOffer(rentalOffer, duration)) + .to.emit(marketplace, 'RentalStarted') + .withArgs(rentalOffer.lender, rentalOffer.nonce, borrower.address, expirationDate) + }) + it('Should accept a rental offer if token has a different registry', async () => { + await marketplaceRoyalties + .connect(operator) + .setRolesRegistry(await mockERC721.getAddress(), await rolesRegistry.getAddress()) + const blockTimestamp = await time.latest() + const expirationDate = Number(blockTimestamp) + duration + 1 + await expect(marketplace.connect(borrower).acceptRentalOffer(rentalOffer, duration)) + .to.emit(marketplace, 'RentalStarted') + .withArgs(rentalOffer.lender, rentalOffer.nonce, borrower.address, expirationDate) + }) + it('Should accept a rental offer more than once', async () => { + const rentalExpirationDate1 = Number(await time.latest()) + duration + 1 + + await expect(marketplace.connect(borrower).acceptRentalOffer(rentalOffer, duration)) + .to.emit(marketplace, 'RentalStarted') + .withArgs(rentalOffer.lender, rentalOffer.nonce, borrower.address, rentalExpirationDate1) + + await ethers.provider.send('evm_increaseTime', [duration + 1]) + await mockERC20.mint(borrower.address, totalFeeAmount.toString()) + await mockERC20.connect(borrower).approve(await marketplace.getAddress(), totalFeeAmount.toString()) + + await expect(marketplace.connect(borrower).acceptRentalOffer(rentalOffer, duration)) + .to.emit(marketplace, 'RentalStarted') + .withArgs(rentalOffer.lender, rentalOffer.nonce, borrower.address, rentalExpirationDate1 + duration + 3) + }) + it('Should accept a rental offer by anyone if borrower is the zero address', async () => { + await time.increase(ONE_DAY) + rentalOffer.borrower = ethers.ZeroAddress + rentalOffer.nonce = `0x${randomBytes(32).toString('hex')}` + rentalOffer.deadline = Number(await time.latest()) + ONE_DAY + await marketplace.connect(lender).createRentalOffer(rentalOffer) + + const blockTimestamp = Number(await time.latest()) + await mockERC20.mint(notOperator.address, totalFeeAmount.toString()) + await mockERC20.connect(notOperator).approve(await marketplace.getAddress(), totalFeeAmount.toString()) + await expect(marketplace.connect(notOperator).acceptRentalOffer(rentalOffer, duration)) + .to.emit(marketplace, 'RentalStarted') + .withArgs(rentalOffer.lender, rentalOffer.nonce, notOperator.address, blockTimestamp + duration + 3) + }) + it('Should accept a rental offer if duration is greater or equal minDuration', async () => { + await time.increase(ONE_DAY) + rentalOffer.minDuration = duration / 2 + rentalOffer.nonce = `0x${randomBytes(32).toString('hex')}` + rentalOffer.deadline = Number(await time.latest()) + ONE_DAY + await marketplace.connect(lender).createRentalOffer(rentalOffer) + await expect(marketplace.connect(borrower).acceptRentalOffer(rentalOffer, duration)).to.emit( + marketplace, + 'RentalStarted', + ) + }) + it('Should accept rental offer if marketplace fee is zero', async () => { + await marketplaceRoyalties + .connect(operator) + .setMarketplaceFeeForCollection(await mockERC721.getAddress(), 0, true) + await expect(marketplace.connect(borrower).acceptRentalOffer(rentalOffer, duration)).to.emit( + marketplace, + 'RentalStarted', + ) + }) + it('Should NOT accept a rental offer if duration is less than minDuration', async () => { + await time.increase(ONE_DAY) + rentalOffer.minDuration = duration + rentalOffer.nonce = `0x${randomBytes(32).toString('hex')}` + rentalOffer.deadline = Number(await time.latest()) + ONE_DAY + await marketplace.connect(lender).createRentalOffer(rentalOffer) + await expect( + marketplace.connect(borrower).acceptRentalOffer(rentalOffer, duration / 2), + ).to.be.revertedWith('OriumNftMarketplace: Duration is less than the offer minimum duration') + }) + it('Should NOT accept a rental offer if contract is paused', async () => { + await marketplace.connect(operator).pause() + await expect(marketplace.connect(borrower).acceptRentalOffer(rentalOffer, duration)).to.be.revertedWith( + 'Pausable: paused', + ) + }) + it('Should NOT accept a rental offer if caller is not the borrower', async () => { + await time.increase(ONE_DAY) + rentalOffer.nonce = `0x${randomBytes(32).toString('hex')}` + rentalOffer.borrower = borrower.address + rentalOffer.deadline = Number(await time.latest()) + ONE_DAY + await marketplace.connect(lender).createRentalOffer(rentalOffer) + await mockERC20.mint(notOperator.address, totalFeeAmount.toString()) + await expect( + marketplace.connect(notOperator).acceptRentalOffer(rentalOffer, duration), + ).to.be.revertedWith('OriumNftMarketplace: Sender is not allowed to rent this NFT') + }) + it('Should NOT accept a rental offer if offer is expired', async () => { + // move foward in time to expire the offer + const blockTimestamp = await time.latest() + const timeToMove = rentalOffer.deadline - Number(blockTimestamp) + 1 + await ethers.provider.send('evm_increaseTime', [timeToMove]) + + await expect(marketplace.connect(borrower).acceptRentalOffer(rentalOffer, duration)).to.be.revertedWith( + 'OriumNftMarketplace: expiration date is greater than offer deadline', + ) + }) + it('Should NOT accept a rental offer if offer is not created', async () => { + rentalOffer.nonce = `0x${randomBytes(32).toString('hex')}` + await expect(marketplace.connect(borrower).acceptRentalOffer(rentalOffer, duration)).to.be.revertedWith( + 'OriumNftMarketplace: Offer not created', + ) + }) + it('Should NOT accept a rental offer if expiration date is higher than offer deadline', async () => { + const maxDuration = rentalOffer.deadline - Number(await time.latest()) + 1 + await expect( + marketplace.connect(borrower).acceptRentalOffer(rentalOffer, maxDuration), + ).to.be.revertedWith('OriumNftMarketplace: expiration date is greater than offer deadline') + }) + describe('Fees', async function () { + const feeAmountPerSecond = toWei('1') + const feeAmount = feeAmountPerSecond * BigInt(duration) + + beforeEach(async () => { + await time.increase(ONE_DAY) + rentalOffer.feeAmountPerSecond = feeAmountPerSecond + rentalOffer.nonce = `0x${randomBytes(32).toString('hex')}` + rentalOffer.deadline = Number(await time.latest()) + ONE_DAY + await marketplace.connect(lender).createRentalOffer(rentalOffer) + await mockERC20.mint(borrower.address, feeAmount * BigInt(2)) + await mockERC20.connect(borrower).approve(await marketplace.getAddress(), feeAmount * BigInt(2)) + }) + + it('Should accept a rental offer with fee', async () => { + const blockTimestamp = await time.latest() + const expirationDate = Number(blockTimestamp) + duration + 1 + await expect(marketplace.connect(borrower).acceptRentalOffer(rentalOffer, duration)) + .to.emit(marketplace, 'RentalStarted') + .withArgs(rentalOffer.lender, rentalOffer.nonce, borrower.address, expirationDate) + .to.emit(mockERC20, 'Transfer') + }) + it('Should accept a rental offer if marketplace fee is 0', async () => { + await marketplaceRoyalties + .connect(operator) + .setMarketplaceFeeForCollection(await mockERC721.getAddress(), 0, true) + await expect(marketplace.connect(borrower).acceptRentalOffer(rentalOffer, duration)).to.emit( + marketplace, + 'RentalStarted', + ) + }) + it('Should accept a rental offer if royalty fee is 0', async () => { + await marketplaceRoyalties + .connect(creator) + .setRoyaltyInfo(creator.address, await mockERC721.getAddress(), '0', creatorTreasury.address) + await expect(marketplace.connect(borrower).acceptRentalOffer(rentalOffer, duration)).to.emit( + marketplace, + 'RentalStarted', + ) + }) + it('Should NOT accept a rental offer if marketplace fee transfer fails', async () => { + await mockERC20.transferReverts(true, 0) + await expect(marketplace.connect(borrower).acceptRentalOffer(rentalOffer, duration)).to.be.revertedWith( + 'OriumNftMarketplace: Transfer failed', + ) + }) + it('Should NOT accept a rental offer if royalty fee transfer fails', async () => { + await mockERC20.transferReverts(true, 1) + await expect(marketplace.connect(borrower).acceptRentalOffer(rentalOffer, duration)).to.be.revertedWith( + 'OriumNftMarketplace: Transfer failed', + ) + }) + it('Should NOT accept a rental offer if lender fee transfer fails', async () => { + await mockERC20.transferReverts(true, 2) + await expect(marketplace.connect(borrower).acceptRentalOffer(rentalOffer, duration)).to.be.revertedWith( + 'OriumNftMarketplace: Transfer failed', + ) + }) + it('Should NOT accept a rental offer twice', async () => { + await marketplace.connect(borrower).acceptRentalOffer(rentalOffer, duration) + await expect(marketplace.connect(borrower).acceptRentalOffer(rentalOffer, duration)).to.be.revertedWith( + 'OriumNftMarketplace: This offer has an ongoing rental', + ) + }) + }) + }) + }) }) }) describe('Core Functions', async () => { diff --git a/test/fixtures/OriumMarketplaceFixture.ts b/test/fixtures/OriumMarketplaceFixture.ts deleted file mode 100644 index d6d5519..0000000 --- a/test/fixtures/OriumMarketplaceFixture.ts +++ /dev/null @@ -1,36 +0,0 @@ -import { ethers, upgrades } from 'hardhat' -import { RolesRegistryAddress, THREE_MONTHS } from '../../utils/constants' -import { IERC7432, MockERC20, MockERC721, OriumMarketplace } from '../../typechain-types' -/** - * @dev deployer, operator needs to be the first accounts in the hardhat ethers.getSigners() - * list respectively. This should be considered to use this fixture in tests - * @returns [marketplace, rolesRegistry, mockERC721, mockERC20] - */ -export async function deployMarketplaceContracts() { - const [, operator] = await ethers.getSigners() - - const rolesRegistry: IERC7432 = await ethers.getContractAt('IERC7432', RolesRegistryAddress) - - const MarketplaceFactory = await ethers.getContractFactory('OriumMarketplace') - const marketplaceProxy = await upgrades.deployProxy(MarketplaceFactory, [ - operator.address, - await rolesRegistry.getAddress(), - THREE_MONTHS, - ]) - await marketplaceProxy.waitForDeployment() - - const marketplace: OriumMarketplace = await ethers.getContractAt( - 'OriumMarketplace', - await marketplaceProxy.getAddress(), - ) - - const MockERC721Factory = await ethers.getContractFactory('MockERC721') - const mockERC721: MockERC721 = await MockERC721Factory.deploy() - await mockERC721.waitForDeployment() - - const MockERC20Factory = await ethers.getContractFactory('MockERC20') - const mockERC20: MockERC20 = await MockERC20Factory.deploy() - await mockERC20.waitForDeployment() - - return [marketplace, rolesRegistry, mockERC721, mockERC20] as const -} diff --git a/test/fixtures/OriumNftMarketplaceFixture.ts b/test/fixtures/OriumNftMarketplaceFixture.ts index ad55468..009b29c 100644 --- a/test/fixtures/OriumNftMarketplaceFixture.ts +++ b/test/fixtures/OriumNftMarketplaceFixture.ts @@ -1,6 +1,12 @@ import { ethers, upgrades } from 'hardhat' -import { AddressZero, RolesRegistryAddress, THREE_MONTHS } from '../../utils/constants' -import { IERC7432, MockERC20, MockERC721, OriumMarketplaceRoyalties, OriumNftMarketplace } from '../../typechain-types' +import { AddressZero, THREE_MONTHS } from '../../utils/constants' +import { + NftRolesRegistryVault, + MockERC20, + MockERC721, + OriumMarketplaceRoyalties, + OriumNftMarketplace, +} from '../../typechain-types' /** * @dev deployer, operator needs to be the first accounts in the hardhat ethers.getSigners() * list respectively. This should be considered to use this fixture in tests @@ -9,7 +15,9 @@ import { IERC7432, MockERC20, MockERC721, OriumMarketplaceRoyalties, OriumNftMar export async function deployNftMarketplaceContracts() { const [, operator] = await ethers.getSigners() - const rolesRegistry: IERC7432 = await ethers.getContractAt('IERC7432', RolesRegistryAddress) + const RolesRegistryFactory = await ethers.getContractFactory('NftRolesRegistryVault') + const rolesRegistry: NftRolesRegistryVault = await RolesRegistryFactory.deploy() + await rolesRegistry.waitForDeployment() const MarketplaceRoyaltiesFactory = await ethers.getContractFactory('OriumMarketplaceRoyalties') const marketplaceRoyaltiesProxy = await upgrades.deployProxy(MarketplaceRoyaltiesFactory, [ @@ -23,12 +31,22 @@ export async function deployNftMarketplaceContracts() { 'OriumMarketplaceRoyalties', await marketplaceRoyaltiesProxy.getAddress(), ) + const LibMarketplaceFactory = await ethers.getContractFactory('LibOriumNftMarketplace') + const libMarketplace = await LibMarketplaceFactory.deploy() + await libMarketplace.waitForDeployment() - const MarketplaceFactory = await ethers.getContractFactory('OriumNftMarketplace') - const marketplaceProxy = await upgrades.deployProxy(MarketplaceFactory, [ - operator.address, - await marketplaceRoyalties.getAddress(), - ]) + const MarketplaceFactory = await ethers.getContractFactory('OriumNftMarketplace', { + libraries: { + LibOriumNftMarketplace: await libMarketplace.getAddress(), + }, + }) + const marketplaceProxy = await upgrades.deployProxy( + MarketplaceFactory, + [operator.address, await marketplaceRoyalties.getAddress()], + { + unsafeAllowLinkedLibraries: true, + }, + ) await marketplaceProxy.waitForDeployment() const marketplace: OriumNftMarketplace = await ethers.getContractAt( diff --git a/utils/types.ts b/utils/types.ts index aea4ff9..2e436b9 100644 --- a/utils/types.ts +++ b/utils/types.ts @@ -34,7 +34,6 @@ export interface DirectRental { } export interface SftRentalOffer extends RentalOffer { - minDuration: number tokenAmount: bigint commitmentId: bigint }