2022-03-22 14:55:14 +00:00
|
|
|
//SPDX-License-Identifier: MIT
|
2021-10-19 22:26:39 +00:00
|
|
|
pragma solidity ^0.7.0;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @title Pangolin.
|
|
|
|
* @dev Decentralized Exchange.
|
|
|
|
*/
|
|
|
|
|
|
|
|
import { TokenInterface } from "../../../common/interfaces.sol";
|
|
|
|
import { Helpers } from "./helpers.sol";
|
|
|
|
import { Events } from "./events.sol";
|
|
|
|
|
|
|
|
abstract contract PangolinResolver is Helpers, Events {
|
|
|
|
/**
|
|
|
|
* @dev Deposit Liquidity.
|
|
|
|
* @notice Deposit Liquidity to a Pangolin pool.
|
|
|
|
* @param tokenA The address of token A.(For AVAX: 0xEeeeeEeeeEeEeeEeEeEeeEEEeeeeEeeeeeeeEEeE)
|
|
|
|
* @param tokenB The address of token B.(For AVAX: 0xEeeeeEeeeEeEeeEeEeEeeEEEeeeeEeeeeeeeEEeE)
|
|
|
|
* @param amtA The amount of A tokens to deposit.
|
2021-12-20 21:50:05 +00:00
|
|
|
* @param unitAmt The unit amount of amtB/amtA with slippage.
|
2021-10-19 22:26:39 +00:00
|
|
|
* @param slippage Slippage amount.
|
|
|
|
* @param getId ID to retrieve amtA.
|
|
|
|
* @param setId ID stores the amount of pools tokens received.
|
|
|
|
*/
|
|
|
|
function deposit(
|
|
|
|
address tokenA,
|
|
|
|
address tokenB,
|
|
|
|
uint256 amtA,
|
|
|
|
uint256 unitAmt,
|
|
|
|
uint256 slippage,
|
|
|
|
uint256 getId,
|
|
|
|
uint256 setId
|
|
|
|
) external payable returns (string memory _eventName, bytes memory _eventParam) {
|
|
|
|
uint _amt = getUint(getId, amtA);
|
|
|
|
|
|
|
|
(uint _amtA, uint _amtB, uint _uniAmt) = _addLiquidity(
|
|
|
|
tokenA,
|
|
|
|
tokenB,
|
|
|
|
_amt,
|
|
|
|
unitAmt,
|
|
|
|
slippage
|
|
|
|
);
|
|
|
|
setUint(setId, _uniAmt);
|
|
|
|
|
|
|
|
_eventName = "LogDepositLiquidity(address,address,uint256,uint256,uint256,uint256,uint256)";
|
|
|
|
_eventParam = abi.encode(tokenA, tokenB, _amtA, _amtB, _uniAmt, getId, setId);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @dev Withdraw Liquidity.
|
|
|
|
* @notice Withdraw Liquidity from a Pangolin pool.
|
|
|
|
* @param tokenA The address of token A.(For AVAX: 0xEeeeeEeeeEeEeeEeEeEeeEEEeeeeEeeeeeeeEEeE)
|
|
|
|
* @param tokenB The address of token B.(For AVAX: 0xEeeeeEeeeEeEeeEeEeEeeEEEeeeeEeeeeeeeEEeE)
|
|
|
|
* @param uniAmt The amount of pool tokens to withdraw.
|
|
|
|
* @param unitAmtA The unit amount of amtA/uniAmt with slippage.
|
|
|
|
* @param unitAmtB The unit amount of amtB/uniAmt with slippage.
|
|
|
|
* @param getId ID to retrieve uniAmt.
|
|
|
|
* @param setIds Array of IDs to store the amount tokens received.
|
|
|
|
*/
|
|
|
|
function withdraw(
|
|
|
|
address tokenA,
|
|
|
|
address tokenB,
|
|
|
|
uint256 uniAmt,
|
|
|
|
uint256 unitAmtA,
|
|
|
|
uint256 unitAmtB,
|
|
|
|
uint256 getId,
|
|
|
|
uint256[] calldata setIds
|
|
|
|
) external payable returns (string memory _eventName, bytes memory _eventParam) {
|
|
|
|
uint _amt = getUint(getId, uniAmt);
|
|
|
|
|
|
|
|
(uint _amtA, uint _amtB, uint _uniAmt) = _removeLiquidity(
|
|
|
|
tokenA,
|
|
|
|
tokenB,
|
|
|
|
_amt,
|
|
|
|
unitAmtA,
|
|
|
|
unitAmtB
|
|
|
|
);
|
|
|
|
|
|
|
|
setUint(setIds[0], _amtA);
|
|
|
|
setUint(setIds[1], _amtB);
|
|
|
|
|
|
|
|
_eventName = "LogWithdrawLiquidity(address,address,uint256,uint256,uint256,uint256,uint256[])";
|
|
|
|
_eventParam = abi.encode(tokenA, tokenB, _amtA, _amtB, _uniAmt, getId, setIds);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @dev Buy AVAX/ERC20_Token.
|
|
|
|
* @notice Buy a token using a Pangolin
|
|
|
|
* @param buyAddr The address of the token to buy.(For AVAX: 0xEeeeeEeeeEeEeeEeEeEeeEEEeeeeEeeeeeeeEEeE)
|
|
|
|
* @param sellAddr The address of the token to sell.(For AVAX: 0xEeeeeEeeeEeEeeEeEeEeeEEEeeeeEeeeeeeeEEeE)
|
|
|
|
* @param buyAmt The amount of tokens to buy.
|
|
|
|
* @param unitAmt The unit amount of sellAmt/buyAmt with slippage.
|
|
|
|
* @param getId ID to retrieve buyAmt.
|
|
|
|
* @param setId ID to store the amount of tokens sold.
|
|
|
|
*/
|
|
|
|
function buy(
|
|
|
|
address buyAddr,
|
|
|
|
address sellAddr,
|
|
|
|
uint256 buyAmt,
|
|
|
|
uint256 unitAmt,
|
|
|
|
uint256 getId,
|
|
|
|
uint256 setId
|
|
|
|
) external payable returns (string memory _eventName, bytes memory _eventParam) {
|
|
|
|
uint _buyAmt = getUint(getId, buyAmt);
|
|
|
|
(TokenInterface _buyAddr, TokenInterface _sellAddr) = changeAvaxAddress(buyAddr, sellAddr);
|
|
|
|
address[] memory paths = getPaths(address(_buyAddr), address(_sellAddr));
|
|
|
|
|
|
|
|
uint _slippageAmt = convert18ToDec(_sellAddr.decimals(),
|
|
|
|
wmul(unitAmt, convertTo18(_buyAddr.decimals(), _buyAmt))
|
|
|
|
);
|
|
|
|
|
|
|
|
checkPair(paths);
|
|
|
|
uint _expectedAmt = getExpectedSellAmt(paths, _buyAmt);
|
|
|
|
require(_slippageAmt >= _expectedAmt, "Too much slippage");
|
|
|
|
|
|
|
|
bool isAvax = address(_sellAddr) == wavaxAddr;
|
|
|
|
convertAvaxToWavax(isAvax, _sellAddr, _expectedAmt);
|
|
|
|
approve(_sellAddr, address(router), _expectedAmt);
|
|
|
|
|
|
|
|
uint _sellAmt = router.swapTokensForExactTokens(
|
|
|
|
_buyAmt,
|
|
|
|
_expectedAmt,
|
|
|
|
paths,
|
|
|
|
address(this),
|
|
|
|
block.timestamp + 1
|
|
|
|
)[0];
|
|
|
|
|
|
|
|
isAvax = address(_buyAddr) == wavaxAddr;
|
|
|
|
convertWavaxToAvax(isAvax, _buyAddr, _buyAmt);
|
|
|
|
|
|
|
|
setUint(setId, _sellAmt);
|
|
|
|
|
|
|
|
_eventName = "LogBuy(address,address,uint256,uint256,uint256,uint256)";
|
|
|
|
_eventParam = abi.encode(buyAddr, sellAddr, _buyAmt, _sellAmt, getId, setId);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @dev Sell AVAX/ERC20_Token.
|
|
|
|
* @notice Sell a token using a Pangolin
|
|
|
|
* @param buyAddr The address of the token to buy.(For AVAX: 0xEeeeeEeeeEeEeeEeEeEeeEEEeeeeEeeeeeeeEEeE)
|
|
|
|
* @param sellAddr The address of the token to sell.(For AVAX: 0xEeeeeEeeeEeEeeEeEeEeeEEEeeeeEeeeeeeeEEeE)
|
|
|
|
* @param sellAmt The amount of the token to sell.
|
|
|
|
* @param unitAmt The unit amount of buyAmt/sellAmt with slippage.
|
|
|
|
* @param getId ID to retrieve sellAmt.
|
|
|
|
* @param setId ID stores the amount of token brought.
|
|
|
|
*/
|
|
|
|
function sell(
|
|
|
|
address buyAddr,
|
|
|
|
address sellAddr,
|
|
|
|
uint256 sellAmt,
|
|
|
|
uint256 unitAmt,
|
|
|
|
uint256 getId,
|
|
|
|
uint256 setId
|
|
|
|
) external payable returns (string memory _eventName, bytes memory _eventParam) {
|
|
|
|
uint _sellAmt = getUint(getId, sellAmt);
|
|
|
|
(TokenInterface _buyAddr, TokenInterface _sellAddr) = changeAvaxAddress(buyAddr, sellAddr);
|
|
|
|
address[] memory paths = getPaths(address(_buyAddr), address(_sellAddr));
|
|
|
|
|
|
|
|
if (_sellAmt == uint(-1)) {
|
|
|
|
_sellAmt = sellAddr == avaxAddr ?
|
|
|
|
address(this).balance :
|
|
|
|
_sellAddr.balanceOf(address(this));
|
|
|
|
}
|
|
|
|
|
|
|
|
uint _slippageAmt = convert18ToDec(_buyAddr.decimals(),
|
|
|
|
wmul(unitAmt, convertTo18(_sellAddr.decimals(), _sellAmt))
|
|
|
|
);
|
|
|
|
|
|
|
|
checkPair(paths);
|
|
|
|
uint _expectedAmt = getExpectedBuyAmt(paths, _sellAmt);
|
|
|
|
require(_slippageAmt <= _expectedAmt, "Too much slippage");
|
|
|
|
|
|
|
|
bool isAvax = address(_sellAddr) == wavaxAddr;
|
|
|
|
convertAvaxToWavax(isAvax, _sellAddr, _sellAmt);
|
|
|
|
approve(_sellAddr, address(router), _sellAmt);
|
|
|
|
|
|
|
|
uint _buyAmt = router.swapExactTokensForTokens(
|
|
|
|
_sellAmt,
|
|
|
|
_expectedAmt,
|
|
|
|
paths,
|
|
|
|
address(this),
|
|
|
|
block.timestamp + 1
|
|
|
|
)[1];
|
|
|
|
|
|
|
|
isAvax = address(_buyAddr) == wavaxAddr;
|
|
|
|
convertWavaxToAvax(isAvax, _buyAddr, _buyAmt);
|
|
|
|
|
|
|
|
setUint(setId, _buyAmt);
|
|
|
|
|
|
|
|
_eventName = "LogSell(address,address,uint256,uint256,uint256,uint256)";
|
|
|
|
_eventParam = abi.encode(buyAddr, sellAddr, _buyAmt, _sellAmt, getId, setId);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
contract ConnectV2PngAvalanche is PangolinResolver {
|
|
|
|
string public constant name = "Pangolin-v1";
|
|
|
|
}
|