mirror of
https://github.com/Instadapp/dsa-connectors.git
synced 2024-07-29 22:37:00 +00:00
Add Pangolin Stake Connector and Tests
Resolve errors Resolve review issues Fix comment Co-authored-by: 0xPradyuman <63545809+pradyuman-verma@users.noreply.github.com> Added NatSpec to Staking Connector Fix NatSpecs
This commit is contained in:
parent
358b9bfebb
commit
7884bb31e0
|
@ -16,7 +16,7 @@ abstract contract PangolinResolver is Helpers, Events {
|
||||||
* @param tokenA The address of token A.(For AVAX: 0xEeeeeEeeeEeEeeEeEeEeeEEEeeeeEeeeeeeeEEeE)
|
* @param tokenA The address of token A.(For AVAX: 0xEeeeeEeeeEeEeeEeEeEeeEEEeeeeEeeeeeeeEEeE)
|
||||||
* @param tokenB The address of token B.(For AVAX: 0xEeeeeEeeeEeEeeEeEeEeeEEEeeeeEeeeeeeeEEeE)
|
* @param tokenB The address of token B.(For AVAX: 0xEeeeeEeeeEeEeeEeEeEeeEEEeeeeEeeeeeeeEEeE)
|
||||||
* @param amtA The amount of A tokens to deposit.
|
* @param amtA The amount of A tokens to deposit.
|
||||||
* @param unitAmt The unit amount of of amtB/amtA with slippage.
|
* @param unitAmt The unit amount of amtB/amtA with slippage.
|
||||||
* @param slippage Slippage amount.
|
* @param slippage Slippage amount.
|
||||||
* @param getId ID to retrieve amtA.
|
* @param getId ID to retrieve amtA.
|
||||||
* @param setId ID stores the amount of pools tokens received.
|
* @param setId ID stores the amount of pools tokens received.
|
||||||
|
|
67
contracts/avalanche/connectors/pangolin/staking/events.sol
Normal file
67
contracts/avalanche/connectors/pangolin/staking/events.sol
Normal file
|
@ -0,0 +1,67 @@
|
||||||
|
pragma solidity ^0.7.0;
|
||||||
|
|
||||||
|
contract Events {
|
||||||
|
event LogDepositLpStake(
|
||||||
|
address indexed lptoken,
|
||||||
|
uint256 indexed pid,
|
||||||
|
uint256 stakedAmount,
|
||||||
|
uint256 getId,
|
||||||
|
uint256 setId
|
||||||
|
);
|
||||||
|
|
||||||
|
event LogWithdrawLpStake(
|
||||||
|
address indexed lptoken,
|
||||||
|
uint256 indexed pid,
|
||||||
|
uint256 withdrawAmount,
|
||||||
|
uint256 getId,
|
||||||
|
uint256 setId
|
||||||
|
);
|
||||||
|
|
||||||
|
event LogWithdrawLpAndClaim(
|
||||||
|
address indexed lptoken,
|
||||||
|
uint256 indexed pid,
|
||||||
|
uint256 withdrawAmount,
|
||||||
|
uint256 rewardAmount,
|
||||||
|
uint256 getId,
|
||||||
|
uint256 setId
|
||||||
|
);
|
||||||
|
|
||||||
|
event LogClaimLpReward(
|
||||||
|
address indexed lptoken,
|
||||||
|
uint256 indexed pid,
|
||||||
|
uint256 rewardAmount
|
||||||
|
);
|
||||||
|
|
||||||
|
event LogEmergencyWithdrawLpStake(
|
||||||
|
address indexed lptoken,
|
||||||
|
uint256 indexed pid,
|
||||||
|
uint256 withdrawAmount
|
||||||
|
);
|
||||||
|
|
||||||
|
event LogDepositPNGStake(
|
||||||
|
address indexed stakingContract,
|
||||||
|
uint256 stakedAmount,
|
||||||
|
uint256 getId,
|
||||||
|
uint256 setId
|
||||||
|
);
|
||||||
|
|
||||||
|
event LogWithdrawPNGStake(
|
||||||
|
address indexed stakingContract,
|
||||||
|
uint256 withdrawAmount,
|
||||||
|
uint256 getId,
|
||||||
|
uint256 setId
|
||||||
|
);
|
||||||
|
|
||||||
|
event LogExitPNGStake(
|
||||||
|
address indexed stakingContract,
|
||||||
|
uint256 exitAmount,
|
||||||
|
uint256 rewardAmount,
|
||||||
|
address indexed rewardToken
|
||||||
|
);
|
||||||
|
|
||||||
|
event LogClaimPNGStakeReward(
|
||||||
|
address indexed stakingContract,
|
||||||
|
uint256 rewardAmount,
|
||||||
|
address indexed rewardToken
|
||||||
|
);
|
||||||
|
}
|
160
contracts/avalanche/connectors/pangolin/staking/helpers.sol
Normal file
160
contracts/avalanche/connectors/pangolin/staking/helpers.sol
Normal file
|
@ -0,0 +1,160 @@
|
||||||
|
pragma solidity ^0.7.0;
|
||||||
|
pragma abicoder v2;
|
||||||
|
|
||||||
|
import { TokenInterface } from "../../../common/interfaces.sol";
|
||||||
|
import { DSMath } from "../../../common/math.sol";
|
||||||
|
import { Basic } from "../../../common/basic.sol";
|
||||||
|
import { IERC20, IMiniChefV2, IStakingRewards } from "./interface.sol";
|
||||||
|
|
||||||
|
abstract contract Helpers is DSMath, Basic {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @dev Pangolin MiniChefV2
|
||||||
|
*/
|
||||||
|
IMiniChefV2 internal constant minichefv2 = IMiniChefV2(0x1f806f7C8dED893fd3caE279191ad7Aa3798E928);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @dev Pangolin Token
|
||||||
|
*/
|
||||||
|
IERC20 internal constant PNG = IERC20(0x60781C2586D68229fde47564546784ab3fACA982);
|
||||||
|
|
||||||
|
// LP Staking, use minichefv2 to staking lp tokens and earn png
|
||||||
|
function _depositLPStake(
|
||||||
|
uint pid,
|
||||||
|
uint amount
|
||||||
|
) internal returns (address lpTokenAddr) {
|
||||||
|
require(pid < minichefv2.poolLength(), "Invalid pid!");
|
||||||
|
IERC20 lptoken = minichefv2.lpToken(pid);
|
||||||
|
|
||||||
|
require(amount > 0, "Invalid amount, amount cannot be 0");
|
||||||
|
require(lptoken.balanceOf(address(this)) > 0, "Invalid LP token balance");
|
||||||
|
require(lptoken.balanceOf(address(this)) >= amount, "Invalid amount, amount greater than balance of LP token");
|
||||||
|
|
||||||
|
approve(
|
||||||
|
lptoken,
|
||||||
|
address(minichefv2),
|
||||||
|
amount
|
||||||
|
);
|
||||||
|
|
||||||
|
minichefv2.deposit(pid, amount, address(this));
|
||||||
|
lpTokenAddr = address(lptoken);
|
||||||
|
}
|
||||||
|
|
||||||
|
function _withdraw_LP_Stake(
|
||||||
|
uint pid,
|
||||||
|
uint amount
|
||||||
|
) internal returns (address lpTokenAddr) {
|
||||||
|
require(pid < minichefv2.poolLength(), "Invalid pid!");
|
||||||
|
|
||||||
|
IMiniChefV2.UserInfo memory userinfo = minichefv2.userInfo(pid, address(this));
|
||||||
|
|
||||||
|
require(userinfo.amount >= amount, "Invalid amount, amount greater than balance of staking");
|
||||||
|
require(amount > 0, "Invalid amount, amount cannot be 0");
|
||||||
|
|
||||||
|
minichefv2.withdraw(pid, amount, address(this));
|
||||||
|
|
||||||
|
IERC20 lptoken = minichefv2.lpToken(pid);
|
||||||
|
lpTokenAddr = address(lptoken);
|
||||||
|
}
|
||||||
|
|
||||||
|
function _withdraw_and_getRewards_LP_Stake(
|
||||||
|
uint pid,
|
||||||
|
uint amount
|
||||||
|
) internal returns (uint256 rewardAmount, address lpTokenAddr) {
|
||||||
|
require(pid < minichefv2.poolLength(), "Invalid pid!");
|
||||||
|
|
||||||
|
IMiniChefV2.UserInfo memory userinfo = minichefv2.userInfo(pid, address(this));
|
||||||
|
|
||||||
|
require(userinfo.amount >= amount, "Invalid amount, amount greater than balance of staking");
|
||||||
|
require(amount > 0, "Invalid amount, amount cannot be 0");
|
||||||
|
|
||||||
|
rewardAmount = minichefv2.pendingReward(pid, address(this));
|
||||||
|
|
||||||
|
minichefv2.withdrawAndHarvest(pid, amount, address(this));
|
||||||
|
|
||||||
|
IERC20 lptoken = minichefv2.lpToken(pid);
|
||||||
|
lpTokenAddr = address(lptoken);
|
||||||
|
}
|
||||||
|
|
||||||
|
function _getLPStakeReward(
|
||||||
|
uint pid
|
||||||
|
) internal returns (uint256 rewardAmount, address lpTokenAddr) {
|
||||||
|
require(pid < minichefv2.poolLength(), "Invalid pid!");
|
||||||
|
|
||||||
|
rewardAmount = minichefv2.pendingReward(pid, address(this));
|
||||||
|
|
||||||
|
require(rewardAmount > 0, "No rewards to claim");
|
||||||
|
|
||||||
|
minichefv2.harvest(pid, address(this));
|
||||||
|
|
||||||
|
IERC20 lptoken = minichefv2.lpToken(pid);
|
||||||
|
lpTokenAddr = address(lptoken);
|
||||||
|
}
|
||||||
|
|
||||||
|
function _emergencyWithdraw_LP_Stake(
|
||||||
|
uint pid
|
||||||
|
) internal returns (uint256 lpAmount, address lpTokenAddr) {
|
||||||
|
require(pid < minichefv2.poolLength(), "Invalid pid!");
|
||||||
|
|
||||||
|
IMiniChefV2.UserInfo memory userinfo = minichefv2.userInfo(pid, address(this));
|
||||||
|
lpAmount = userinfo.amount;
|
||||||
|
|
||||||
|
minichefv2.emergencyWithdraw(pid, address(this));
|
||||||
|
IERC20 lptoken = minichefv2.lpToken(pid);
|
||||||
|
lpTokenAddr = address(lptoken);
|
||||||
|
}
|
||||||
|
|
||||||
|
// PNG Staking (Stake PNG, earn another token)
|
||||||
|
function _depositPNGStake(
|
||||||
|
address stakingContract_addr,
|
||||||
|
uint amount
|
||||||
|
) internal {
|
||||||
|
IStakingRewards stakingContract = IStakingRewards(stakingContract_addr);
|
||||||
|
|
||||||
|
require(amount > 0, "Invalid amount, amount cannot be 0");
|
||||||
|
require(PNG.balanceOf(address(this)) > 0, "Invalid PNG balance");
|
||||||
|
require(PNG.balanceOf(address(this)) >= amount, "Invalid amount, amount greater than balance of PNG");
|
||||||
|
|
||||||
|
approve(PNG, stakingContract_addr, amount);
|
||||||
|
|
||||||
|
stakingContract.stake(amount);
|
||||||
|
}
|
||||||
|
|
||||||
|
function _withdrawPNGStake(
|
||||||
|
address stakingContract_addr,
|
||||||
|
uint amount
|
||||||
|
) internal {
|
||||||
|
IStakingRewards stakingContract = IStakingRewards(stakingContract_addr);
|
||||||
|
|
||||||
|
require(stakingContract.balanceOf(address(this)) >= amount, "Invalid amount, amount greater than balance of staking");
|
||||||
|
require(amount > 0, "Invalid amount, amount cannot be 0");
|
||||||
|
|
||||||
|
stakingContract.withdraw(amount);
|
||||||
|
}
|
||||||
|
|
||||||
|
function _exitPNGStake(
|
||||||
|
address stakingContract_addr
|
||||||
|
) internal returns (uint256 exitAmount, uint256 rewardAmount, address rewardToken){
|
||||||
|
IStakingRewards stakingContract = IStakingRewards(stakingContract_addr);
|
||||||
|
|
||||||
|
exitAmount = stakingContract.balanceOf(address(this));
|
||||||
|
rewardAmount = stakingContract.rewards(address(this));
|
||||||
|
|
||||||
|
require(exitAmount > 0, "No balance to exit");
|
||||||
|
|
||||||
|
stakingContract.exit();
|
||||||
|
}
|
||||||
|
|
||||||
|
function _claimPNGStakeReward(
|
||||||
|
address stakingContract_addr
|
||||||
|
) internal returns (uint256 rewardAmount, address rewardToken) {
|
||||||
|
IStakingRewards stakingContract = IStakingRewards(stakingContract_addr);
|
||||||
|
|
||||||
|
rewardAmount = stakingContract.rewards(address(this));
|
||||||
|
rewardToken = stakingContract.rewardsToken();
|
||||||
|
|
||||||
|
require(rewardAmount > 0, "No rewards to claim");
|
||||||
|
|
||||||
|
stakingContract.getReward();
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,49 @@
|
||||||
|
pragma solidity >=0.6.2;
|
||||||
|
pragma abicoder v2;
|
||||||
|
|
||||||
|
import { TokenInterface } from "../../../common/interfaces.sol";
|
||||||
|
|
||||||
|
interface IERC20 is TokenInterface{
|
||||||
|
|
||||||
|
// EIP 2612
|
||||||
|
function permit(address owner, address spender, uint256 value, uint256 deadline, uint8 v, bytes32 r, bytes32 s) external;
|
||||||
|
}
|
||||||
|
|
||||||
|
interface IStakingRewards {
|
||||||
|
// Storage
|
||||||
|
function rewards(address account) view external returns (uint256);
|
||||||
|
|
||||||
|
// View
|
||||||
|
function balanceOf(address account) external view returns (uint256);
|
||||||
|
function rewardsToken() external view returns (address);
|
||||||
|
|
||||||
|
// Mutative
|
||||||
|
function exit() external;
|
||||||
|
function getReward() external;
|
||||||
|
function stake(uint256 amount) external;
|
||||||
|
function withdraw(uint256 amount) external;
|
||||||
|
}
|
||||||
|
|
||||||
|
interface IMiniChefV2 {
|
||||||
|
struct UserInfo {
|
||||||
|
uint256 amount;
|
||||||
|
int256 rewardDebt;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Storage
|
||||||
|
function addedTokens(address token) external returns (bool);
|
||||||
|
function lpToken(uint256 _pid) external view returns (IERC20);
|
||||||
|
function userInfo(uint256 _pid, address _user) external view returns (UserInfo memory);
|
||||||
|
|
||||||
|
// View
|
||||||
|
function pendingReward(uint256 _pid, address _user) external view returns (uint256);
|
||||||
|
function poolLength() external view returns (uint256);
|
||||||
|
|
||||||
|
// Mutative
|
||||||
|
function deposit(uint256 pid, uint256 amount, address to) external;
|
||||||
|
function depositWithPermit(uint256 pid, uint256 amount, address to, uint deadline, uint8 v, bytes32 r, bytes32 s) external;
|
||||||
|
function withdraw(uint256 pid, uint256 amount, address to) external;
|
||||||
|
function harvest(uint256 pid, address to) external;
|
||||||
|
function withdrawAndHarvest(uint256 pid, uint256 amount, address to) external;
|
||||||
|
function emergencyWithdraw(uint256 pid, address to) external;
|
||||||
|
}
|
190
contracts/avalanche/connectors/pangolin/staking/main.sol
Normal file
190
contracts/avalanche/connectors/pangolin/staking/main.sol
Normal file
|
@ -0,0 +1,190 @@
|
||||||
|
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 PangolinStakeResolver is Helpers, Events {
|
||||||
|
|
||||||
|
// LP Staking
|
||||||
|
/**
|
||||||
|
* @notice Deposit LP token in MiniChefV2
|
||||||
|
* @dev Use the Pangolin Stake resolver to get the pid
|
||||||
|
* @param pid The index of the LP token in MiniChefV2.
|
||||||
|
* @param amount The amount of the LP token to deposit.
|
||||||
|
* @param getId ID to retrieve sellAmt.
|
||||||
|
* @param setId ID stores the amount of token brought.
|
||||||
|
*/
|
||||||
|
function depositLpStake(
|
||||||
|
uint pid,
|
||||||
|
uint amount,
|
||||||
|
uint256 getId,
|
||||||
|
uint256 setId
|
||||||
|
) external returns (string memory _eventName, bytes memory _eventParam) {
|
||||||
|
uint _amt = getUint(getId, amount);
|
||||||
|
|
||||||
|
address lpTokenAddr = _depositLPStake(pid, _amt);
|
||||||
|
|
||||||
|
setUint(setId, _amt);
|
||||||
|
_eventName = "LogDepositLpStake(address,uint256,uint256,uint256,uint256)";
|
||||||
|
_eventParam = abi.encode(lpTokenAddr, pid, _amt, getId, setId);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @notice Withdraw LP token from MiniChefV2
|
||||||
|
* @dev Use the Pangolin Stake resolver to get the pid
|
||||||
|
* @param pid The index of the LP token in MiniChefV2.
|
||||||
|
* @param amount The amount of the LP token to withdraw.
|
||||||
|
* @param getId ID to retrieve sellAmt.
|
||||||
|
* @param setId ID stores the amount of token brought.
|
||||||
|
*/
|
||||||
|
function withdrawLpStake(
|
||||||
|
uint pid,
|
||||||
|
uint amount,
|
||||||
|
uint256 getId,
|
||||||
|
uint256 setId
|
||||||
|
) external returns (string memory _eventName, bytes memory _eventParam) {
|
||||||
|
uint _amt = getUint(getId, amount);
|
||||||
|
|
||||||
|
address lpTokenAddr = _withdraw_LP_Stake(pid, _amt);
|
||||||
|
|
||||||
|
setUint(setId, _amt);
|
||||||
|
|
||||||
|
_eventName = "LogWithdrawLpStake(address,uint256,uint256,uint256,uint256)";
|
||||||
|
_eventParam = abi.encode(lpTokenAddr, pid, _amt, getId, setId);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @notice Withdraw LP token staked and claim rewards from MiniChefV2
|
||||||
|
* @dev Use the Pangolin Stake resolver to get the pid
|
||||||
|
* @param pid The index of the LP token in MiniChefV2.
|
||||||
|
* @param amount The amount of the LP token to withdraw.
|
||||||
|
* @param getId ID to retrieve sellAmt.
|
||||||
|
* @param setId ID stores the amount of token brought.
|
||||||
|
*/
|
||||||
|
function withdrawAndClaimLpRewards(
|
||||||
|
uint pid,
|
||||||
|
uint amount,
|
||||||
|
uint256 getId,
|
||||||
|
uint256 setId
|
||||||
|
) external returns (string memory _eventName, bytes memory _eventParam) {
|
||||||
|
uint _amt = getUint(getId, amount);
|
||||||
|
|
||||||
|
(uint256 rewardAmount, address lpTokenAddr) = _withdraw_and_getRewards_LP_Stake(pid, _amt);
|
||||||
|
|
||||||
|
setUint(setId, _amt);
|
||||||
|
|
||||||
|
_eventName = "LogWithdrawLpAndClaim(address,uint256,uint256,uint256,uint256,uint256)";
|
||||||
|
_eventParam = abi.encode(lpTokenAddr, pid, _amt, rewardAmount, getId, setId);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @notice Claim rewards from MiniChefV2
|
||||||
|
* @dev Use the Pangolin Stake resolver to get the pid
|
||||||
|
* @param pid The index of the LP token in MiniChefV2.
|
||||||
|
*/
|
||||||
|
function claimLpRewards(
|
||||||
|
uint pid
|
||||||
|
) external returns (string memory _eventName, bytes memory _eventParam) {
|
||||||
|
(uint256 rewardAmount, address lpTokenAddr) = _getLPStakeReward(pid);
|
||||||
|
|
||||||
|
_eventName = "LogClaimLpReward(address,uint256,uint256)";
|
||||||
|
_eventParam = abi.encode(lpTokenAddr, pid, rewardAmount);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @notice Emergency withdraw all LP token staked from MiniChefV2
|
||||||
|
* @dev Use the Pangolin Stake resolver to get the pid
|
||||||
|
* @param pid The index of the LP token in MiniChefV2.
|
||||||
|
*/
|
||||||
|
function emergencyWithdrawLpStake(
|
||||||
|
uint pid
|
||||||
|
) external returns (string memory _eventName, bytes memory _eventParam) {
|
||||||
|
(uint amount, address lpTokenAddr) = _emergencyWithdraw_LP_Stake(pid);
|
||||||
|
|
||||||
|
_eventName = "LogEmergencyWithdrawLpStake(address,uint256,uint256)";
|
||||||
|
_eventParam = abi.encode(lpTokenAddr, pid, amount);
|
||||||
|
}
|
||||||
|
|
||||||
|
// PNG Staking
|
||||||
|
/**
|
||||||
|
* @notice Deposit PNG in staking contract
|
||||||
|
* @param stakingContract The address of the single PNG staking contract
|
||||||
|
* @param amount The amount of the PNG to deposit.
|
||||||
|
* @param getId ID to retrieve sellAmt.
|
||||||
|
* @param setId ID stores the amount of token brought.
|
||||||
|
*/
|
||||||
|
function depositPNGStake(
|
||||||
|
address stakingContract,
|
||||||
|
uint256 amount,
|
||||||
|
uint256 getId,
|
||||||
|
uint256 setId
|
||||||
|
) external returns (string memory _eventName, bytes memory _eventParam) {
|
||||||
|
uint _amt = getUint(getId, amount);
|
||||||
|
|
||||||
|
_depositPNGStake(stakingContract, _amt);
|
||||||
|
|
||||||
|
setUint(setId, _amt);
|
||||||
|
|
||||||
|
_eventName = "LogDepositPNGStake(address,uint256,uint256,uint256)";
|
||||||
|
_eventParam = abi.encode(stakingContract, _amt, getId, setId);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @notice Withdraw PNG staked from staking contract
|
||||||
|
* @param stakingContract The address of the single PNG staking contract
|
||||||
|
* @param amount The amount of the PNG to withdraw.
|
||||||
|
* @param getId ID to retrieve sellAmt.
|
||||||
|
* @param setId ID stores the amount of token brought.
|
||||||
|
*/
|
||||||
|
function withdrawPNGStake(
|
||||||
|
address stakingContract,
|
||||||
|
uint256 amount,
|
||||||
|
uint256 getId,
|
||||||
|
uint256 setId
|
||||||
|
) external returns (string memory _eventName, bytes memory _eventParam) {
|
||||||
|
uint _amt = getUint(getId, amount);
|
||||||
|
|
||||||
|
_withdrawPNGStake(stakingContract, _amt);
|
||||||
|
|
||||||
|
setUint(setId, _amt);
|
||||||
|
|
||||||
|
_eventName = "LogWithdrawPNGStake(address,uint256,uint256,uint256)";
|
||||||
|
_eventParam = abi.encode(stakingContract, _amt, getId, setId);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @notice Withdraw all PNG staked from staking contract
|
||||||
|
* @param stakingContract The address of the single PNG staking contract
|
||||||
|
*/
|
||||||
|
function exitPNGStake(
|
||||||
|
address stakingContract
|
||||||
|
) external returns (string memory _eventName, bytes memory _eventParam) {
|
||||||
|
(uint256 exitAmount, uint256 rewardAmount, address rewardToken) = _exitPNGStake(stakingContract);
|
||||||
|
|
||||||
|
_eventName = "LogExitPNGStake(address,uint256,uint256,address)";
|
||||||
|
_eventParam = abi.encode(stakingContract, exitAmount, rewardAmount, rewardToken);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @notice Claim rewards from staking contract
|
||||||
|
* @param stakingContract The address of the single PNG staking contract
|
||||||
|
*/
|
||||||
|
function claimPNGStakeReward(
|
||||||
|
address stakingContract
|
||||||
|
) external returns (string memory _eventName, bytes memory _eventParam) {
|
||||||
|
(uint256 rewardAmount, address rewardToken) = _claimPNGStakeReward(stakingContract);
|
||||||
|
|
||||||
|
_eventName = "LogClaimPNGStakeReward(address,uint256,address)";
|
||||||
|
_eventParam = abi.encode(stakingContract, rewardAmount, rewardToken);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
contract ConnectV2PngStakeAvalanche is PangolinStakeResolver {
|
||||||
|
string public constant name = "Pangolin-Stake-v1";
|
||||||
|
}
|
|
@ -1,17 +1,16 @@
|
||||||
import { expect } from "chai";
|
import { expect } from "chai";
|
||||||
import hre from "hardhat";
|
import hre from "hardhat";
|
||||||
|
|
||||||
const { web3, deployments, waffle, ethers } = hre;
|
const { waffle, ethers } = hre;
|
||||||
const { provider, deployContract } = waffle;
|
const { provider } = waffle;
|
||||||
|
|
||||||
import { deployAndEnableConnector } from "../../../scripts/tests/deployAndEnableConnector";
|
import { deployAndEnableConnector } from "../../../scripts/tests/deployAndEnableConnector";
|
||||||
import { buildDSAv2 } from "../../../scripts/tests/buildDSAv2";
|
import { buildDSAv2 } from "../../../scripts/tests/buildDSAv2";
|
||||||
import { encodeSpells } from "../../../scripts/tests/encodeSpells";
|
import { encodeSpells } from "../../../scripts/tests/encodeSpells";
|
||||||
import { getMasterSigner } from "../../../scripts/tests/getMasterSigner";
|
import { getMasterSigner } from "../../../scripts/tests/getMasterSigner";
|
||||||
import { addLiquidity } from "../../../scripts/tests/addLiquidity";
|
|
||||||
import { addresses } from "../../../scripts/tests/avalanche/addresses";
|
import { addresses } from "../../../scripts/tests/avalanche/addresses";
|
||||||
import { abis } from "../../../scripts/constant/abis";
|
import { abis } from "../../../scripts/constant/abis";
|
||||||
import type { Signer, Contract } from "ethers";
|
import { Signer, Contract } from "ethers";
|
||||||
|
|
||||||
import { ConnectV2PngAvalanche__factory } from "../../../typechain";
|
import { ConnectV2PngAvalanche__factory } from "../../../typechain";
|
||||||
|
|
||||||
|
@ -22,13 +21,13 @@ const PNG_AVAX_LP_ADDRESS = "0xd7538cABBf8605BdE1f4901B47B8D42c61DE0367";
|
||||||
describe("Pangolin DEX - Avalanche", function () {
|
describe("Pangolin DEX - Avalanche", function () {
|
||||||
const pangolinConnectorName = "PANGOLIN-TEST-A"
|
const pangolinConnectorName = "PANGOLIN-TEST-A"
|
||||||
|
|
||||||
let dsaWallet0: any;
|
let dsaWallet0: Contract;
|
||||||
let masterSigner: any;
|
let masterSigner: Signer;
|
||||||
let instaConnectorsV2: any;
|
let instaConnectorsV2: Contract;
|
||||||
let pangolinConnector: any;
|
let pangolinConnector: Contract;
|
||||||
|
|
||||||
const wallets = provider.getWallets()
|
const wallets = provider.getWallets()
|
||||||
const [wallet0, wallet1, wallet2, wallet3] = wallets
|
const [wallet0, wallet1] = wallets
|
||||||
before(async () => {
|
before(async () => {
|
||||||
await hre.network.provider.request({
|
await hre.network.provider.request({
|
||||||
method: "hardhat_reset",
|
method: "hardhat_reset",
|
||||||
|
@ -61,12 +60,12 @@ describe("Pangolin DEX - Avalanche", function () {
|
||||||
it("Should have contracts deployed.", async function () {
|
it("Should have contracts deployed.", async function () {
|
||||||
expect(!!instaConnectorsV2.address).to.be.true;
|
expect(!!instaConnectorsV2.address).to.be.true;
|
||||||
expect(!!pangolinConnector.address).to.be.true;
|
expect(!!pangolinConnector.address).to.be.true;
|
||||||
expect(!!masterSigner.address).to.be.true;
|
expect(!!(await masterSigner.getAddress())).to.be.true;
|
||||||
});
|
});
|
||||||
|
|
||||||
describe("DSA wallet setup", function () {
|
describe("DSA wallet setup", function () {
|
||||||
it("Should build DSA v2", async function () {
|
it("Should build DSA v2", async function () {
|
||||||
dsaWallet0 = await buildDSAv2(wallet0.address)
|
dsaWallet0 = await buildDSAv2(wallet0.getAddress())
|
||||||
expect(!!dsaWallet0.address).to.be.true;
|
expect(!!dsaWallet0.address).to.be.true;
|
||||||
});
|
});
|
||||||
|
|
821
test/avalanche/pangolin/pangolin_stake.test.ts
Normal file
821
test/avalanche/pangolin/pangolin_stake.test.ts
Normal file
|
@ -0,0 +1,821 @@
|
||||||
|
import { expect } from "chai";
|
||||||
|
import hre from "hardhat";
|
||||||
|
|
||||||
|
const { waffle, ethers } = hre;
|
||||||
|
const { provider } = waffle;
|
||||||
|
|
||||||
|
import { deployAndEnableConnector } from "../../../scripts/tests/deployAndEnableConnector";
|
||||||
|
import { buildDSAv2 } from "../../../scripts/tests/buildDSAv2";
|
||||||
|
import { encodeSpells } from "../../../scripts/tests/encodeSpells";
|
||||||
|
import { getMasterSigner } from "../../../scripts/tests/getMasterSigner";
|
||||||
|
import { addresses } from "../../../scripts/tests/avalanche/addresses";
|
||||||
|
import { abis } from "../../../scripts/constant/abis";
|
||||||
|
import { Signer, Contract, BigNumber } from "ethers";
|
||||||
|
|
||||||
|
import { ConnectV2PngAvalanche__factory, ConnectV2PngStakeAvalanche__factory } from "../../../typechain";
|
||||||
|
|
||||||
|
const PNG_ADDRESS = "0x60781C2586D68229fde47564546784ab3fACA982";
|
||||||
|
const WAVAX_ADDRESS = "0xB31f66AA3C1e785363F0875A1B74E27b85FD66c7";
|
||||||
|
const PNG_AVAX_LP_ADDRESS = "0xd7538cABBf8605BdE1f4901B47B8D42c61DE0367";
|
||||||
|
const PNG_STAKING_ADDRESS = "0x88afdaE1a9F58Da3E68584421937E5F564A0135b";
|
||||||
|
|
||||||
|
describe("Pangolin Stake - Avalanche", function () {
|
||||||
|
const pangolinConnectorName = "PANGOLIN-TEST-A"
|
||||||
|
const pangolinStakeConnectorName = "PANGOLIN-STAKE-TEST-A"
|
||||||
|
|
||||||
|
let dsaWallet0: Contract;
|
||||||
|
let masterSigner: Signer;
|
||||||
|
let instaConnectorsV2: Contract;
|
||||||
|
let pangolinConnector: Contract;
|
||||||
|
let pangolinStakeConnector: Contract;
|
||||||
|
|
||||||
|
let PNG: Contract;
|
||||||
|
|
||||||
|
const wallets = provider.getWallets()
|
||||||
|
const [wallet0, wallet1] = wallets
|
||||||
|
before(async () => {
|
||||||
|
await hre.network.provider.request({
|
||||||
|
method: "hardhat_reset",
|
||||||
|
params: [
|
||||||
|
{
|
||||||
|
forking: {
|
||||||
|
jsonRpcUrl: `https://api.avax.network/ext/bc/C/rpc`,
|
||||||
|
blockNumber: 8197390
|
||||||
|
},
|
||||||
|
},
|
||||||
|
],
|
||||||
|
});
|
||||||
|
|
||||||
|
PNG = await ethers.getContractAt(
|
||||||
|
abis.basic.erc20,
|
||||||
|
PNG_ADDRESS
|
||||||
|
);
|
||||||
|
|
||||||
|
masterSigner = await getMasterSigner();
|
||||||
|
instaConnectorsV2 = await ethers.getContractAt(
|
||||||
|
abis.core.connectorsV2,
|
||||||
|
addresses.core.connectorsV2
|
||||||
|
);
|
||||||
|
|
||||||
|
// Deploy and enable Pangolin Connector
|
||||||
|
pangolinConnector = await deployAndEnableConnector({
|
||||||
|
connectorName: pangolinConnectorName,
|
||||||
|
contractArtifact: ConnectV2PngAvalanche__factory,
|
||||||
|
signer: masterSigner,
|
||||||
|
connectors: instaConnectorsV2
|
||||||
|
});
|
||||||
|
console.log("Pangolin Connector address: "+ pangolinConnector.address);
|
||||||
|
|
||||||
|
// Deploy and enable Pangolin Stake Connector
|
||||||
|
pangolinStakeConnector = await deployAndEnableConnector({
|
||||||
|
connectorName: pangolinStakeConnectorName,
|
||||||
|
contractArtifact: ConnectV2PngStakeAvalanche__factory,
|
||||||
|
signer: masterSigner,
|
||||||
|
connectors: instaConnectorsV2
|
||||||
|
});
|
||||||
|
console.log("Pangolin Stake Connector address: "+ pangolinStakeConnector.address);
|
||||||
|
})
|
||||||
|
|
||||||
|
it("Should have contracts deployed.", async function () {
|
||||||
|
expect(!!instaConnectorsV2.address).to.be.true;
|
||||||
|
expect(!!pangolinConnector.address).to.be.true;
|
||||||
|
expect(!!pangolinStakeConnector.address).to.be.true;
|
||||||
|
expect(!!(await masterSigner.getAddress())).to.be.true;
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("DSA wallet setup", function () {
|
||||||
|
it("Should build DSA v2", async function () {
|
||||||
|
dsaWallet0 = await buildDSAv2(wallet0.getAddress())
|
||||||
|
expect(!!dsaWallet0.address).to.be.true;
|
||||||
|
});
|
||||||
|
|
||||||
|
it("Deposit 10 AVAX into DSA wallet", async function () {
|
||||||
|
await wallet0.sendTransaction({
|
||||||
|
to: dsaWallet0.address,
|
||||||
|
value: ethers.utils.parseEther("10")
|
||||||
|
});
|
||||||
|
expect(await ethers.provider.getBalance(dsaWallet0.address)).to.be.gte(ethers.utils.parseEther("10"));
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("Pangolin Staking - LP Stake Test", function () {
|
||||||
|
let lpAmount: BigNumber;
|
||||||
|
let pangolinLPToken: Contract;
|
||||||
|
// Buy 100 PNG and deposity in PNG/AVAX LP
|
||||||
|
before(async () => {
|
||||||
|
const amount = ethers.utils.parseEther("100"); // 100 PNG
|
||||||
|
const int_slippage = 0.03
|
||||||
|
const slippage = ethers.utils.parseEther(int_slippage.toString());
|
||||||
|
const setId = "0";
|
||||||
|
|
||||||
|
const PangolinRouterABI = [
|
||||||
|
"function getAmountsOut(uint amountIn, address[] calldata path) external view returns (uint[] memory amounts)"
|
||||||
|
];
|
||||||
|
|
||||||
|
// Get amount of AVAX for 200 PNG from Pangolin
|
||||||
|
const PangolinRouter = await ethers.getContractAt(
|
||||||
|
PangolinRouterABI,
|
||||||
|
"0xE54Ca86531e17Ef3616d22Ca28b0D458b6C89106"
|
||||||
|
);
|
||||||
|
const amounts = await PangolinRouter.getAmountsOut(
|
||||||
|
amount,
|
||||||
|
[
|
||||||
|
PNG_ADDRESS,
|
||||||
|
WAVAX_ADDRESS
|
||||||
|
]
|
||||||
|
);
|
||||||
|
|
||||||
|
const amtA = amounts[0];
|
||||||
|
const amtB = amounts[1];
|
||||||
|
const unitAmt = (amtB * (1 + int_slippage)) / amtA;
|
||||||
|
const unitAmount = ethers.utils.parseEther(unitAmt.toString());
|
||||||
|
|
||||||
|
const spells = [
|
||||||
|
{
|
||||||
|
connector: pangolinConnectorName,
|
||||||
|
method: "buy",
|
||||||
|
args: [
|
||||||
|
PNG_ADDRESS,
|
||||||
|
"0xeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeee",
|
||||||
|
amount,
|
||||||
|
unitAmount,
|
||||||
|
0,
|
||||||
|
0
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
connector: pangolinConnectorName,
|
||||||
|
method: "deposit",
|
||||||
|
args: [
|
||||||
|
PNG_ADDRESS,
|
||||||
|
"0xeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeee",
|
||||||
|
amount,
|
||||||
|
unitAmount,
|
||||||
|
slippage,
|
||||||
|
0,
|
||||||
|
setId
|
||||||
|
]
|
||||||
|
},
|
||||||
|
];
|
||||||
|
// Run spell transaction
|
||||||
|
const tx = await dsaWallet0.connect(wallet0).cast(
|
||||||
|
...encodeSpells(spells), wallet1.address
|
||||||
|
);
|
||||||
|
const receipt = await tx.wait();
|
||||||
|
pangolinLPToken = await ethers.getContractAt(
|
||||||
|
abis.basic.erc20,
|
||||||
|
PNG_AVAX_LP_ADDRESS
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("Check if has PNG/AVAX LP", async function () {
|
||||||
|
const pangolinPoolAVAXBalance = await pangolinLPToken.balanceOf(dsaWallet0.address);
|
||||||
|
expect(pangolinPoolAVAXBalance, `Pangolin PNG/AVAX LP greater than 0`).to.be.gt(0);
|
||||||
|
console.log("PNG/AVAX LP: ", ethers.utils.formatUnits(pangolinPoolAVAXBalance, "ether").toString())
|
||||||
|
lpAmount = pangolinPoolAVAXBalance;
|
||||||
|
});
|
||||||
|
|
||||||
|
it("Check if all functions reverts by: Invalid pid!", async function () {
|
||||||
|
const pid = BigNumber.from("999999999999");
|
||||||
|
const amount = ethers.utils.parseEther("1");
|
||||||
|
const getId = 0;
|
||||||
|
const setId = 0;
|
||||||
|
|
||||||
|
let spells = [
|
||||||
|
{
|
||||||
|
connector: pangolinStakeConnectorName,
|
||||||
|
method: "depositLpStake",
|
||||||
|
args: [
|
||||||
|
pid,
|
||||||
|
amount,
|
||||||
|
getId,
|
||||||
|
setId
|
||||||
|
]
|
||||||
|
}
|
||||||
|
];
|
||||||
|
await expect(
|
||||||
|
dsaWallet0.connect(wallet0).cast(
|
||||||
|
...encodeSpells(spells),
|
||||||
|
wallet1.address
|
||||||
|
)
|
||||||
|
).to.be.revertedWith("Invalid pid!");
|
||||||
|
|
||||||
|
spells[0].method = "withdrawLpStake"
|
||||||
|
await expect(
|
||||||
|
dsaWallet0.connect(wallet0).cast(
|
||||||
|
...encodeSpells(spells),
|
||||||
|
wallet1.address
|
||||||
|
)
|
||||||
|
).to.be.revertedWith("Invalid pid!");
|
||||||
|
|
||||||
|
spells[0].method = "withdrawAndClaimLpRewards"
|
||||||
|
await expect(
|
||||||
|
dsaWallet0.connect(wallet0).cast(
|
||||||
|
...encodeSpells(spells),
|
||||||
|
wallet1.address
|
||||||
|
)
|
||||||
|
).to.be.revertedWith("Invalid pid!");
|
||||||
|
|
||||||
|
spells = [
|
||||||
|
{
|
||||||
|
connector: pangolinStakeConnectorName,
|
||||||
|
method: "claimLpRewards",
|
||||||
|
args: [
|
||||||
|
pid
|
||||||
|
]
|
||||||
|
}
|
||||||
|
];
|
||||||
|
await expect(
|
||||||
|
dsaWallet0.connect(wallet0).cast(
|
||||||
|
...encodeSpells(spells),
|
||||||
|
wallet1.address
|
||||||
|
)
|
||||||
|
).to.be.revertedWith("Invalid pid!");
|
||||||
|
|
||||||
|
spells[0].method = "emergencyWithdrawLpStake"
|
||||||
|
await expect(
|
||||||
|
dsaWallet0.connect(wallet0).cast(
|
||||||
|
...encodeSpells(spells),
|
||||||
|
wallet1.address
|
||||||
|
)
|
||||||
|
).to.be.revertedWith("Invalid pid!");
|
||||||
|
});
|
||||||
|
|
||||||
|
it("Check if all functions reverts by: 'Invalid amount, amount cannot be 0'", async function () {
|
||||||
|
let spells = [
|
||||||
|
{
|
||||||
|
connector: pangolinStakeConnectorName,
|
||||||
|
method: "depositLpStake",
|
||||||
|
args: [
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
0
|
||||||
|
]
|
||||||
|
}
|
||||||
|
];
|
||||||
|
await expect(
|
||||||
|
dsaWallet0.connect(wallet0).cast(
|
||||||
|
...encodeSpells(spells),
|
||||||
|
wallet1.address
|
||||||
|
)
|
||||||
|
).to.be.revertedWith("Invalid amount, amount cannot be 0");
|
||||||
|
|
||||||
|
spells[0].method = "withdrawLpStake"
|
||||||
|
await expect(
|
||||||
|
dsaWallet0.connect(wallet0).cast(
|
||||||
|
...encodeSpells(spells),
|
||||||
|
wallet1.address
|
||||||
|
)
|
||||||
|
).to.be.revertedWith("Invalid amount, amount cannot be 0");
|
||||||
|
|
||||||
|
spells[0].method = "withdrawLpStake"
|
||||||
|
await expect(
|
||||||
|
dsaWallet0.connect(wallet0).cast(
|
||||||
|
...encodeSpells(spells),
|
||||||
|
wallet1.address
|
||||||
|
)
|
||||||
|
).to.be.revertedWith("Invalid amount, amount cannot be 0");
|
||||||
|
|
||||||
|
spells[0].method = "withdrawAndClaimLpRewards"
|
||||||
|
await expect(
|
||||||
|
dsaWallet0.connect(wallet0).cast(
|
||||||
|
...encodeSpells(spells),
|
||||||
|
wallet1.address
|
||||||
|
)
|
||||||
|
).to.be.revertedWith("Invalid amount, amount cannot be 0");
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("depositLpStake function", function () {
|
||||||
|
it("Check if depositLpStake function reverts by: Invalid amount, amount greater than balance of LP token", async function () {
|
||||||
|
const amount = lpAmount.mul(2);
|
||||||
|
const spells = [
|
||||||
|
{
|
||||||
|
connector: pangolinStakeConnectorName,
|
||||||
|
method: "depositLpStake",
|
||||||
|
args: [
|
||||||
|
0,
|
||||||
|
amount,
|
||||||
|
0,
|
||||||
|
0
|
||||||
|
]
|
||||||
|
}
|
||||||
|
];
|
||||||
|
await expect(
|
||||||
|
dsaWallet0.connect(wallet0).cast(
|
||||||
|
...encodeSpells(spells),
|
||||||
|
wallet1.address
|
||||||
|
)
|
||||||
|
).to.be.revertedWith("Invalid amount, amount greater than balance of LP token");
|
||||||
|
});
|
||||||
|
|
||||||
|
it("Check if success in depositLpStake", async function () {
|
||||||
|
const spells = [
|
||||||
|
{
|
||||||
|
connector: pangolinStakeConnectorName,
|
||||||
|
method: "depositLpStake",
|
||||||
|
args: [
|
||||||
|
0,
|
||||||
|
lpAmount,
|
||||||
|
0,
|
||||||
|
0
|
||||||
|
]
|
||||||
|
}
|
||||||
|
];
|
||||||
|
await expect(
|
||||||
|
dsaWallet0.connect(wallet0).cast(
|
||||||
|
...encodeSpells(spells),
|
||||||
|
wallet1.address
|
||||||
|
)
|
||||||
|
).to.be.not.reverted;
|
||||||
|
// Check if PNG/AVAX LP is equal 0
|
||||||
|
const balance = await pangolinLPToken.balanceOf(dsaWallet0.address);
|
||||||
|
expect(balance).to.be.eq(0);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("Check if depositLpStake function reverts by: Invalid LP token balance", async function () {
|
||||||
|
const spells = [
|
||||||
|
{
|
||||||
|
connector: pangolinStakeConnectorName,
|
||||||
|
method: "depositLpStake",
|
||||||
|
args: [
|
||||||
|
0,
|
||||||
|
lpAmount,
|
||||||
|
0,
|
||||||
|
0
|
||||||
|
]
|
||||||
|
}
|
||||||
|
];
|
||||||
|
await expect(
|
||||||
|
dsaWallet0.connect(wallet0).cast(
|
||||||
|
...encodeSpells(spells),
|
||||||
|
wallet1.address
|
||||||
|
)
|
||||||
|
).to.be.revertedWith("Invalid LP token balance");
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("claimLpRewards function", function () {
|
||||||
|
it("Check if success in claimLpRewards", async function () {
|
||||||
|
// Increase Time in 20 seconds
|
||||||
|
await hre.network.provider.send("evm_increaseTime", [20]);
|
||||||
|
// Mine new block
|
||||||
|
await hre.network.provider.send("evm_mine");
|
||||||
|
const spells = [
|
||||||
|
{
|
||||||
|
connector: pangolinStakeConnectorName,
|
||||||
|
method: "claimLpRewards",
|
||||||
|
args: [0]
|
||||||
|
}
|
||||||
|
];
|
||||||
|
await expect(
|
||||||
|
dsaWallet0.connect(wallet0).cast(
|
||||||
|
...encodeSpells(spells),
|
||||||
|
wallet1.address
|
||||||
|
)
|
||||||
|
).to.be.not.reverted;
|
||||||
|
// Checks if the wallet has more than 100 PNG
|
||||||
|
const balance = await PNG.balanceOf(dsaWallet0.address);
|
||||||
|
expect(balance).to.be.gt(0);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("withdrawLpStake function", function () {
|
||||||
|
it("Check if withdrawLpStake function reverts by: Invalid amount, amount greater than balance of staking", async function () {
|
||||||
|
const amount = lpAmount.mul(2);
|
||||||
|
const spells = [
|
||||||
|
{
|
||||||
|
connector: pangolinStakeConnectorName,
|
||||||
|
method: "withdrawLpStake",
|
||||||
|
args: [
|
||||||
|
0,
|
||||||
|
amount,
|
||||||
|
0,
|
||||||
|
0
|
||||||
|
]
|
||||||
|
}
|
||||||
|
];
|
||||||
|
await expect(
|
||||||
|
dsaWallet0.connect(wallet0).cast(
|
||||||
|
...encodeSpells(spells),
|
||||||
|
wallet1.address
|
||||||
|
)
|
||||||
|
).to.be.revertedWith("Invalid amount, amount greater than balance of staking");
|
||||||
|
});
|
||||||
|
|
||||||
|
it("Check if success in withdrawLpStake", async function () {
|
||||||
|
const spells = [
|
||||||
|
{
|
||||||
|
connector: pangolinStakeConnectorName,
|
||||||
|
method: "withdrawLpStake",
|
||||||
|
args: [
|
||||||
|
0,
|
||||||
|
lpAmount.div(2),
|
||||||
|
0,
|
||||||
|
0
|
||||||
|
]
|
||||||
|
}
|
||||||
|
];
|
||||||
|
await expect(
|
||||||
|
dsaWallet0.connect(wallet0).cast(
|
||||||
|
...encodeSpells(spells),
|
||||||
|
wallet1.address
|
||||||
|
)
|
||||||
|
).to.be.not.reverted;
|
||||||
|
// Check if PNG/AVAX LP is equal 0
|
||||||
|
const balance = await pangolinLPToken.balanceOf(dsaWallet0.address);
|
||||||
|
expect(balance).to.be.eq(lpAmount.div(2));
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("withdrawAndClaimLpRewards function", function () {
|
||||||
|
it("Check if withdrawAndClaimLpRewards function reverts by: Invalid amount, amount greater than balance of staking", async function () {
|
||||||
|
const amount = lpAmount.mul(2);
|
||||||
|
const spells = [
|
||||||
|
{
|
||||||
|
connector: pangolinStakeConnectorName,
|
||||||
|
method: "withdrawAndClaimLpRewards",
|
||||||
|
args: [
|
||||||
|
0,
|
||||||
|
amount,
|
||||||
|
0,
|
||||||
|
0
|
||||||
|
]
|
||||||
|
}
|
||||||
|
];
|
||||||
|
await expect(
|
||||||
|
dsaWallet0.connect(wallet0).cast(
|
||||||
|
...encodeSpells(spells),
|
||||||
|
wallet1.address
|
||||||
|
)
|
||||||
|
).to.be.revertedWith("Invalid amount, amount greater than balance of staking");
|
||||||
|
});
|
||||||
|
|
||||||
|
it("Check if success in withdrawAndClaimLpRewards", async function () {
|
||||||
|
let balance = await pangolinLPToken.balanceOf(dsaWallet0.address);
|
||||||
|
const png_balance = await PNG.balanceOf(dsaWallet0.address);
|
||||||
|
const amount = lpAmount.sub(balance)
|
||||||
|
const spells = [
|
||||||
|
{
|
||||||
|
connector: pangolinStakeConnectorName,
|
||||||
|
method: "withdrawAndClaimLpRewards",
|
||||||
|
args: [
|
||||||
|
0,
|
||||||
|
amount,
|
||||||
|
0,
|
||||||
|
0
|
||||||
|
]
|
||||||
|
}
|
||||||
|
];
|
||||||
|
await expect(
|
||||||
|
dsaWallet0.connect(wallet0).cast(
|
||||||
|
...encodeSpells(spells),
|
||||||
|
wallet1.address
|
||||||
|
)
|
||||||
|
).to.be.not.reverted;
|
||||||
|
// Check if PNG/AVAX LP is equal 0
|
||||||
|
balance = await pangolinLPToken.balanceOf(dsaWallet0.address);
|
||||||
|
expect(balance).to.be.eq(lpAmount);
|
||||||
|
const new_png_balance = await PNG.balanceOf(dsaWallet0.address);
|
||||||
|
expect(new_png_balance).to.be.gt(png_balance);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("emergencyWithdrawLpStake function", function () {
|
||||||
|
// Deposit LP again
|
||||||
|
before(async () => {
|
||||||
|
const spells = [
|
||||||
|
{
|
||||||
|
connector: pangolinStakeConnectorName,
|
||||||
|
method: "depositLpStake",
|
||||||
|
args: [
|
||||||
|
0,
|
||||||
|
lpAmount,
|
||||||
|
0,
|
||||||
|
0
|
||||||
|
]
|
||||||
|
}
|
||||||
|
];
|
||||||
|
await dsaWallet0.connect(wallet0).cast(
|
||||||
|
...encodeSpells(spells),
|
||||||
|
wallet1.address
|
||||||
|
)
|
||||||
|
});
|
||||||
|
|
||||||
|
it("Check if success in emergencyWithdrawLpStake", async function () {
|
||||||
|
let balance = await pangolinLPToken.balanceOf(dsaWallet0.address);
|
||||||
|
const amount = lpAmount.sub(balance)
|
||||||
|
const spells = [
|
||||||
|
{
|
||||||
|
connector: pangolinStakeConnectorName,
|
||||||
|
method: "emergencyWithdrawLpStake",
|
||||||
|
args: [0]
|
||||||
|
}
|
||||||
|
];
|
||||||
|
await expect(
|
||||||
|
dsaWallet0.connect(wallet0).cast(
|
||||||
|
...encodeSpells(spells),
|
||||||
|
wallet1.address
|
||||||
|
)
|
||||||
|
).to.be.not.reverted;
|
||||||
|
// Check if PNG/AVAX LP is equal 0
|
||||||
|
balance = await pangolinLPToken.balanceOf(dsaWallet0.address);
|
||||||
|
expect(balance).to.be.eq(lpAmount);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("Pangolin Staking - Single Stake Test (PNG)", function () {
|
||||||
|
let pngToken: Contract;
|
||||||
|
let stakingContract: Contract;
|
||||||
|
let stakingBalance: BigNumber;
|
||||||
|
before(async () => {
|
||||||
|
const amount = ethers.utils.parseEther("100"); // 100 PNG
|
||||||
|
const int_slippage = 0.03
|
||||||
|
|
||||||
|
const PangolinRouterABI = [
|
||||||
|
"function getAmountsOut(uint amountIn, address[] calldata path) external view returns (uint[] memory amounts)"
|
||||||
|
];
|
||||||
|
|
||||||
|
// Get amount of AVAX for 200 PNG from Pangolin
|
||||||
|
const PangolinRouter = await ethers.getContractAt(
|
||||||
|
PangolinRouterABI,
|
||||||
|
"0xE54Ca86531e17Ef3616d22Ca28b0D458b6C89106"
|
||||||
|
);
|
||||||
|
const amounts = await PangolinRouter.getAmountsOut(
|
||||||
|
amount,
|
||||||
|
[
|
||||||
|
PNG_ADDRESS,
|
||||||
|
WAVAX_ADDRESS
|
||||||
|
]
|
||||||
|
);
|
||||||
|
|
||||||
|
const amtA = amounts[0];
|
||||||
|
const amtB = amounts[1];
|
||||||
|
const unitAmt = (amtB * (1 + int_slippage)) / amtA;
|
||||||
|
const unitAmount = ethers.utils.parseEther(unitAmt.toString());
|
||||||
|
|
||||||
|
const spells = [
|
||||||
|
{
|
||||||
|
connector: pangolinConnectorName,
|
||||||
|
method: "buy",
|
||||||
|
args: [
|
||||||
|
PNG_ADDRESS,
|
||||||
|
"0xeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeee",
|
||||||
|
amount,
|
||||||
|
unitAmount,
|
||||||
|
0,
|
||||||
|
0
|
||||||
|
]
|
||||||
|
}
|
||||||
|
];
|
||||||
|
// Run spell transaction
|
||||||
|
const tx = await dsaWallet0.connect(wallet0).cast(
|
||||||
|
...encodeSpells(spells), wallet1.address
|
||||||
|
);
|
||||||
|
const receipt = await tx.wait();
|
||||||
|
|
||||||
|
pngToken = await ethers.getContractAt(abis.basic.erc20, PNG_ADDRESS);
|
||||||
|
stakingContract = await ethers.getContractAt(abis.basic.erc20, PNG_STAKING_ADDRESS);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("Check if has 100 PNG", async function () {
|
||||||
|
const amount = ethers.utils.parseEther("100");
|
||||||
|
const pngBalance = await pngToken.balanceOf(dsaWallet0.address);
|
||||||
|
expect(pngBalance, `PNG Token is equal 100`).to.be.gt(amount.toString());
|
||||||
|
});
|
||||||
|
|
||||||
|
it("Check if some functions reverts by: Invalid amount, amount cannot be 0", async function () {
|
||||||
|
const amount = 0;
|
||||||
|
const getId = 0;
|
||||||
|
const setId = 0;
|
||||||
|
let spells = [
|
||||||
|
{
|
||||||
|
connector: pangolinStakeConnectorName,
|
||||||
|
method: "depositPNGStake",
|
||||||
|
args: [
|
||||||
|
PNG_STAKING_ADDRESS,
|
||||||
|
amount,
|
||||||
|
getId,
|
||||||
|
setId
|
||||||
|
]
|
||||||
|
}
|
||||||
|
];
|
||||||
|
await expect(
|
||||||
|
dsaWallet0.connect(wallet0).cast(
|
||||||
|
...encodeSpells(spells),
|
||||||
|
wallet1.address
|
||||||
|
)
|
||||||
|
).to.be.revertedWith("Invalid amount, amount cannot be 0");
|
||||||
|
|
||||||
|
spells[0].method = "withdrawPNGStake"
|
||||||
|
await expect(
|
||||||
|
dsaWallet0.connect(wallet0).cast(
|
||||||
|
...encodeSpells(spells),
|
||||||
|
wallet1.address
|
||||||
|
)
|
||||||
|
).to.be.revertedWith("Invalid amount, amount cannot be 0");
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("depositPNGStake function", function () {
|
||||||
|
it("Check if reverts by: Invalid amount, amount greater than balance of PNG", async function () {
|
||||||
|
const amount = ethers.utils.parseEther("200")
|
||||||
|
let spells = [
|
||||||
|
{
|
||||||
|
connector: pangolinStakeConnectorName,
|
||||||
|
method: "depositPNGStake",
|
||||||
|
args: [
|
||||||
|
PNG_STAKING_ADDRESS,
|
||||||
|
amount,
|
||||||
|
0,
|
||||||
|
0
|
||||||
|
]
|
||||||
|
}
|
||||||
|
];
|
||||||
|
await expect(
|
||||||
|
dsaWallet0.connect(wallet0).cast(
|
||||||
|
...encodeSpells(spells),
|
||||||
|
wallet1.address
|
||||||
|
)
|
||||||
|
).to.be.revertedWith("Invalid amount, amount greater than balance of PNG");
|
||||||
|
});
|
||||||
|
|
||||||
|
it("Check if success in depositPNGStake", async function () {
|
||||||
|
const amount = await pngToken.balanceOf(dsaWallet0.address);
|
||||||
|
let spells = [
|
||||||
|
{
|
||||||
|
connector: pangolinStakeConnectorName,
|
||||||
|
method: "depositPNGStake",
|
||||||
|
args: [
|
||||||
|
PNG_STAKING_ADDRESS,
|
||||||
|
amount,
|
||||||
|
0,
|
||||||
|
0
|
||||||
|
]
|
||||||
|
}
|
||||||
|
];
|
||||||
|
await expect(
|
||||||
|
dsaWallet0.connect(wallet0).cast(
|
||||||
|
...encodeSpells(spells),
|
||||||
|
wallet1.address
|
||||||
|
)
|
||||||
|
).to.be.not.reverted;
|
||||||
|
const new_png_balance = await pngToken.balanceOf(dsaWallet0.address);
|
||||||
|
expect(new_png_balance).to.be.eq(0);
|
||||||
|
const staking_balance = await stakingContract.balanceOf(dsaWallet0.address);
|
||||||
|
expect(staking_balance).to.be.gt(0);
|
||||||
|
stakingBalance = staking_balance
|
||||||
|
});
|
||||||
|
|
||||||
|
it("Check if reverts by: Invalid PNG balance", async function () {
|
||||||
|
const amount = ethers.utils.parseEther("100")
|
||||||
|
let spells = [
|
||||||
|
{
|
||||||
|
connector: pangolinStakeConnectorName,
|
||||||
|
method: "depositPNGStake",
|
||||||
|
args: [
|
||||||
|
PNG_STAKING_ADDRESS,
|
||||||
|
amount,
|
||||||
|
0,
|
||||||
|
0
|
||||||
|
]
|
||||||
|
}
|
||||||
|
];
|
||||||
|
await expect(
|
||||||
|
dsaWallet0.connect(wallet0).cast(
|
||||||
|
...encodeSpells(spells),
|
||||||
|
wallet1.address
|
||||||
|
)
|
||||||
|
).to.be.revertedWith("Invalid PNG balance");
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("withdrawPNGStake function", function () {
|
||||||
|
it("Check if reverts by: Invalid amount, amount greater than balance of staking", async function () {
|
||||||
|
const amount = ethers.utils.parseEther("200")
|
||||||
|
let spells = [
|
||||||
|
{
|
||||||
|
connector: pangolinStakeConnectorName,
|
||||||
|
method: "withdrawPNGStake",
|
||||||
|
args: [
|
||||||
|
PNG_STAKING_ADDRESS,
|
||||||
|
amount,
|
||||||
|
0,
|
||||||
|
0
|
||||||
|
]
|
||||||
|
}
|
||||||
|
];
|
||||||
|
await expect(
|
||||||
|
dsaWallet0.connect(wallet0).cast(
|
||||||
|
...encodeSpells(spells),
|
||||||
|
wallet1.address
|
||||||
|
)
|
||||||
|
).to.be.revertedWith("Invalid amount, amount greater than balance of staking");
|
||||||
|
});
|
||||||
|
|
||||||
|
it("Check if success in withdrawPNGStake", async function () {
|
||||||
|
const amount = ethers.utils.parseEther("50");
|
||||||
|
let spells = [
|
||||||
|
{
|
||||||
|
connector: pangolinStakeConnectorName,
|
||||||
|
method: "withdrawPNGStake",
|
||||||
|
args: [
|
||||||
|
PNG_STAKING_ADDRESS,
|
||||||
|
amount,
|
||||||
|
0,
|
||||||
|
0
|
||||||
|
]
|
||||||
|
}
|
||||||
|
];
|
||||||
|
await expect(
|
||||||
|
dsaWallet0.connect(wallet0).cast(
|
||||||
|
...encodeSpells(spells),
|
||||||
|
wallet1.address
|
||||||
|
)
|
||||||
|
).to.be.not.reverted;
|
||||||
|
|
||||||
|
const balance = await pngToken.balanceOf(dsaWallet0.address);
|
||||||
|
expect(balance).to.be.eq(amount);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("claimPNGStakeReward function", function () {
|
||||||
|
it("Check if success in claimPNGStakeReward", async function () {
|
||||||
|
// Increase Time in 20 seconds
|
||||||
|
await hre.network.provider.send("evm_increaseTime", [20]);
|
||||||
|
// Mine new block
|
||||||
|
await hre.network.provider.send("evm_mine");
|
||||||
|
const amount = ethers.utils.parseEther("50");
|
||||||
|
let spells = [
|
||||||
|
{
|
||||||
|
connector: pangolinStakeConnectorName,
|
||||||
|
method: "claimPNGStakeReward",
|
||||||
|
args: [PNG_STAKING_ADDRESS]
|
||||||
|
}
|
||||||
|
];
|
||||||
|
await expect(
|
||||||
|
dsaWallet0.connect(wallet0).cast(
|
||||||
|
...encodeSpells(spells),
|
||||||
|
wallet1.address
|
||||||
|
)
|
||||||
|
).to.be.not.reverted;
|
||||||
|
|
||||||
|
const balance = await pngToken.balanceOf(dsaWallet0.address);
|
||||||
|
expect(balance).to.be.gt(amount);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("Check if reverts by: No rewards to claim", async function () {
|
||||||
|
let spells = [
|
||||||
|
{
|
||||||
|
connector: pangolinStakeConnectorName,
|
||||||
|
method: "claimPNGStakeReward",
|
||||||
|
args: [PNG_STAKING_ADDRESS]
|
||||||
|
}
|
||||||
|
];
|
||||||
|
await expect(
|
||||||
|
dsaWallet0.connect(wallet0).cast(
|
||||||
|
...encodeSpells(spells),
|
||||||
|
wallet1.address
|
||||||
|
)
|
||||||
|
).to.be.revertedWith("No rewards to claim");
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("exitPNGStake function", function () {
|
||||||
|
it("Check if success in exitPNGStake", async function () {
|
||||||
|
let spells = [
|
||||||
|
{
|
||||||
|
connector: pangolinStakeConnectorName,
|
||||||
|
method: "exitPNGStake",
|
||||||
|
args: [PNG_STAKING_ADDRESS]
|
||||||
|
}
|
||||||
|
];
|
||||||
|
await expect(
|
||||||
|
dsaWallet0.connect(wallet0).cast(
|
||||||
|
...encodeSpells(spells),
|
||||||
|
wallet1.address
|
||||||
|
)
|
||||||
|
).to.be.not.reverted;
|
||||||
|
|
||||||
|
const balance = await stakingContract.balanceOf(dsaWallet0.address);
|
||||||
|
expect(balance).to.be.eq(0);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("Check if reverts by: No balance to exit", async function () {
|
||||||
|
let spells = [
|
||||||
|
{
|
||||||
|
connector: pangolinStakeConnectorName,
|
||||||
|
method: "exitPNGStake",
|
||||||
|
args: [PNG_STAKING_ADDRESS]
|
||||||
|
}
|
||||||
|
];
|
||||||
|
await expect(
|
||||||
|
dsaWallet0.connect(wallet0).cast(
|
||||||
|
...encodeSpells(spells),
|
||||||
|
wallet1.address
|
||||||
|
)
|
||||||
|
).to.be.revertedWith("No balance to exit");
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
Loading…
Reference in New Issue
Block a user