-
Notifications
You must be signed in to change notification settings - Fork 25
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Loading branch information
JoscelynFarr
committed
Mar 6, 2024
1 parent
361e597
commit 027666e
Showing
1 changed file
with
106 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,106 @@ | ||
/* | ||
Copyright 2022 JOJO Exchange | ||
SPDX-License-Identifier: BUSL-1.1 | ||
*/ | ||
|
||
pragma solidity ^0.8.19; | ||
|
||
import "@openzeppelin/contracts/access/Ownable.sol"; | ||
import "@openzeppelin/contracts/token/ERC20/ERC20.sol"; | ||
import "@openzeppelin/contracts/token/ERC20/utils/SafeERC20.sol"; | ||
import "./interfaces/IJUSDBank.sol"; | ||
import "./libraries/SignedDecimalMath.sol"; | ||
|
||
contract FlashLoanLiquidateGM is Ownable { | ||
using SafeERC20 for IERC20; | ||
using SignedDecimalMath for uint256; | ||
|
||
address public immutable USDC; | ||
address public immutable JUSD; | ||
address public jusdBank; | ||
address public insurance; | ||
|
||
mapping(address => bool) public whiteListAsset; | ||
|
||
struct LiquidateData { | ||
uint256 actualCollateral; | ||
uint256 insuranceFee; | ||
uint256 actualLiquidatedT0; | ||
uint256 actualLiquidated; | ||
uint256 liquidatedRemainUSDC; | ||
} | ||
|
||
constructor(address _jusdBank, address _USDC, address _JUSD, address _insurance) { | ||
jusdBank = _jusdBank; | ||
USDC = _USDC; | ||
JUSD = _JUSD; | ||
insurance = _insurance; | ||
} | ||
|
||
modifier onlyBank() { | ||
require(jusdBank == msg.sender, "Ownable: caller only can be JUSDBank"); | ||
_; | ||
} | ||
|
||
function setWhiteListAsset(address token, bool isValid) public onlyOwner { | ||
whiteListAsset[token] = isValid; | ||
} | ||
|
||
function JOJOFlashLoan(address asset, uint256, address to, bytes calldata param) external onlyBank { | ||
(LiquidateData memory liquidateData, bytes memory liquidatorParam) = abi.decode(param, (LiquidateData, bytes)); | ||
address liquidator; | ||
assembly { | ||
liquidator := mload(add(liquidatorParam, 20)) | ||
} | ||
require(whiteListAsset[asset], "asset is not in the whitelist"); | ||
IERC20(asset).safeTransfer(liquidator, IERC20(asset).balanceOf(address(this))); | ||
IERC20(JUSD).safeTransferFrom(liquidator, address(this), liquidateData.actualLiquidated); | ||
IERC20(JUSD).approve(jusdBank, liquidateData.actualLiquidated); | ||
IJUSDBank(jusdBank).repay(liquidateData.actualLiquidated, to); | ||
|
||
IERC20(USDC).safeTransferFrom( | ||
liquidator, address(this), liquidateData.insuranceFee + liquidateData.liquidatedRemainUSDC | ||
); | ||
IERC20(USDC).safeTransfer(insurance, liquidateData.insuranceFee); | ||
if (liquidateData.liquidatedRemainUSDC != 0) { | ||
IERC20(USDC).safeTransfer(address(jusdBank), liquidateData.liquidatedRemainUSDC); | ||
} | ||
} | ||
|
||
function getMultiCall( | ||
address withdrawalVault, | ||
address gmToken, | ||
address receiver, | ||
uint256 executionFee, | ||
uint256 gmTokenAmount, | ||
uint256[] memory minimumRecive | ||
) | ||
public | ||
pure | ||
returns (bytes memory) | ||
{ | ||
bytes[] memory param = new bytes[](3); | ||
param[0] = abi.encodeWithSignature("sendWnt(address,uint256)", withdrawalVault, executionFee); | ||
param[1] = abi.encodeWithSignature( | ||
"sendTokens(address,address,uint256)", | ||
gmToken, | ||
withdrawalVault, | ||
gmTokenAmount | ||
); | ||
param[2] = abi.encodeWithSignature( | ||
"createWithdrawal(address,address,address,address,address[],address[],uint256,uint256,bool,uint256,uint256)", | ||
receiver, | ||
0x0000000000000000000000000000000000000000, | ||
0xff00000000000000000000000000000000000001, | ||
gmToken, | ||
new address[](0), | ||
new address[](0), | ||
minimumRecive[0], | ||
minimumRecive[1], | ||
false, | ||
executionFee, | ||
0 | ||
); | ||
return abi.encodeWithSignature("multicall(bytes[])", param); | ||
} | ||
} |