一、说明
通常在合约开发过程中不涉及到添加流动性后计算实际获取的LP数量的操作。因为,用户添加流动性后实际获取的LP数量通常由pancakeswap自动计算完成,pancakeswap通过调用IUniswapV2Pair合约接口计算添加到资金池的代币数量和对标usdt/bnb数量,来计算实际应该mint到用户的LP数量。
但是,在实际的合约开发过程中为了限制用户将所得的LP任意转移到其他钱包地址,绕开合约中的税费限制。为了实现该需要就需要实时的记录用添加流动性获取的LP数量,同步保存到合约中。如果用户将LP转移到钱包地址,这样接着钱包的合约中登记LP余额与实际的LP持有数量不一致,就会禁止接收LP的钱包地址从事一切活动。
通常LP的计算方式是加入资金池的两种代币数量的乘积,再做开方即为实际获取的LP数量。
二、以下为实时统计用户添加流动性获取LP数量,同步到合约地址的过程
- 通过以下函数计算用户添加流动性后实际获得的LP数量
function calLiquidity(
uint256 balanceA,
uint256 amount,
uint256 r0,
uint256 r1
) private view returns (uint256 liquidity, uint256 feeToLiquidity) {
uint256 pairTotalSupply = IUniswapV2Pair(liquidityMainPair).totalSupply();
address feeTo = IUniswapV2Factory(uniswapV2Router.factory()).feeTo();
bool feeOn = feeTo != address(0);
uint256 _kLast = IUniswapV2Pair(liquidityMainPair).kLast();
if (feeOn) {
if (_kLast != 0) {
uint256 rootK = SafeMath.sqrt(r0 * r1);
uint256 rootKLast = SafeMath.sqrt(_kLast);
if (rootK > rootKLast) {
uint256 numerator = pairTotalSupply * (rootK - rootKLast) * 8;
uint256 denominator = rootK * 17 + (rootKLast * 8);
feeToLiquidity = numerator / denominator;
if (feeToLiquidity > 0) pairTotalSupply += feeToLiquidity;
}
}
}
uint256 amount0 = balanceA - r0;
if (pairTotalSupply == 0) {
liquidity = SafeMath.sqrt(amount0 * amount) - 1000;
} else {
liquidity = SafeMath.min(
(amount0 * pairTotalSupply) / r0,
(amount * pairTotalSupply) / r1
);
}
}
通过传递两种加入资金池的代币数量,比较pancakeswap的登记余额,来最终确定添加流动性后用户实际应该获得的LP数量。
2. 通过如下方式识别用户的操作为为添加流动性
function _isAddLiquidity(uint256 amount) internal view returns (uint256 liquidity) {
(uint256 rOther, uint256 rThis, uint256 balanceOther) = _getReserves();
uint256 amountOther;
if (rOther > 0 && rThis > 0) {
amountOther = amount * rOther / rThis;
}
//isAddLP
if (balanceOther >= rOther + amountOther) {
(liquidity,) = calLiquidity(balanceOther, amount, rOther, rThis);
}
}
正确识别用户的操作行为为添加流动性后,再计算用户添加流动性实际获得的LP数量。
3. 实时同步合约中记录的LP数量和用户实际从pancakeswap处mint的LP数量
uint256 addLPLiquidity = _isAddLiquidity(amount);
if(addLPLiquidity > 0) {
isAddLiquidity = true;
UserInfo storage userInfo = _userInfo[from];
userInfo.lpAmount += addLPLiquidity;
}
4. 通过如下方式限制非法转移接收LP的地址不允许一切线上活动
uint256 removeLPLiquidity = _isRemoveLiquidity(amount);
if (removeLPLiquidity > 0) {
isRemoveLiquidity = true;
(uint256 lpAmount, uint256 lpLockAmount, uint256 releaseAmount, uint256 lpBalance) = getUserInfo(to);
if(lpLockAmount > 0) {
require(lpBalance + releaseAmount >= lpLockAmount, "ERC20: Current LP Balance less than locked quantity");
}
require(lpAmount >= removeLPLiquidity, "ERC20: Remove LP quantity exceeding balance");
_userInfo[to].lpAmount -= removeLPLiquidity;
}
三、完整版本合约代码
源码及合约部署、开源、上线交易所、动态参数配置教程下载地址:
五、综述
在计算用户添加流动性实际获得的LP数量和计算LP数量同步到合约地址中的余额时,会存在微小的误差,通常计算所得的LP数量要小于实际mint到的LP数量,目的是避免用户将所有的LP全部撤销,确保LP的持币数量不断的递增。
至此,完成DEFI智能合约开发过程中怎样计算添加流动性后实际获得的LP数量,并同步LP数量到链上,以此限制用户任意转账LP所有操作流程。
pdf+视频币安智能链BSC发币教程及多模式组合合约源代码下载:
币安智能链BSC发币(合约部署、开源、锁仓、LP、参数配置、开发、故障处理、工具使用)教程下载:
多模式(燃烧、回流指定营销地址、分红本币及任意币种,邀请推广八代收益,LP加池分红、交易分红、复利分红、NFT分红、自动筑池、动态手续费、定时开盘、回购)组合合约源代码下载:
pdf+视频币安智能链BSC发币教程及多模式组合合约源代码下载地址:
添加VX或者telegram获取全程线上免费指导
评论前必须登录!
注册