Pool contract transaction fee logic development
Author of this section: @Fish
this will be achieved. Pool
the logic of the fee charged in the contract.
Introduction
in addition to the need to consider deducting fees from users, we should also consider how to allocate fee income according to the liquidity contributed by LP.
First we need Pool
two variables are defined in the contract:
/// @inheritdoc IPool
uint256 public override feeGrowthGlobal0X128;
/// @inheritdoc IPool
uint256 public override feeGrowthGlobal1X128;
they represent the fees collected since the pool was created, so why do you need to record these two values? Because LP can withdraw the handling fee at any time, and the time of each LP withdrawal is different, so when LP withdraws the handling fee, we need to calculate his historical accumulated handling fee income.
Calculation of specific values feeGrowthGlobal0X128
and feeGrowthGlobal1X128
is multiplied by the fee FixedPoint128.Q128
(2 to the 96 power), and then divided by the amount of liquidity. Similar to the transaction in the previous lecture, multiply FixedPoint128.Q128
in order to avoid accuracy problems, the actual token number will be calculated when the handling fee is finally extracted by LP.
Development
The complete code is in demo-contract/contracts/wtfswap/Pool.sol in.
As stated in the introduction, in Pool.sol
the following definition needs to be added:
/// @inheritdoc IPool
uint256 public override feeGrowthGlobal0X128;
/// @inheritdoc IPool
uint256 public override feeGrowthGlobal1X128;
we are in Position
also need to add feeGrowthInside0LastX128
and feeGrowthInside1LastX128
it represents the global fee income when LP last withdrew the fee, so that when LP withdraws the fee, we can calculate the income he can withdraw with the accumulated fee income of the pool.
struct Position {
// 该 Position 拥有的流动性
uint128 liquidity;
// 可提取的 token0 数量
uint128 tokensOwed0;
// 可提取的 token1 数量
uint128 tokensOwed1;
// 上次提取手续费时的 feeGrowthGlobal0X128
+ uint256 feeGrowthInside0LastX128;
// 上次提取手续费是的 feeGrowthGlobal1X128
+ uint256 feeGrowthInside1LastX128;
}
For example, if the pool feeGrowthGlobal0X128
100, when LP withdraws the handling fee Position
medium feeGrowthInside0LastX128
it is also 100, then it means that there is no new handling fee that can be withdrawn from LP.
Next, let's implement the specific logic. First of all, we are in swap
method to update the value of the fee after each transaction:
// 计算手续费
state.feeGrowthGlobalX128 += FullMath.mulDiv(
state.feeAmount,
FixedPoint128.Q128,
liquidity
);
// 更新手续费相关信息
if (zeroForOne) {
feeGrowthGlobal0X128 = state.feeGrowthGlobalX128;
} else {
feeGrowthGlobal1X128 = state.feeGrowthGlobalX128;
}
Among them FullMath.mulDiv
method takes three arguments and returns the product of the first argument and the second argument divided by the third argument.
Then in _Modifyposition
add relevant logic in, each LP call mint
or burn
update the position when the method ( Position
) in tokensOwed0
and tokensOwed1
, record the previously accumulated handling fee and start recording the handling fee again.
function _modifyPosition(
ModifyPositionParams memory params
) private returns (int256 amount0, int256 amount1) {
// 通过新增的流动性计算 amount0 和 amount1
// 参考 UniswapV3 的代码
amount0 = SqrtPriceMath.getAmount0Delta(
sqrtPriceX96,
TickMath.getSqrtPriceAtTick(tickUpper),
params.liquidityDelta
);
amount1 = SqrtPriceMath.getAmount1Delta(
TickMath.getSqrtPriceAtTick(tickLower),
sqrtPriceX96,
params.liquidityDelta
);
Position storage position = positions[params.owner];
+ // 提取手续费,计算从上一次提取到当前的手续费
+ uint128 tokensOwed0 = uint128(
+ FullMath.mulDiv(
+ feeGrowthGlobal0X128 - position.feeGrowthInside0LastX128,
+ position.liquidity,
+ FixedPoint128.Q128
+ )
+ );
+ uint128 tokensOwed1 = uint128(
+ FullMath.mulDiv(
+ feeGrowthGlobal1X128 - position.feeGrowthInside1LastX128,
+ position.liquidity,
+ FixedPoint128.Q128
+ )
+ );
+
+ // 更新提取手续费的记录,同步到当前最新的 feeGrowthGlobal0X128,代表都提取完了
+ position.feeGrowthInside0LastX128 = feeGrowthGlobal0X128;
+ position.feeGrowthInside1LastX128 = feeGrowthGlobal1X128;
+ // 把可以提取的手续费记录到 tokensOwed0 和 tokensOwed1 中
+ // LP 可以通过 collect 来最终提取到用户自己账户上
+ if (tokensOwed0 > 0 || tokensOwed1 > 0) {
+ position.tokensOwed0 += tokensOwed0;
+ position.tokensOwed1 += tokensOwed1;
+ }
// 修改 liquidity
liquidity = LiquidityMath.addDelta(liquidity, params.liquidityDelta);
position.liquidity = LiquidityMath.addDelta(
position.liquidity,
params.liquidityDelta
);
}
In the above code, we pass FullMath.mulDiv
calculate the final fee that can be withdrawn because the calculation is multiplied. FixedPoint128.Q128
, so you need to divide here. FixedPoint128.Q128
.
This way, when LP calls collect
method, you can Position
In tokensOwed0
and tokensOwed1
transferred to the user.
One thing to mention, why are we in burn
or mint
invoked _Modifyposition
the fee is calculated in the user, not in the user. swap
when you record the handling fee that each pool should receive? Because there may be a lot of liquidity in a pool, if it is recorded at the time of trading, it will generate a lot of operations, which will cause the Gas to be too high. In this calculation, the liquidity held by LP is the "holding" Share of LP, and the method of calculating Token by "holding" (Share) is also used in many Defi scenarios.
Contract Testing
we try to continue in the last lecture test/wtfswap/Pool.ts
of swap
additional test code in the sample:
// 提取流动性,调用 burn 方法
await testLP.write.burn([liquidityDelta, pool.address]);
// 查看当前 token 数量
expect(await token0.read.balanceOf([testLP.address])).to.equal(
99995000161384542080378486215n
);
// 提取 token
await testLP.write.collect([testLP.address, pool.address]);
// 判断 token 是否返回给 testLP,并且大于原来的数量,因为收到了手续费,并且有交易换入了 token0
// 初始的 token0 是 const initBalanceValue = 100000000000n * 10n ** 18n;
expect(await token0.read.balanceOf([testLP.address])).to.equal(
100000000099999999999999999998n
);
looking closely at the above test sample, you will find that the number of token 0 of LP is changed from the original 100000000000n * 10n **18n
has become (100000000000n + 100n) * 10n **18n;
(Not exactly equal, there will be a little loss in calculation due to the rounding problem). Because The Middle deal swapped in. 100n * 10n **18n
token0, which includes a handling fee.
So far, we 've done it all. Pool
development of contract logic.
Complete code you can in here. View, the complete test code you can also in here. View. It should be noted that in actual projects, you should write more complete test examples.