Hooks
Hooks have been added to strategic places of the Lock contract to allow lock managers to modify behaviour by calling a 3rd party contract.
We currently support 7 hooks (as of v12). If your lock uses a previous version, it may have a smaller number of hooks, but you can upgrade it to v12:
- onGrantKeyHook: called when a key is granted
- onKeyExtendHook: called when a key is extended or renewed
- onKeyPurchaseHook: called when a key purchase is triggered
- onKeyCancelHook: called when a key is canceled
- onTokenUriHook: called when the tokenURI is fetched
- onValidKeyHook: called when checking if a user has a valid key
- onKeyTransferHook: called when a key is transfered from an address to another.
onGrantKey Hook
The onGrantKeyHook
allows you to create custom logic that is called each time a key is granted. This could enable use cases to create custom logic when keys are granted outside of a purchase flow.
A KeyGrantedHook should implement the following interface.
interface ILockKeyGrantHook
{
/**
* @notice If the lock owner has registered an implementer then this hook
* is called with every key granted.
* @param tokenId the id of the granted key
* @param from the msg.sender granting the key
* @param recipient the account which will be granted a key
* @param keyManager an additional keyManager for the key
* @param expiration the expiration timestamp of the key
* @dev the lock's address is the `msg.sender` when this function is called
*/
function onKeyGranted(
uint tokenId,
address from,
address recipient,
address keyManager,
uint expiration
) external;
}
onKeyExtend Hook
The onKeyExtendHook
allows you create custom logic when a key is extended or renewed. This could enable use cases where for instance key metadata is updated, maybe you want to update the image when renewals happen. It could enable rewards programs where you increment a reward point total everytime a membership is renewed.
A KeyExtendHook should implement the following interface.
interface ILockKeyExtendHook
{
/**
* @notice This hook every time a key is extended.
* @param tokenId tje id of the key
* @param from the msg.sender making the purchase
* @param newTimestamp the new expiration timestamp after the key extension
* @param prevTimestamp the expiration timestamp of the key before being extended
*/
function onKeyExtend(
uint tokenId,
address from,
uint newTimestamp,
uint prevTimestamp
) external;
}
OnKeyPurchase Hook
The onKeyPurchaseHook
allows you to create custom purchase logic, for instance dynamic pricing, etc.
It contains 2 main functions:
keyPurchasePrice
which is used to determine the purchase price before issuing a transaction,onKeyPurchase
which is called every time a key is sold.
If either of these fail, then the whole purchase will revert.
The ILockKeyPurchaseHook
contract interface describes the parameters of each function (from, recipient, original price, price paid, etc), so the hook can be properly implemented.
For instance, you can find out how to implement discount codes or invite-only purchases using contract extensions we developed in the Key Purchase Hook doc.
OnKeyCancel Hook
Called when a key is cancelled, it can be useful to use with onKeyPurchaseHook
to track lock ownership on a 3rd party contract.
The ILockKeyCancelHook
interface is quite straightforward:
interface ILockKeyCancelHook
{
function onKeyCancel(
address operator, // msg.sender issuing the cancel
address to, // account which had the key canceled
uint256 refund // amount to refund in ETH OR ERC-20
) external;
OnTokenUri Hook
This hook is called every time the tokenURI()
is called. This allows customization of the metadata for each token.
Want each key owner to have his/her own profile pic? Change description based on your own NFT? Just hook a contract compatible with the ILockTokenURIHook
interface and return your own tokenURI.
interface ILockTokenURIHook
{
function tokenURI(
address lockAddress, // the address of the lock
address operator, // the msg.sender issuing the call
address owner, // the owner of the key
uint256 keyId, // the id (tokenId) of the key (if applicable)
uint expirationTimestamp // the key expiration timestamp
) external view returns(string memory);
}
OnValidKey Hook
This hook is called every time the (ERC721) balanceOf
method is called. This allows you to override the balance of the actual contract and define if an account owns a valid key or not based using your own logic.
That way you could whitelist your own NFT holders or DAO members, and provide them access without having them to register. Just use a connector contract compatible with ILockValidKeyHook
that checks if the account is allowed or not, and register it as a hook.
interface ILockValidKeyHook
{
function hasValidKey(
address lockAddress, // the address of the current lock
address keyOwner, // the potential owner of a key
uint256 expirationTimestamp, // the key expiration timestamp
bool isValidKey // the validity in the lock contrat
)
external view
returns (bool);
}
onKeyTransferHook Hook
Called when a key is transfered, it can be useful to use with onKeyPurchaseHook
to track key ownership on a 3rd party contract.
The ILockKeyTransferHook
interface is quite straightforward:
interface ILockKeyTransferHook
{
/**
* @notice If the lock owner has registered an implementer then this hook
* is called every time balanceOf is called
* @param lockAddress the address of the current lock
* @param tokenId the Id of the transferred key
* @param operator who initiated the transfer
* @param from the previous owner of transferred key
* @param from the previous owner of transferred key
* @param to the new owner of the key
* @param expirationTimestamp the key expiration timestamp (after transfer)
*/
function onKeyTransfer(
address lockAddress,
uint tokenId,
address operator,
address from,
address to,
uint expirationTimestamp
) external;
}
Configuration
Register a hook
To register a hook, call the setEventHooks
method with the contract address(es) containing the hook logic :
function setEventHooks(
address _onKeyPurchaseHook,
address _onKeyCancelHook,
address _onValidKeyHook,
address _onTokenURIHook,
address _onKeyTransferHook
) external
Once a hook address is registered, the function at the address will be executed as an additional step to the original logic.
If you just want to set a single hook, just use the address zero for the others 0x0000000000000000000000000000000000000000
. Additionally, you can de-register any hook anytime by setting it back to 0.
Note that you could create a single contract containing multiple hooks logic, but you will still have to pass the contract address for each hook you want to register.