Solidity Uniswap V2 Pair中添加流动性

文章详细解释了UniswapV2中的流动性管理机制,涉及LP-Token的创建、添加流动性函数的计算逻辑,以及如何通过Solidity进行智能合约测试。着重讨论了初始流动性分配和不平衡流动性处理的方法。
摘要由CSDN通过智能技术生成

        添加流动性的功能的用户入口,UniswapV2在UniswapV2Router中实现,它用来计算新的流动性并发行LP-Token,流动性管理简单地视为LP-Token管理。当你为一个pair增加流动性时,合约会创造LP Token;当你移除流动性时,LP-Token就会被销毁。pair合约中的添加流动性函数,是只执行核心操作的低级函数。

        添加流动性函数

        这是添加流动性的底层函数,

function mint() public {
   uint256 balance0 = IERC20(token0).balanceOf(address(this));
   uint256 balance1 = IERC20(token1).balanceOf(address(this));
   uint256 amount0 = balance0 - reserve0;
   uint256 amount1 = balance1 - reserve1;

   uint256 liquidity;


   if (totalSupply == 0) {
      liquidity = ???
      _mint(address(0), MINIMUM_LIQUIDITY);
   } else {
      liquidity = ???
   }

   if (liquidity <= 0) revert InsufficientLiquidityMinted();
   _mint(msg.sender, liquidity);
   _update(balance0, balance1);
   emit Mint(msg.sender, amount0, amount1);
}

        首先,我们需要计算用户新存入的金额(即没有保存到reserve中的)。然后,计算必须发行的 LP-Token的数量,作为对提供流动性的奖励。然后,我们发行 LP-Token并更新reserve,函数 _update 只是将余额保存到reserve中。

        从代码中可以看出,最初存入pool时,根据totalSupply 是否为0 ,流动性的计算方法是不同的。想想看,当池中没有流动性时,我们需要发行多少 LP-Token?

        对于初始 LP-token的数量,Uniswap V2 最终使用了存入金额的几何平均数:

        这样的主要好处是,确保了初始流动性比率不会影响资产池份额的价值。

        然后,我们在计算,当pool里已经有一些流动性时发行LP-Token的情况。

        这里的主要有两个要求,第一是,按照比例,存入的Token;第二是,按照比例,发行LP-Token。

        发行LP-Token的数量,与token的存款数量成正比。但是,在一个pair中,有两个潜在的Token——我们应该在公式中使用哪一个呢?

        我们可以选择其中任何一个,但存款金额的比例与reserve的比例越接近,差异就越小。因此,如果存款金额的比例不同,LP-Token金额也会不同,而且其中一个会比另一个大。

        如果我们选择更大的,那么我们将通过提供流动性来激励价格变化,这将导致价格操纵。如果我们选择较小的一个,我们将惩罚不平衡流动性的存款(流动性提供者将获得更少的LP-Token)。很明显,选择更小的数字更有益。

if (totalSupply == 0) {

   liquidity = Math.sqrt(amount0 * amount1) - MINIMUM_LIQUIDITY;

   _mint(address(0), MINIMUM_LIQUIDITY);

} else {

   liquidity = Math.min(

      (amount0 * totalSupply) / _reserve0,

      (amount1 * totalSupply) / _reserve1

   );

}

        在第一个分支中,我们在提供初始流动性时减去 MINIMUM_LIQUIDITY(即常数 1000,或 1e-15)。这样可以防止有人把一个pool的份额做得太贵,从而把小规模的流动性提供者拒之门外。对于大多数pool来说,1000 wei的 LP-Token可以忽略不计,但如果有人试图让一份pool的代币成本过高(比如 100 美元),他们就必须烧掉 1000 倍的成本(即 100,000 美元)。

使用solidity编写测试

        我将使用 Foundry 测试我们的智能合约,它无需与 JavaScript 打交道。我们使用智能合约测试智能合约。这就相当于,测试智能合约的智能合约。

这就是我们设置pair合约测试所需要的:

contract ZuniswapV2PairTest is Test {

  ERC20Mintable token0;

  ERC20Mintable token1;

  ZuniswapV2Pair pair;


  function setUp() public {

    token0 = new ERC20Mintable("Token A", "TKNA");

    token1 = new ERC20Mintable("Token B", "TKNB");

    pair = new ZuniswapV2Pair(address(token0), address(token1));


    token0.mint(10 ether);

    token1.mint(10 ether);

  }

}

测试一下交易对初始化(提供初始流动性):

function testMintBootstrap() public {

  token0.transfer(address(pair), 1 ether);

  token1.transfer(address(pair), 1 ether);


  pair.mint();


  assertEq(pair.balanceOf(address(this)), 1 ether - 1000);

  assertReserves(1 ether, 1 ether);

  assertEq(pair.totalSupply(), 1 ether);

}

1 个ether的Token0 和 1 个ether的Token1 加入测试池。因此,1 个ether的 LP-Token被发行,我们得到 1 个ether - 1000(减去最小流动性)。pool的reserve和总供应量也会相应改变。

如果向一个已有一定流动性的pool提供平衡的流动性,会发生什么情况呢?让我们来看看:

function testMintWhenTheresLiquidity() public {

  token0.transfer(address(pair), 1 ether);

  token1.transfer(address(pair), 1 ether);



  pair.mint(); // + 1 LP



  token0.transfer(address(pair), 2 ether);

  token1.transfer(address(pair), 2 ether);



  pair.mint(); // + 2 LP



  assertEq(pair.balanceOf(address(this)), 3 ether - 1000);

  assertEq(pair.totalSupply(), 3 ether);

  assertReserves(3 ether, 3 ether);

}

让我们看看提供不平衡流动性时会发生什么:

function testMintUnbalanced() public {

  token0.transfer(address(pair), 1 ether);

  token1.transfer(address(pair), 1 ether);



  pair.mint(); // + 1 LP

  assertEq(pair.balanceOf(address(this)), 1 ether - 1000);

  assertReserves(1 ether, 1 ether);



  token0.transfer(address(pair), 2 ether);

  token1.transfer(address(pair), 1 ether);



  pair.mint(); // + 1 LP

  assertEq(pair.balanceOf(address(this)), 2 ether - 1000);

  assertReserves(3 ether, 2 ether);

}

即使用户提供的Token0流动性多于Token1流动性,他们仍然只能获得1 LP-Token。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

0xweb3q

有钱的捧个钱场,没钱的捧个人场

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值