From 102eb7200fa5461142ebba1edd09dd9f178950ee Mon Sep 17 00:00:00 2001 From: saucepoint Date: Wed, 2 Oct 2024 17:10:30 -0400 Subject: [PATCH] standing up first guide --- docs/contracts/v4/guides/_category_.json | 5 + .../v4/guides/hooks/01-your-first-hook.md | 404 ++++++++++++++++++ .../hooks}/04-Volatility-fee-hook.mdx | 0 .../contracts/v4/guides/hooks/_category_.json | 5 + docs/contracts/v4/reference/_category_.json | 2 +- 5 files changed, 415 insertions(+), 1 deletion(-) create mode 100644 docs/contracts/v4/guides/_category_.json create mode 100644 docs/contracts/v4/guides/hooks/01-your-first-hook.md rename docs/contracts/v4/{quickstart/04-hooks => guides/hooks}/04-Volatility-fee-hook.mdx (100%) create mode 100644 docs/contracts/v4/guides/hooks/_category_.json diff --git a/docs/contracts/v4/guides/_category_.json b/docs/contracts/v4/guides/_category_.json new file mode 100644 index 000000000..318ff6c96 --- /dev/null +++ b/docs/contracts/v4/guides/_category_.json @@ -0,0 +1,5 @@ +{ + "label": "Guides", + "position": 5, + "collapsed": true +} diff --git a/docs/contracts/v4/guides/hooks/01-your-first-hook.md b/docs/contracts/v4/guides/hooks/01-your-first-hook.md new file mode 100644 index 000000000..3a8591e30 --- /dev/null +++ b/docs/contracts/v4/guides/hooks/01-your-first-hook.md @@ -0,0 +1,404 @@ +--- +title: Building Your First Hook +--- + +# Introduction + +Uniswap introduced the v4 of their protocol introducing several new concepts such as hooks, flash accounting, singleton architecture and more. The most interesting of these for developers is hooks, and that’s what we’ll be learning about today. + +In this guide, we’ll be conceptualizing, understanding and building a basic points hook, which will give you some idea of how to build your own hook. + +## What are we building? + +Let’s start by conceptualizing what we’re building today and why. Let’s say you have a token named `TOKEN` that you want to incentivize people to buy. One way of doing so is awarding people points when they buy your token. Prior to v4, you’d have to do this off-chain or via your own helper contract outside of the swap logic, but in v4 you can enable universal access using hooks. + +Let’s start by defining when users will be rewarded with these points: + +1. When the user swaps `ETH` into `TOKEN` they will get awarded points equal to how much `ETH` they swapped the token with. +2. When the user adds liquidity, we award them with points equal to the amount of `ETH` they added. +3. [todo] + +In order to keep track of these points, we’ll mint the `POINTS` token to the user, this has an added benefit for the user to be able to track it in their wallet. + +# Hook Design + +Let’s figure out how our hook will work. + +From the Uniswap v4 Documentation, there are several hooks available for developers to integrate with. In our use case, we specifically need the ability to read swaps and figure out what amounts they are swapping for and who they are. + +[consider adding a callout for Universal Router here] + +For our hook, we’ll be using `afterSwap` and `afterAddLiquidity` hooks. Why these instead of the `before...` hooks? We’ll dig deeper into this later in this guide. + +# Let’s create our hook! + +We’ll call this hook `PointsHook` and create it in such a way that any pool paired with `TOKEN` can use it. + +## Setting things up + +[todo: base this on the new template repo] + +## Basic Structure + +So far, we’ve created the file named `PointsHook.sol` which only contains the outline of a hook contract. Let’s add our `afterSwap` and `afterAddLiquidity` hooks to it. + +```solidity +contract PointsHook is BaseHook { + constructor(IPoolManager _poolManager) BaseHook(_poolManager) {} + + function getHookPermissions() + public + pure + override + returns (Hooks.Permissions memory) + { + return + Hooks.Permissions({ + beforeInitialize: false, + afterInitialize: false, + beforeAddLiquidity: false, + afterAddLiquidity: true, + beforeRemoveLiquidity: false, + afterRemoveLiquidity: false, + beforeSwap: false, + afterSwap: true, + beforeDonate: false, + afterDonate: false, + beforeSwapReturnDelta: false, + afterSwapReturnDelta: false, + afterAddLiquidityReturnDelta: false, + afterRemoveLiquidityReturnDelta: false + }); + } + + function afterSwap( + address, + PoolKey calldata key, + IPoolManager.SwapParams calldata, + BalanceDelta delta, + bytes calldata + ) external override returns (bytes4, int128) { + return (BaseHook.afterSwap.selector, 0); + } + + function afterAddLiquidity( + address sender, + PoolKey calldata key, + IPoolManager.ModifyLiquidityParams calldata params, + BalanceDelta delta, + BalanceDelta feesAccrued, + bytes calldata hookData + ) external override returns (bytes4, BalanceDelta) { + return (BaseHook.afterAddLiquidity.selector, delta); + } +} +``` + +You’ll notice that both hooks return their own selector in the functions, this is pattern used by the protocol to signal “successful” invocation. We’ll talk about rest of the return parameters when we start adding the functionality. + +Most of the code at this point should be self-explanatory. It’s not doing anything yet, but it’s a great place to start adding the functionality we need. + +## Points Logic + +Up until here, the hook isn’t actually doing anything, so let’s add some functionality! First, let’s setup the `POINTS` token that we’ll reward people with. + +```solidity +contract PointsToken is ERC20, Owned { + constructor() ERC20("Points Token", "POINTS", 18) Owned(msg.sender) {} + + function mint(address to, uint256 amount) external onlyOwner { + _mint(to, amount); + } +} +``` + +Let’s make it so that our hook can mint some! + +```solidity +contract PointsHook is BaseHook { + PointsToken pointsToken; + + constructor(IPoolManager _poolManager) BaseHook(_poolManager) { + pointsToken = new PointsToken(); + } + + [...] +} +``` + +Next, we need to calculate how many points to assign based on the `ETH` value of the swap or liquidity action. We’ll be awarding `POINTS` in 1:1 ratio for the `ETH`, so if the user swapped 1 `ETH`, we’ll give them 1 `POINTS`. Let’s also create a function to award these to the user. + +```solidity + function getPointsForAmount( + uint256 amount + ) internal pure returns (uint256) { + return amount; // 1:1 with ETH + } + + function awardPoints(address to, uint256 amount) internal { + pointsToken.mint(to, getPointsForAmount(amount)); + } +``` + +## Hook Logic + +Now we need to actually get the value that the user is swapping or adding liquidity with. We’ll be using the two hooks to achieve that functionality. + +### Getting the user address + +Before we go into the logic for the hook, we have a side quest! How do we actually get the address of the user? The `PositionManager` doesn’t pass the user address directly to the hook, mainly because of the complexity of getting that data in the first place. + +You’d have noticed, both of our hooks have a `hookData` field at the end. This allows any arbitrary data to be passed to the hook at the time of invocation, and we’ll use this field to encode the user address. + +Let’s create some helper functions to encode and decode this data: + +```solidity + function getHookData(address user) public pure returns (bytes memory) { + return abi.encode(user); + } + + function parseHookData( + bytes calldata data + ) public pure returns (address user) { + return abi.decode(data, (address)); + } +``` + +### Hook Logic: `afterSwap` + +In order for us to award these points to the user, we need a few things and we also need to create a few conditions. + +Let’s start with the most basic ones. We want the user to be swapping in the `ETH/TOKEN` pool and be buying the `TOKEN` in order to get awarded these `POINTS` token. Next, we need to figure out who the user is and how much ETH they are spending, and finally award the points accordingly. + +```solidity + function afterSwap( + address, + PoolKey calldata key, + IPoolManager.SwapParams calldata swapParams, + BalanceDelta delta, + bytes calldata hookData + ) external override onlyPoolManager returns (bytes4, int128) { + // We only award points in the ETH/TOKEN pools. + if (!key.currency0.isAddressZero()) { + return (BaseHook.afterSwap.selector, 0); + } + + // We only award points if the user is buying the TOKEN + if (!swapParams.zeroForOne) { + return (BaseHook.afterSwap.selector, 0); + } + + // Let's figure out who's the user + address user = parseHookData(hookData); + + // How much ETH are they spending? + uint256 ethSpendAmount = swapParams.amountSpecified < 0 + ? uint256(-swapParams.amountSpecified) + : uint256(int256(-delta.amount0())); + + // And award the points! + _awardPoints(user, ethSpendAmount); + + return (BaseHook.afterSwap.selector, 0); + } +``` + +That middle section about figuring out how much `ETH` the user spent seems a little fishy, what’s going on there? Let’s break it down. + +When `amountSpecified` is less than 0, it means this is an `exact input for output` swap, essentially where the user is coming in with an exact amount of ETH. In the other case, it’s an `exact output for input` swap, where the user is expecting a specific amount out. In our case, we get this from the precalculated `delta` that Uniswap V4 gives us as a part of the `afterSwap` hook. + +### Hook Logic: `afterAddLiquidity` + +Similar to what we did for the `afterSwap` hook, now we need to award users for adding liquidity. We’ll do the exact same thing here, except we’ll award the points based on the added liquidity. + +```solidity + function afterAddLiquidity( + address sender, + PoolKey calldata key, + IPoolManager.ModifyLiquidityParams calldata params, + BalanceDelta delta, + BalanceDelta feesAccrued, + bytes calldata hookData + ) external override onlyPoolManager returns (bytes4, BalanceDelta) { + // We only award points in the ETH/TOKEN pools. + if (!key.currency0.isAddressZero()) { + return (BaseHook.afterAddLiquidity.selector, delta); + } + + // Let's figure out who's the user + address user = parseHookData(hookData); + + // How much ETH are they spending? + uint256 ethSpendAmount = uint256(int256(-delta.amount0())); + + // And award the points! + _awardPoints(user, ethSpendAmount); + + return (BaseHook.afterAddLiquidity.selector, delta); + } +``` + +# Testing + +We’re using Foundry for building our hook, and we’ll continue using it to write our tests. One of the great things about Foundry is that you can write tests in Solidity itself instead of context switching between another language. + +### Hook Contract Address + +The `PositionManager` for Uniswap v4 expects the hook address to indicate supported flags. + +[todo: this section is completely wrong on UHI atm and needs to be rewritten, should consider moving it out of here and have a singular place explaining hook bits] + +### Test Suite + +The starter repo you cloned already has an existing base test file, let’s start by copying it into `PointsHook.t.sol`. + +```solidity +contract PointsHookTest is Test, Fixtures { + using EasyPosm for IPositionManager; + using StateLibrary for IPoolManager; + + PointsHook hook; + PointsToken pointsToken; + PoolId poolId; + + uint256 tokenId; + int24 tickLower; + int24 tickUpper; + + function setUp() public { + // creates the pool manager, utility routers, and test tokens + deployFreshManagerAndRouters(); + deployMintAndApprove2Currencies(); + + deployAndApprovePosm(manager); + + // Deploy the hook to an address with the correct flags + address flags = address( + uint160(Hooks.AFTER_SWAP_FLAG | Hooks.AFTER_ADD_LIQUIDITY_FLAG) ^ + (0x4444 << 144) // Namespace the hook to avoid collisions + ); + bytes memory constructorArgs = abi.encode(manager); //Add all the necessary constructor arguments from the hook + deployCodeTo("PointsHook.sol:PointsHook", constructorArgs, flags); + hook = PointsHook(flags); + pointsToken = hook.pointsToken(); + + // Create the pool + key = PoolKey( + Currency.wrap(address(0)), + currency1, + 3000, + 60, + IHooks(hook) + ); + poolId = key.toId(); + manager.initialize(key, SQRT_PRICE_1_1, ZERO_BYTES); + + // Provide full-range liquidity to the pool + tickLower = TickMath.minUsableTick(key.tickSpacing); + tickUpper = TickMath.maxUsableTick(key.tickSpacing); + + deal(address(this), 200 ether); + + (uint256 amount0, uint256 amount1) = LiquidityAmounts + .getAmountsForLiquidity( + SQRT_PRICE_1_1, + TickMath.getSqrtPriceAtTick(tickLower), + TickMath.getSqrtPriceAtTick(tickUpper), + uint128(100e18) + ); + + (tokenId, ) = posm.mint( + key, + tickLower, + tickUpper, + 100e18, + amount0 + 1, + amount1 + 1, + address(this), + block.timestamp, + hook.getHookData(address(this)) + ); + } + + function test_PointsHook_Swap() public { + // [code here] + } +} +``` + +So far this test setup is fairly simple, we create a bunch of tokens and deploy v4 along with the position manager inside our test. Then, we create a pool with our hook and add some liquidity using the position manager. + +Now, let’s write our test. We’ll start by testing the points awarded during the swap. + +```solidity + function test_PointsHook_Swap() public { + // We already have some points because we added some liquidity during setup. + // So, we'll subtract those from the total points to get the points awarded for this swap. + uint256 startingPoints = pointsToken.balanceOf(address(this)); + + // Let's swap some ETH for the token. + bool zeroForOne = true; + int256 amountSpecified = -1e18; // negative number indicates exact input swap! + BalanceDelta swapDelta = swap( + key, + zeroForOne, + amountSpecified, + hook.getHookData(address(this)) + ); + + uint256 endingPoints = pointsToken.balanceOf(address(this)); + + // Let's make sure we got the right amount of points! + assertEq( + endingPoints - startingPoints, + uint256(-amountSpecified), + "Points awarded for swap should be 1:1 with ETH" + ); + } +``` + +This test case is fairly straightforward and simply swaps 1 ETH for the target token and compares if we got the right amount of points awarded for it. + +Next, we should test our liquidity addition. + +```solidity +function test_PointsHook_Liquidity() public { + // We already have some points because we added some liquidity during setup. + // So, we'll subtract those from the total points to get the points awarded for this swap. + uint256 startingPoints = pointsToken.balanceOf(address(this)); + + uint128 liqToAdd = 100e18; + + (uint256 amount0, uint256 amount1) = LiquidityAmounts + .getAmountsForLiquidity( + SQRT_PRICE_1_1, + TickMath.getSqrtPriceAtTick(tickLower), + TickMath.getSqrtPriceAtTick(tickUpper), + liqToAdd + ); + + // Let's swap some ETH for the token. + posm.mint( + key, + tickLower, + tickUpper, + liqToAdd, + amount0 + 1, + amount1 + 1, + address(this), + block.timestamp, + hook.getHookData(address(this)) + ); + + uint256 endingPoints = pointsToken.balanceOf(address(this)); + + // Let's make sure we got the right amount of points! + assertApproxEqAbs(endingPoints - startingPoints, uint256(liqToAdd), 10); + } +``` + +This test case looks very similar to the `afterSwap` one, except we’re testing based on the liquidity added. You’ll notice at the end we’re testing for approximate equality within 10 points. This is to account for minor differences in actual liquidity added due to ticks and pricing. + +# What’s next? + +[todo: explain docs and introduce advance concepts] \ No newline at end of file diff --git a/docs/contracts/v4/quickstart/04-hooks/04-Volatility-fee-hook.mdx b/docs/contracts/v4/guides/hooks/04-Volatility-fee-hook.mdx similarity index 100% rename from docs/contracts/v4/quickstart/04-hooks/04-Volatility-fee-hook.mdx rename to docs/contracts/v4/guides/hooks/04-Volatility-fee-hook.mdx diff --git a/docs/contracts/v4/guides/hooks/_category_.json b/docs/contracts/v4/guides/hooks/_category_.json new file mode 100644 index 000000000..4a4dd7399 --- /dev/null +++ b/docs/contracts/v4/guides/hooks/_category_.json @@ -0,0 +1,5 @@ +{ + "label": "Hooks", + "position": 1, + "collapsed": true +} diff --git a/docs/contracts/v4/reference/_category_.json b/docs/contracts/v4/reference/_category_.json index 3661fbbed..b654b6d9c 100644 --- a/docs/contracts/v4/reference/_category_.json +++ b/docs/contracts/v4/reference/_category_.json @@ -1,5 +1,5 @@ { "label": "Technical Reference", - "position": 5, + "position": 6, "collapsed": true }