dsa-connectors/contracts/polygon/connectors/mstable/main.sol

400 lines
9.9 KiB
Solidity
Raw Normal View History

2022-03-22 16:17:41 +00:00
//SPDX-License-Identifier: MIT
2021-12-29 10:37:38 +00:00
pragma solidity ^0.7.6;
/**
* @title mStable SAVE.
* @dev Depositing and withdrawing directly to Save
*/
import { Helpers } from "./helpers.sol";
import { Events } from "./events.sol";
2022-01-24 15:55:44 +00:00
import { IMasset, IStakingRewardsWithPlatformToken, IFeederPool } from "./interface.sol";
2021-12-29 10:37:38 +00:00
import { TokenInterface } from "../../common/interfaces.sol";
2021-12-30 10:43:01 +00:00
abstract contract PmStableResolver is Events, Helpers {
2021-12-29 10:37:38 +00:00
/***************************************
CORE
****************************************/
/**
* @dev Deposit to Save via mUSD or bAsset
2021-12-29 10:37:38 +00:00
* @notice Deposits token supported by mStable to Save
* @param _token Address of token to deposit
* @param _amount Amount of token to deposit
* @param _minOut Minimum amount of token to mint/deposit, equal to _amount if mUSD
2022-01-27 12:34:47 +00:00
* @param _stake stake token in Vault?
* @param _getId ID to retrieve amt
* @param _setId ID stores the amount of tokens deposited
2021-12-30 05:05:31 +00:00
* @return _eventName Event name
* @return _eventParam Event parameters
2021-12-29 10:37:38 +00:00
*/
function deposit(
2021-12-29 10:37:38 +00:00
address _token,
uint256 _amount,
2022-01-27 12:34:47 +00:00
uint256 _minOut,
bool _stake,
uint256 _getId,
uint256 _setId
2022-02-01 15:41:20 +00:00
)
external
payable
returns (string memory _eventName, bytes memory _eventParam)
{
2022-01-27 12:34:47 +00:00
uint256 amount = getUint(_getId, _amount);
amount = amount == uint256(-1)
? TokenInterface(_token).balanceOf(address(this))
: amount;
uint256 mintedAmount;
2022-01-20 05:31:42 +00:00
address path;
// Check if needs to be minted first
if (IMasset(mUsdToken).bAssetIndexes(_token) != 0) {
// mint first
2022-01-27 12:34:47 +00:00
approve(TokenInterface(_token), mUsdToken, amount);
mintedAmount = IMasset(mUsdToken).mint(
_token,
2022-01-27 12:34:47 +00:00
amount,
_minOut,
address(this)
);
2022-01-20 05:31:42 +00:00
path = mUsdToken;
} else {
2022-01-27 12:34:47 +00:00
require(amount >= _minOut, "mintedAmount < _minOut");
mintedAmount = amount;
2022-01-20 06:56:13 +00:00
path = imUsdToken;
}
2021-12-29 10:37:38 +00:00
2022-01-27 12:34:47 +00:00
setUint(_setId, mintedAmount);
(_eventName, _eventParam) = _deposit(
_token,
mintedAmount,
path,
_stake
);
2021-12-29 10:37:38 +00:00
}
2021-12-29 13:07:27 +00:00
/**
* @dev Deposit to Save via feeder pool
* @notice Deposits token, requires _minOut for minting and _path
* @param _token Address of token to deposit
* @param _amount Amount of token to deposit
* @param _minOut Minimum amount of token to mint
* @param _path Feeder Pool address for _token
2022-01-27 12:34:47 +00:00
* @param _stake stake token in Vault?
* @param _getId ID to retrieve amt
* @param _setId ID stores the amount of tokens deposited
2021-12-30 05:05:31 +00:00
* @return _eventName Event name
* @return _eventParam Event parameters
2021-12-29 13:07:27 +00:00
*/
2021-12-29 10:37:38 +00:00
2021-12-29 13:07:27 +00:00
function depositViaSwap(
address _token,
uint256 _amount,
uint256 _minOut,
2022-01-27 12:34:47 +00:00
address _path,
bool _stake,
uint256 _getId,
uint256 _setId
2022-02-01 15:41:20 +00:00
)
external
payable
returns (string memory _eventName, bytes memory _eventParam)
{
2021-12-29 13:07:27 +00:00
require(_path != address(0), "Path must be set");
require(
IMasset(mUsdToken).bAssetIndexes(_token) == 0,
"Token is bAsset"
);
2022-01-27 12:34:47 +00:00
uint256 amount = getUint(_getId, _amount);
amount = amount == uint256(-1)
? TokenInterface(_token).balanceOf(address(this))
: amount;
approve(TokenInterface(_token), _path, amount);
2021-12-29 13:07:27 +00:00
uint256 mintedAmount = IFeederPool(_path).swap(
_token,
mUsdToken,
2022-01-27 12:34:47 +00:00
amount,
2021-12-29 13:07:27 +00:00
_minOut,
address(this)
);
2022-01-20 06:56:13 +00:00
2022-01-27 12:34:47 +00:00
setUint(_setId, mintedAmount);
(_eventName, _eventParam) = _deposit(
_token,
mintedAmount,
_path,
_stake
);
2021-12-29 13:07:27 +00:00
}
/**
* @dev Withdraw from Save to mUSD or bAsset
2021-12-29 13:07:27 +00:00
* @notice Withdraws from Save Vault to mUSD
* @param _token Address of token to withdraw
2021-12-29 13:07:27 +00:00
* @param _credits Credits to withdraw
* @param _minOut Minimum amount of token to withdraw
2022-01-27 12:34:47 +00:00
* @param _unstake from the Vault first?
* @param _getId ID to retrieve amt
* @param _setId ID stores the amount of tokens withdrawn
2021-12-30 05:05:31 +00:00
* @return _eventName Event name
* @return _eventParam Event parameters
2021-12-29 13:07:27 +00:00
*/
2021-12-30 09:22:24 +00:00
function withdraw(
2021-12-29 13:07:27 +00:00
address _token,
uint256 _credits,
2022-01-27 12:34:47 +00:00
uint256 _minOut,
bool _unstake,
uint256 _getId,
uint256 _setId
2022-02-01 15:41:20 +00:00
)
external
payable
returns (string memory _eventName, bytes memory _eventParam)
{
2022-01-27 12:34:47 +00:00
uint256 credits = getUint(_getId, _credits);
uint256 amountWithdrawn = _withdraw(credits, _unstake);
2021-12-29 13:07:27 +00:00
// Check if needs to be redeemed
if (IMasset(mUsdToken).bAssetIndexes(_token) != 0) {
amountWithdrawn = IMasset(mUsdToken).redeem(
_token,
amountWithdrawn,
_minOut,
address(this)
);
} else {
require(amountWithdrawn >= _minOut, "amountWithdrawn < _minOut");
}
2022-01-27 12:34:47 +00:00
setUint(_setId, amountWithdrawn);
_eventName = "LogWithdraw(address,uint256,address,bool)";
_eventParam = abi.encode(
mUsdToken,
amountWithdrawn,
imUsdToken,
_unstake
);
2021-12-29 13:07:27 +00:00
}
/**
* @dev Withdraw from Save via Feeder Pool
* @notice Withdraws from Save Vault to asset via Feeder Pool
* @param _token bAsset to withdraw to
* @param _credits Credits to withdraw
* @param _minOut Minimum amount of token to mint
* @param _path Feeder Pool address for _token
2022-01-27 12:34:47 +00:00
* @param _unstake from the Vault first?
* @param _getId ID to retrieve amt
* @param _setId ID stores the amount of tokens withdrawn
2021-12-30 05:05:31 +00:00
* @return _eventName Event name
* @return _eventParam Event parameters
2021-12-29 13:07:27 +00:00
*/
function withdrawViaSwap(
address _token,
uint256 _credits,
uint256 _minOut,
2022-01-27 12:34:47 +00:00
address _path,
bool _unstake,
uint256 _getId,
uint256 _setId
2022-02-01 15:41:20 +00:00
)
external
payable
returns (string memory _eventName, bytes memory _eventParam)
{
2021-12-29 13:07:27 +00:00
require(_path != address(0), "Path must be set");
require(
IMasset(mUsdToken).bAssetIndexes(_token) == 0,
"Token is bAsset"
);
2022-01-27 12:34:47 +00:00
uint256 credits = getUint(_getId, _credits);
uint256 amountWithdrawn = _withdraw(credits, _unstake);
2021-12-29 13:07:27 +00:00
approve(TokenInterface(mUsdToken), _path, amountWithdrawn);
uint256 amountRedeemed = IFeederPool(_path).swap(
mUsdToken,
_token,
amountWithdrawn,
_minOut,
address(this)
);
2022-01-27 12:34:47 +00:00
setUint(_setId, amountRedeemed);
_eventName = "LogWithdraw(address,uint256,address,bool)";
_eventParam = abi.encode(_token, amountRedeemed, _path, _unstake);
2021-12-29 13:07:27 +00:00
}
/**
* @dev Claims Rewards
* @notice Claims accrued rewards from the Vault
2022-01-27 12:34:47 +00:00
* @param _getId ID to retrieve amt
* @param _setId ID stores the amount of tokens withdrawn
2021-12-30 05:05:31 +00:00
* @return _eventName Event name
* @return _eventParam Event parameters
2021-12-29 13:07:27 +00:00
*/
2022-01-27 12:34:47 +00:00
function claimRewards(uint256 _getId, uint256 _setId)
2021-12-30 05:05:31 +00:00
external
2022-02-01 15:41:20 +00:00
payable
2021-12-30 05:05:31 +00:00
returns (string memory _eventName, bytes memory _eventParam)
{
(address rewardToken, address platformToken) = _getRewardTokens();
(uint256 rewardAmount, uint256 platformAmount) = _getRewardInternalBal(
rewardToken,
platformToken
);
2021-12-29 13:07:27 +00:00
IStakingRewardsWithPlatformToken(imUsdVault).claimReward();
2021-12-30 05:05:31 +00:00
(
uint256 rewardAmountUpdated,
uint256 platformAmountUpdated
) = _getRewardInternalBal(rewardToken, platformToken);
uint256 claimedRewardToken = sub(rewardAmountUpdated, rewardAmount);
uint256 claimedPlatformToken = sub(
platformAmountUpdated,
platformAmount
);
2022-01-27 12:34:47 +00:00
setUint(_setId, claimedRewardToken);
2022-01-20 05:31:42 +00:00
_eventName = "LogClaimRewards(address,uint256,address,uint256)";
2021-12-30 05:05:31 +00:00
_eventParam = abi.encode(
rewardToken,
claimedRewardToken,
platformToken,
claimedPlatformToken
);
2021-12-29 13:07:27 +00:00
}
2021-12-29 10:37:38 +00:00
2021-12-30 09:22:24 +00:00
/**
* @dev Swap tokens
* @notice Swaps tokens via Masset basket
* @param _input Token address to swap from
* @param _output Token address to swap to
* @param _amount Amount of tokens to swap
* @param _minOut Minimum amount of token to mint
2022-01-27 12:34:47 +00:00
* @param _getId ID to retrieve amt
* @param _setId ID stores the amount of tokens swapped
2021-12-30 09:22:24 +00:00
* @return _eventName Event name
* @return _eventParam Event parameters
*/
function swap(
address _input,
address _output,
uint256 _amount,
2022-01-27 12:34:47 +00:00
uint256 _minOut,
uint256 _getId,
uint256 _setId
2022-02-01 15:41:20 +00:00
)
external
payable
returns (string memory _eventName, bytes memory _eventParam)
{
2022-01-27 12:34:47 +00:00
uint256 amount = getUint(_getId, _amount);
amount = amount == uint256(-1)
? TokenInterface(_input).balanceOf(address(this))
: amount;
approve(TokenInterface(_input), mUsdToken, amount);
2021-12-30 09:22:24 +00:00
uint256 amountSwapped;
// Check the assets and swap accordingly
if (_output == mUsdToken) {
// bAsset to mUSD => mint
amountSwapped = IMasset(mUsdToken).mint(
_input,
2022-01-27 12:34:47 +00:00
amount,
2021-12-30 09:22:24 +00:00
_minOut,
address(this)
);
} else if (_input == mUsdToken) {
// mUSD to bAsset => redeem
amountSwapped = IMasset(mUsdToken).redeem(
_output,
2022-01-27 12:34:47 +00:00
amount,
2021-12-30 09:22:24 +00:00
_minOut,
address(this)
);
} else {
// bAsset to another bAsset => swap
amountSwapped = IMasset(mUsdToken).swap(
_input,
_output,
2022-01-27 12:34:47 +00:00
amount,
2021-12-30 09:22:24 +00:00
_minOut,
address(this)
);
}
2022-01-27 12:34:47 +00:00
setUint(_setId, amountSwapped);
2022-01-20 05:31:42 +00:00
_eventName = "LogSwap(address,address,uint256,uint256)";
2022-01-27 12:34:47 +00:00
_eventParam = abi.encode(_input, _output, amount, amountSwapped);
2021-12-30 09:22:24 +00:00
}
/**
* @dev Swap tokens via Feeder Pool
* @notice Swaps tokens via Feeder Pool
* @param _input Token address to swap from
* @param _output Token address to swap to
* @param _amount Amount of tokens to swap
* @param _minOut Minimum amount of token to mint
* @param _path Feeder Pool address to use
2022-01-27 12:34:47 +00:00
* @param _getId ID to retrieve amt
* @param _setId ID stores the amount of tokens swapped
2021-12-30 09:22:24 +00:00
* @return _eventName Event name
* @return _eventParam Event parameters
*/
function swapViaFeeder(
address _input,
address _output,
uint256 _amount,
uint256 _minOut,
2022-01-27 12:34:47 +00:00
address _path,
uint256 _getId,
uint256 _setId
2022-02-01 15:41:20 +00:00
)
external
payable
returns (string memory _eventName, bytes memory _eventParam)
{
2021-12-30 09:22:24 +00:00
uint256 amountSwapped;
2022-01-27 12:34:47 +00:00
uint256 amount = getUint(_getId, _amount);
amount = amount == uint256(-1)
? TokenInterface(_input).balanceOf(address(this))
: amount;
2021-12-30 09:22:24 +00:00
2022-01-27 12:34:47 +00:00
approve(TokenInterface(_input), _path, amount);
2021-12-30 09:22:24 +00:00
// swaps fAsset to mUSD via Feeder Pool
// swaps mUSD to fAsset via Feeder Pool
amountSwapped = IFeederPool(_path).swap(
_input,
_output,
2022-01-27 12:34:47 +00:00
amount,
2021-12-30 09:22:24 +00:00
_minOut,
address(this)
);
2022-01-27 12:34:47 +00:00
setUint(_setId, amountSwapped);
2022-01-20 05:31:42 +00:00
_eventName = "LogSwap(address,address,uint256,uint256)";
2022-01-27 12:34:47 +00:00
_eventParam = abi.encode(_input, _output, amount, amountSwapped);
2021-12-30 09:22:24 +00:00
}
2021-12-29 10:37:38 +00:00
}
2021-12-30 10:43:01 +00:00
contract ConnectV2PmStable is PmStableResolver {
2022-01-20 05:31:42 +00:00
string public constant name = "mStable-Polygon-v1.0";
2021-12-29 10:37:38 +00:00
}