Overview
ETH Balance
0 ETH
Eth Value
$0.00More Info
Private Name Tags
ContractCreator
Latest 1 internal transaction
Advanced mode:
Parent Transaction Hash | Block | From | To | |||
---|---|---|---|---|---|---|
19263960 | 274 days ago | Contract Creation | 0 ETH |
Loading...
Loading
Minimal Proxy Contract for 0xfd586c11ca3d9feb3ab762078db92cd021b0c2ad
Contract Name:
LOTM Renting Protocol Vault Contract
Compiler Version
vyper:0.3.9
Contract Source Code (Vyper language format)
# @version 0.3.9 """ @title LOTM Renting Protocol Vault Contract @author [Zharta](https://zharta.io/) @notice This contract is the vault implementation for the LOTM Renting Protocol. @dev This is the implementation contract for each vault, which is deployed as a minimal proxy (ERC1167) by `Renting.vy` and accepts only calls from it. This contract holds the assets (NFT, payment tokens) for each token, holds the listing and rental states, performs rewards and fee payments and sets the delegation to hot wallets. Delegations are performed by warm.xyz HotWalletProxy. The information regarding listings and rentals was externalized in order to reduce the gas costs while using the protocol. That requires the state to be passed as an argument to each function and validated by matching it's hash against the one stored in the contract. Conversly, changes to the state are hashed and stored, and the resulting state variables returned to the caller (the Renting contract), to either be published as events or returned directly to the user. The structures that hold the state are the `Listing` and the `Rentals`, although not every member is part of the state if is not required to keep the integrity of the contract. """ # Interfaces from vyper.interfaces import ERC20 as IERC20 from vyper.interfaces import ERC721 as IERC721 interface IDelegationRegistry: def getHotWallet(cold_wallet: address) -> address: view def setHotWallet(hot_wallet_address: address, expiration_timestamp: uint256, lock_hot_wallet_address: bool): nonpayable def setExpirationTimestamp(expiration_timestamp: uint256): nonpayable # Structs struct Rental: id: bytes32 # keccak256 of the renter, token_id, start and expiration owner: address renter: address delegate: address token_id: uint256 start: uint256 min_expiration: uint256 expiration: uint256 amount: uint256 protocol_fee: uint256 protocol_wallet: address struct Listing: token_id: uint256 price: uint256 # price per hour, 0 means not listed min_duration: uint256 # min duration in hours max_duration: uint256 # max duration in hours, 0 means unlimited struct VaultState: active_rental: Rental listing: Listing # Global Variables empty_state_hash: immutable(bytes32) owner: public(address) caller: public(address) state: public(bytes32) unclaimed_rewards: public(uint256) unclaimed_protocol_fee: public(uint256) payment_token_addr: public(immutable(address)) nft_contract_addr: public(immutable(address)) delegation_registry_addr: public(immutable(address)) ##### EXTERNAL METHODS - WRITE ##### @payable @external def __init__( _payment_token_addr: address, _nft_contract_addr: address, _delegation_registry_addr: address ): """ @dev Sets up the contract by initializing the payment token, NFT contract, and delegation registry addresses. @param _payment_token_addr The address of the payment token contract. @param _nft_contract_addr The address of the NFT contract. @param _delegation_registry_addr The address of the delegation registry contract. """ payment_token_addr = _payment_token_addr nft_contract_addr = _nft_contract_addr delegation_registry_addr = _delegation_registry_addr empty_state_hash = self._state_hash(empty(VaultState)) @external def initialise(owner: address): """ @notice Initialize a vault with the given owner, enabling it to receive a token @dev Ensures that the vault is not already initialized before setting the owner and caller. @param owner The address of the vault's owner. """ assert not self._is_initialised(), "already initialised" if self.caller != empty(address): assert msg.sender == self.caller, "not caller" else: self.caller = msg.sender self.owner = owner self.state = empty_state_hash @external def deposit(token_id: uint256, price: uint256, min_duration: uint256, max_duration: uint256, delegate: address): """ @notice Deposit an NFT into the vault with specified listing terms and optionaly sets up delegation. @dev Validates the initialization and state of the contract before proceeding with the deposit. @param token_id The id of the NFT to be deposited. @param price The rental price per hour, a value of 0 means unlisted. @param min_duration The minimum rental duration in hours. @param max_duration The maximum rental duration in hours. @param delegate The address to delegate the NFT to while listed. If empty no delegation is done. """ assert self._is_initialised(), "not initialised" assert msg.sender == self.caller, "not caller" assert self.state == empty_state_hash, "invalid state" if max_duration != 0 and min_duration > max_duration: raise "min duration > max duration" self.state = self._state_hash2( Listing( { token_id: token_id, price: price, min_duration: min_duration, max_duration: max_duration } ), empty(Rental) ) # transfer token to this contract IERC721(nft_contract_addr).safeTransferFrom(self.owner, self, token_id, b"") # create delegation if delegate != empty(address): self._delegate_to_wallet(delegate) @external def set_listing(state: VaultState, token_id: uint256, sender: address, price: uint256, min_duration: uint256, max_duration: uint256, delegate: address): """ @notice Set or update the listing for an NFT in the vault. @dev Updates the listing details, including price and duration and optionaly sets up delegation. @param state The current state of the vault. @param token_id The id of the NFT to be listed. @param sender The address setting the listing, expected to be the owner. @param price The rental price per hour, a value of 0 means unlisted. @param min_duration The minimum rental duration in hours. @param max_duration The maximum rental duration in hours, 0 for unlimited. @param delegate The address to delegate the NFT to while unlisted. If empty, no delegation is done, neither any possible current delegation is changed. """ assert self._is_initialised(), "not initialised" assert msg.sender == self.caller, "not caller" assert sender == self.owner, "not owner of vault" assert self.state == self._state_hash(state), "invalid state" assert state.listing.token_id == token_id, "invalid token_id" self._set_listing(token_id, price, min_duration, max_duration, state.active_rental) # create delegation if delegate != empty(address): assert state.active_rental.expiration < block.timestamp, "deleg while rental ongoing" self._delegate_to_wallet(delegate) @external def start_rental(state: VaultState, renter: address, expiration: uint256, delegate: address, protocol_fee: uint256, protocol_wallet: address) -> Rental: """ @notice Start a rental of an NFT from the vault. @dev Handles the transfer of rental amount, updates the rental state, and sets up delegation. Any previous rentals' amounts are consolidated into unclaimed_rewards. @param state The current state of the vault. @param renter The address of the renter. @param expiration The expiration timestamp of the rental. @param delegate The address to delegate the NFT to. @param protocol_fee The protocol fee in bps. @param protocol_wallet The wallet to receive the protocol fee. @return The details of the new rental. """ assert self._is_initialised(), "not initialised" assert msg.sender == self.caller, "not caller" assert state.listing.price > 0, "listing does not exist" assert state.active_rental.expiration < block.timestamp, "active rental ongoing" assert self._is_within_duration_range(state.listing, block.timestamp, expiration), "duration not respected" assert delegate != empty(address), "delegate is zero address" assert self.state == self._state_hash(state), "invalid state" rental_amount: uint256 = self._compute_rental_amount(block.timestamp, expiration, state.listing.price) assert IERC20(payment_token_addr).allowance(renter, self) >= rental_amount, "insufficient allowance" # transfer rental amount from renter to this contract assert IERC20(payment_token_addr).transferFrom(renter, self, rental_amount), "transferFrom failed" # create delegation if IDelegationRegistry(delegation_registry_addr).getHotWallet(self) == delegate: IDelegationRegistry(delegation_registry_addr).setExpirationTimestamp(expiration) else: IDelegationRegistry(delegation_registry_addr).setHotWallet(delegate, expiration, False) # store unclaimed rewards self._consolidate_claims(state) # create rental rental_id: bytes32 = self._compute_rental_id(renter, state.listing.token_id, block.timestamp, expiration) new_rental: Rental = Rental({ id: rental_id, owner: self.owner, renter: renter, delegate: delegate, token_id: state.listing.token_id, start: block.timestamp, min_expiration: block.timestamp + state.listing.min_duration * 3600, expiration: expiration, amount: rental_amount, protocol_fee: protocol_fee, protocol_wallet: protocol_wallet }) self.state = self._state_hash2(state.listing, new_rental) return new_rental @external def close_rental(state: VaultState, sender: address) -> uint256: """ @notice Close an active rental and handle any transfers of fees or refunds. @dev Calculates pro-rata rental amounts and handles transfers of funds and revocation of delegation. The revised rental's amount are consolidated into unclaimed_rewards. @param state The current state of the vault. @param sender The address closing the rental, must be the renter. @return The pro-rata rental amount. """ assert self._is_initialised(), "not initialised" assert msg.sender == self.caller, "not caller" assert self.state == self._state_hash(state), "invalid state" assert state.active_rental.expiration > block.timestamp, "active rental does not exist" assert sender == state.active_rental.renter, "not renter of active rental" # compute amount to send back to renter real_expiration_adjusted: uint256 = block.timestamp if block.timestamp < state.active_rental.min_expiration: real_expiration_adjusted = state.active_rental.min_expiration pro_rata_rental_amount: uint256 = self._compute_real_rental_amount( state.active_rental.expiration - state.active_rental.start, real_expiration_adjusted - state.active_rental.start, state.active_rental.amount ) payback_amount: uint256 = state.active_rental.amount - pro_rata_rental_amount protocol_fee_amount: uint256 = pro_rata_rental_amount * state.active_rental.protocol_fee / 10000 # clear active rental self.state = self._state_hash2(state.listing, empty(Rental)) # set unclaimed rewards self.unclaimed_rewards += pro_rata_rental_amount - protocol_fee_amount # clear unclaimed protocol fee if self.unclaimed_protocol_fee > 0: protocol_fee_amount += self.unclaimed_protocol_fee self.unclaimed_protocol_fee = 0 # revoke delegation IDelegationRegistry(delegation_registry_addr).setHotWallet(empty(address), 0, False) # transfer unused payment to renter assert IERC20(payment_token_addr).transfer(state.active_rental.renter, payback_amount), "transfer failed" # transfer protocol fee to protocol wallet if protocol_fee_amount > 0: assert IERC20(payment_token_addr).transfer(state.active_rental.protocol_wallet, protocol_fee_amount), "transfer failed" return pro_rata_rental_amount @external def claim(state: VaultState, sender: address) -> (Rental, uint256, uint256): """ @notice Claim unclaimed rewards and protocol fees from the vault. @dev Transfers accumulated rewards to the owner and fees to protocol wallet. @param state The current state of the vault. @param sender The address claiming the rewards, must be the owner. @return A tuple with the updated rental, amount of rewards claimed, and amount of protocol fees claimed. """ assert self._is_initialised(), "not initialised" assert msg.sender == self.caller, "not caller" assert sender == self.owner, "not owner of vault" assert self.state == self._state_hash(state), "invalid state" assert self._claimable_rewards(state.active_rental) > 0, "no rewards to claim" # consolidate last renting rewards if existing result_active_rental: Rental = self._consolidate_claims(state) rewards_to_claim: uint256 = self.unclaimed_rewards protocol_fee_to_claim: uint256 = self.unclaimed_protocol_fee # clear unclaimed rewards and protocol fee self.unclaimed_rewards = 0 self.unclaimed_protocol_fee = 0 # transfer reward to nft owner assert IERC20(payment_token_addr).transfer(self.owner, rewards_to_claim), "transfer failed" # transfer protocol fee to protocol wallet if protocol_fee_to_claim > 0: assert IERC20(payment_token_addr).transfer(state.active_rental.protocol_wallet, protocol_fee_to_claim), "transfer failed" return result_active_rental, rewards_to_claim, protocol_fee_to_claim @external def withdraw(state: VaultState, sender: address) -> (uint256, uint256): """ @notice Withdraw an NFT from the vault and claim any unclaimed rewards and protocol fees. @dev Handles the transfer of the NFT back to the owner and clears the vault's state. @param state The current state of the vault. @param sender The address withdrawing the NFT, must be the owner. @return A tuple with the amount of rewards and protocol fees claimed. """ assert self._is_initialised(), "not initialised" assert msg.sender == self.caller, "not caller" assert sender == self.owner, "not owner of vault" assert state.active_rental.expiration < block.timestamp, "active rental ongoing" assert self.state == self._state_hash(state), "invalid state" # consolidate last renting rewards if existing rental: Rental = self._consolidate_claims(state) rewards_to_claim: uint256 = self.unclaimed_rewards protocol_fee_to_claim: uint256 = self.unclaimed_protocol_fee owner: address = self.owner # clear vault and set state to zero to uninitialize self.unclaimed_rewards = 0 self.unclaimed_protocol_fee = 0 self.state = empty(bytes32) self.owner = empty(address) # transfer token to owner IERC721(nft_contract_addr).safeTransferFrom(self, owner, state.listing.token_id, b"") # transfer unclaimed rewards to owner if rewards_to_claim > 0: assert IERC20(payment_token_addr).transfer(owner, rewards_to_claim), "transfer failed" # transfer protocol fee to protocol wallet if protocol_fee_to_claim > 0: assert IERC20(payment_token_addr).transfer(state.active_rental.protocol_wallet, protocol_fee_to_claim), "transfer failed" return rewards_to_claim, protocol_fee_to_claim @external def delegate_to_wallet(state: VaultState, sender: address, delegate: address): """ @notice Delegate the NFT in the vault to a specified wallet. @dev Validates that no rental is ongoing. @param state The current state of the vault. @param sender The address delegating the NFT, must be the owner. @param delegate The address to delegate the NFT to. """ assert self._is_initialised(), "not initialised" assert msg.sender == self.caller, "not caller" assert sender == self.owner, "not owner of vault" assert state.active_rental.expiration < block.timestamp, "active rental ongoing" assert self.state == self._state_hash(state), "invalid state" self._delegate_to_wallet(delegate) ##### INTERNAL METHODS ##### @internal def _consolidate_claims(state: VaultState) -> Rental: if state.active_rental.amount == 0 or state.active_rental.expiration >= block.timestamp: return state.active_rental else: protocol_fee_amount: uint256 = state.active_rental.amount * state.active_rental.protocol_fee / 10000 self.unclaimed_rewards += state.active_rental.amount - protocol_fee_amount self.unclaimed_protocol_fee += protocol_fee_amount new_rental: Rental = Rental({ id: state.active_rental.id, owner: state.active_rental.owner, renter: state.active_rental.renter, delegate: state.active_rental.delegate, token_id: state.active_rental.token_id, start: state.active_rental.start, min_expiration: state.active_rental.min_expiration, expiration: state.active_rental.expiration, amount: 0, protocol_fee: state.active_rental.protocol_fee, protocol_wallet: state.active_rental.protocol_wallet }) self.state = self._state_hash2(state.listing, new_rental) return new_rental @internal def _is_within_duration_range(listing: Listing, start: uint256, expiration: uint256) -> bool: return expiration - start >= listing.min_duration * 3600 and (listing.max_duration == 0 or expiration - start <= listing.max_duration * 3600) @pure @internal def _compute_rental_id(renter: address, token_id: uint256, start: uint256, expiration: uint256) -> bytes32: return keccak256(concat(convert(renter, bytes32), convert(token_id, bytes32), convert(start, bytes32), convert(expiration, bytes32))) @pure @internal def _compute_rental_amount(start: uint256, expiration: uint256, price: uint256) -> uint256: return (expiration - start) * price / 3600 @pure @internal def _compute_real_rental_amount(duration: uint256, real_duration: uint256, rental_amount: uint256) -> uint256: return rental_amount * real_duration / duration @view @internal def _claimable_rewards(active_rental: Rental) -> uint256: if active_rental.expiration < block.timestamp: return self.unclaimed_rewards + active_rental.amount * (10000 - active_rental.protocol_fee) / 10000 else: return self.unclaimed_rewards * (10000 - active_rental.protocol_fee) / 10000 @internal def _delegate_to_wallet(wallet: address): delegation_registry: IDelegationRegistry = IDelegationRegistry(delegation_registry_addr) if delegation_registry.getHotWallet(self) != wallet: delegation_registry.setHotWallet(wallet, max_value(uint256), False) @internal def _set_listing(token_id: uint256, price: uint256, min_duration: uint256, max_duration: uint256, active_rental: Rental): if max_duration != 0 and min_duration > max_duration: raise "min duration > max duration" self.state = self._state_hash2( Listing( { token_id: token_id, price: price, min_duration: min_duration, max_duration: max_duration } ), active_rental ) @pure @internal def _state_hash(state: VaultState) -> bytes32: return self._state_hash2(state.listing, state.active_rental) @pure @internal def _state_hash2(listing: Listing, rental: Rental) -> bytes32: return keccak256( concat( rental.id, convert(rental.owner, bytes32), convert(rental.renter, bytes32), convert(rental.token_id, bytes32), convert(rental.start, bytes32), convert(rental.min_expiration, bytes32), convert(rental.expiration, bytes32), convert(rental.amount, bytes32), convert(rental.protocol_fee, bytes32), convert(rental.protocol_wallet, bytes32), convert(listing.token_id, bytes32), convert(listing.price, bytes32), convert(listing.min_duration, bytes32), convert(listing.max_duration, bytes32), ) ) @view @internal def _is_initialised() -> bool: return self.state != empty(bytes32) ##### EXTERNAL METHODS - VIEW ##### @view @external def claimable_rewards(active_rental: Rental) -> uint256: return self._claimable_rewards(active_rental) @view @external def onERC721Received(_operator: address, _from: address, _tokenId: uint256, _data: Bytes[1024]) -> bytes4: return method_id("onERC721Received(address,address,uint256,bytes)", output_type=bytes4) @view @external def is_initialised() -> bool: return self._is_initialised()
[{"stateMutability":"payable","type":"constructor","inputs":[{"name":"_payment_token_addr","type":"address"},{"name":"_nft_contract_addr","type":"address"},{"name":"_delegation_registry_addr","type":"address"}],"outputs":[]},{"stateMutability":"nonpayable","type":"function","name":"initialise","inputs":[{"name":"owner","type":"address"}],"outputs":[]},{"stateMutability":"nonpayable","type":"function","name":"deposit","inputs":[{"name":"token_id","type":"uint256"},{"name":"price","type":"uint256"},{"name":"min_duration","type":"uint256"},{"name":"max_duration","type":"uint256"},{"name":"delegate","type":"address"}],"outputs":[]},{"stateMutability":"nonpayable","type":"function","name":"set_listing","inputs":[{"name":"state","type":"tuple","components":[{"name":"active_rental","type":"tuple","components":[{"name":"id","type":"bytes32"},{"name":"owner","type":"address"},{"name":"renter","type":"address"},{"name":"delegate","type":"address"},{"name":"token_id","type":"uint256"},{"name":"start","type":"uint256"},{"name":"min_expiration","type":"uint256"},{"name":"expiration","type":"uint256"},{"name":"amount","type":"uint256"},{"name":"protocol_fee","type":"uint256"},{"name":"protocol_wallet","type":"address"}]},{"name":"listing","type":"tuple","components":[{"name":"token_id","type":"uint256"},{"name":"price","type":"uint256"},{"name":"min_duration","type":"uint256"},{"name":"max_duration","type":"uint256"}]}]},{"name":"token_id","type":"uint256"},{"name":"sender","type":"address"},{"name":"price","type":"uint256"},{"name":"min_duration","type":"uint256"},{"name":"max_duration","type":"uint256"},{"name":"delegate","type":"address"}],"outputs":[]},{"stateMutability":"nonpayable","type":"function","name":"start_rental","inputs":[{"name":"state","type":"tuple","components":[{"name":"active_rental","type":"tuple","components":[{"name":"id","type":"bytes32"},{"name":"owner","type":"address"},{"name":"renter","type":"address"},{"name":"delegate","type":"address"},{"name":"token_id","type":"uint256"},{"name":"start","type":"uint256"},{"name":"min_expiration","type":"uint256"},{"name":"expiration","type":"uint256"},{"name":"amount","type":"uint256"},{"name":"protocol_fee","type":"uint256"},{"name":"protocol_wallet","type":"address"}]},{"name":"listing","type":"tuple","components":[{"name":"token_id","type":"uint256"},{"name":"price","type":"uint256"},{"name":"min_duration","type":"uint256"},{"name":"max_duration","type":"uint256"}]}]},{"name":"renter","type":"address"},{"name":"expiration","type":"uint256"},{"name":"delegate","type":"address"},{"name":"protocol_fee","type":"uint256"},{"name":"protocol_wallet","type":"address"}],"outputs":[{"name":"","type":"tuple","components":[{"name":"id","type":"bytes32"},{"name":"owner","type":"address"},{"name":"renter","type":"address"},{"name":"delegate","type":"address"},{"name":"token_id","type":"uint256"},{"name":"start","type":"uint256"},{"name":"min_expiration","type":"uint256"},{"name":"expiration","type":"uint256"},{"name":"amount","type":"uint256"},{"name":"protocol_fee","type":"uint256"},{"name":"protocol_wallet","type":"address"}]}]},{"stateMutability":"nonpayable","type":"function","name":"close_rental","inputs":[{"name":"state","type":"tuple","components":[{"name":"active_rental","type":"tuple","components":[{"name":"id","type":"bytes32"},{"name":"owner","type":"address"},{"name":"renter","type":"address"},{"name":"delegate","type":"address"},{"name":"token_id","type":"uint256"},{"name":"start","type":"uint256"},{"name":"min_expiration","type":"uint256"},{"name":"expiration","type":"uint256"},{"name":"amount","type":"uint256"},{"name":"protocol_fee","type":"uint256"},{"name":"protocol_wallet","type":"address"}]},{"name":"listing","type":"tuple","components":[{"name":"token_id","type":"uint256"},{"name":"price","type":"uint256"},{"name":"min_duration","type":"uint256"},{"name":"max_duration","type":"uint256"}]}]},{"name":"sender","type":"address"}],"outputs":[{"name":"","type":"uint256"}]},{"stateMutability":"nonpayable","type":"function","name":"claim","inputs":[{"name":"state","type":"tuple","components":[{"name":"active_rental","type":"tuple","components":[{"name":"id","type":"bytes32"},{"name":"owner","type":"address"},{"name":"renter","type":"address"},{"name":"delegate","type":"address"},{"name":"token_id","type":"uint256"},{"name":"start","type":"uint256"},{"name":"min_expiration","type":"uint256"},{"name":"expiration","type":"uint256"},{"name":"amount","type":"uint256"},{"name":"protocol_fee","type":"uint256"},{"name":"protocol_wallet","type":"address"}]},{"name":"listing","type":"tuple","components":[{"name":"token_id","type":"uint256"},{"name":"price","type":"uint256"},{"name":"min_duration","type":"uint256"},{"name":"max_duration","type":"uint256"}]}]},{"name":"sender","type":"address"}],"outputs":[{"name":"","type":"tuple","components":[{"name":"id","type":"bytes32"},{"name":"owner","type":"address"},{"name":"renter","type":"address"},{"name":"delegate","type":"address"},{"name":"token_id","type":"uint256"},{"name":"start","type":"uint256"},{"name":"min_expiration","type":"uint256"},{"name":"expiration","type":"uint256"},{"name":"amount","type":"uint256"},{"name":"protocol_fee","type":"uint256"},{"name":"protocol_wallet","type":"address"}]},{"name":"","type":"uint256"},{"name":"","type":"uint256"}]},{"stateMutability":"nonpayable","type":"function","name":"withdraw","inputs":[{"name":"state","type":"tuple","components":[{"name":"active_rental","type":"tuple","components":[{"name":"id","type":"bytes32"},{"name":"owner","type":"address"},{"name":"renter","type":"address"},{"name":"delegate","type":"address"},{"name":"token_id","type":"uint256"},{"name":"start","type":"uint256"},{"name":"min_expiration","type":"uint256"},{"name":"expiration","type":"uint256"},{"name":"amount","type":"uint256"},{"name":"protocol_fee","type":"uint256"},{"name":"protocol_wallet","type":"address"}]},{"name":"listing","type":"tuple","components":[{"name":"token_id","type":"uint256"},{"name":"price","type":"uint256"},{"name":"min_duration","type":"uint256"},{"name":"max_duration","type":"uint256"}]}]},{"name":"sender","type":"address"}],"outputs":[{"name":"","type":"uint256"},{"name":"","type":"uint256"}]},{"stateMutability":"nonpayable","type":"function","name":"delegate_to_wallet","inputs":[{"name":"state","type":"tuple","components":[{"name":"active_rental","type":"tuple","components":[{"name":"id","type":"bytes32"},{"name":"owner","type":"address"},{"name":"renter","type":"address"},{"name":"delegate","type":"address"},{"name":"token_id","type":"uint256"},{"name":"start","type":"uint256"},{"name":"min_expiration","type":"uint256"},{"name":"expiration","type":"uint256"},{"name":"amount","type":"uint256"},{"name":"protocol_fee","type":"uint256"},{"name":"protocol_wallet","type":"address"}]},{"name":"listing","type":"tuple","components":[{"name":"token_id","type":"uint256"},{"name":"price","type":"uint256"},{"name":"min_duration","type":"uint256"},{"name":"max_duration","type":"uint256"}]}]},{"name":"sender","type":"address"},{"name":"delegate","type":"address"}],"outputs":[]},{"stateMutability":"view","type":"function","name":"claimable_rewards","inputs":[{"name":"active_rental","type":"tuple","components":[{"name":"id","type":"bytes32"},{"name":"owner","type":"address"},{"name":"renter","type":"address"},{"name":"delegate","type":"address"},{"name":"token_id","type":"uint256"},{"name":"start","type":"uint256"},{"name":"min_expiration","type":"uint256"},{"name":"expiration","type":"uint256"},{"name":"amount","type":"uint256"},{"name":"protocol_fee","type":"uint256"},{"name":"protocol_wallet","type":"address"}]}],"outputs":[{"name":"","type":"uint256"}]},{"stateMutability":"view","type":"function","name":"onERC721Received","inputs":[{"name":"_operator","type":"address"},{"name":"_from","type":"address"},{"name":"_tokenId","type":"uint256"},{"name":"_data","type":"bytes"}],"outputs":[{"name":"","type":"bytes4"}]},{"stateMutability":"view","type":"function","name":"is_initialised","inputs":[],"outputs":[{"name":"","type":"bool"}]},{"stateMutability":"view","type":"function","name":"owner","inputs":[],"outputs":[{"name":"","type":"address"}]},{"stateMutability":"view","type":"function","name":"caller","inputs":[],"outputs":[{"name":"","type":"address"}]},{"stateMutability":"view","type":"function","name":"state","inputs":[],"outputs":[{"name":"","type":"bytes32"}]},{"stateMutability":"view","type":"function","name":"unclaimed_rewards","inputs":[],"outputs":[{"name":"","type":"uint256"}]},{"stateMutability":"view","type":"function","name":"unclaimed_protocol_fee","inputs":[],"outputs":[{"name":"","type":"uint256"}]},{"stateMutability":"view","type":"function","name":"payment_token_addr","inputs":[],"outputs":[{"name":"","type":"address"}]},{"stateMutability":"view","type":"function","name":"nft_contract_addr","inputs":[],"outputs":[{"name":"","type":"address"}]},{"stateMutability":"view","type":"function","name":"delegation_registry_addr","inputs":[],"outputs":[{"name":"","type":"address"}]}]
Loading...
Loading
Loading...
Loading
Multichain Portfolio | 30 Chains
Chain | Token | Portfolio % | Price | Amount | Value |
---|
Loading...
Loading
[ Download: CSV Export ]
A contract address hosts a smart contract, which is a set of code stored on the blockchain that runs when predetermined conditions are met. Learn more about addresses in our Knowledge Base.