mirror of
				https://github.com/Instadapp/dsa-connectors.git
				synced 2024-07-29 22:37:00 +00:00 
			
		
		
		
	
						commit
						1321cec354
					
				
							
								
								
									
										22
									
								
								.prettierrc
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										22
									
								
								.prettierrc
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,22 @@ | |||
| { | ||||
|   "printWidth": 120, | ||||
|   "singleQuote": false, | ||||
|   "useTabs": false, | ||||
|   "tabWidth": 2, | ||||
|   "trailingComma": "none", | ||||
|   "semi": true, | ||||
|   "plugins": ["./node_modules/prettier-plugin-solidity"], | ||||
|   "overrides": [ | ||||
|     { | ||||
|       "files": "*.sol", | ||||
|       "options": { | ||||
|         "printWidth": 80, | ||||
|         "tabWidth": 4, | ||||
|         "useTabs": true, | ||||
|         "singleQuote": false, | ||||
|         "bracketSpacing": true, | ||||
|         "explicitTypes": "always" | ||||
|       } | ||||
|     } | ||||
|   ] | ||||
| } | ||||
							
								
								
									
										25
									
								
								contracts/mainnet/connectors/ubiquity/events.sol
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										25
									
								
								contracts/mainnet/connectors/ubiquity/events.sol
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,25 @@ | |||
| // SPDX-License-Identifier: MIT | ||||
| pragma solidity 0.7.6; | ||||
| 
 | ||||
| contract Events { | ||||
| 	event LogDeposit( | ||||
| 		address indexed userAddress, | ||||
| 		address indexed token, | ||||
| 		uint256 amount, | ||||
| 		uint256 indexed bondingShareId, | ||||
| 		uint256 lpAmount, | ||||
| 		uint256 durationWeeks, | ||||
| 		uint256 getId, | ||||
| 		uint256 setId | ||||
| 	); | ||||
| 	event LogWithdraw( | ||||
| 		address indexed userAddress, | ||||
| 		uint256 indexed bondingShareId, | ||||
| 		uint256 lpAmount, | ||||
| 		uint256 endBlock, | ||||
| 		address indexed token, | ||||
| 		uint256 amount, | ||||
| 		uint256 getId, | ||||
| 		uint256 setId | ||||
| 	); | ||||
| } | ||||
							
								
								
									
										55
									
								
								contracts/mainnet/connectors/ubiquity/helpers.sol
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										55
									
								
								contracts/mainnet/connectors/ubiquity/helpers.sol
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,55 @@ | |||
| // SPDX-License-Identifier: MIT | ||||
| pragma solidity 0.7.6; | ||||
| 
 | ||||
| import { Basic } from "../../common/basic.sol"; | ||||
| import { IUbiquityAlgorithmicDollarManager } from "./interfaces.sol"; | ||||
| 
 | ||||
| abstract contract Helpers is Basic { | ||||
| 	/** | ||||
| 	 * @dev Ubiquity Algorithmic Dollar Manager | ||||
| 	 */ | ||||
| 	IUbiquityAlgorithmicDollarManager internal constant ubiquityManager = | ||||
| 		IUbiquityAlgorithmicDollarManager( | ||||
| 			0x4DA97a8b831C345dBe6d16FF7432DF2b7b776d98 | ||||
| 		); | ||||
| 
 | ||||
| 	/** | ||||
| 	 * @dev DAI Address | ||||
| 	 */ | ||||
| 	address internal constant DAI = 0x6B175474E89094C44Da98b954EedeAC495271d0F; | ||||
| 
 | ||||
| 	/** | ||||
| 	 * @dev USDC Address | ||||
| 	 */ | ||||
| 	address internal constant USDC = 0xA0b86991c6218b36c1d19D4a2e9Eb0cE3606eB48; | ||||
| 
 | ||||
| 	/** | ||||
| 	 * @dev USDT Address | ||||
| 	 */ | ||||
| 	address internal constant USDT = 0xdAC17F958D2ee523a2206206994597C13D831ec7; | ||||
| 
 | ||||
| 	/** | ||||
| 	 * @dev Curve 3CRV Token Address | ||||
| 	 */ | ||||
| 	address internal constant CRV3 = 0x6c3F90f043a72FA612cbac8115EE7e52BDe6E490; | ||||
| 
 | ||||
| 	/** | ||||
| 	 * @dev Curve 3Pool Address | ||||
| 	 */ | ||||
| 	address internal constant Pool3 = | ||||
| 		0xbEbc44782C7dB0a1A60Cb6fe97d0b483032FF1C7; | ||||
| 
 | ||||
| 	/** | ||||
| 	 * @dev Ubiquity Algorithmic Dollar Address | ||||
| 	 */ | ||||
| 	function getUAD() internal returns (address) { | ||||
| 		return ubiquityManager.dollarTokenAddress(); | ||||
| 	} | ||||
| 
 | ||||
| 	/** | ||||
| 	 * @dev Ubiquity Metapool uAD / 3CRV Address | ||||
| 	 */ | ||||
| 	function getUADCRV3() internal returns (address) { | ||||
| 		return ubiquityManager.stableSwapMetaPoolAddress(); | ||||
| 	} | ||||
| } | ||||
							
								
								
									
										63
									
								
								contracts/mainnet/connectors/ubiquity/interfaces.sol
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										63
									
								
								contracts/mainnet/connectors/ubiquity/interfaces.sol
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,63 @@ | |||
| // SPDX-License-Identifier: MIT | ||||
| pragma solidity 0.7.6; | ||||
| pragma abicoder v2; | ||||
| 
 | ||||
| interface IUbiquityBondingV2 { | ||||
| 	struct Bond { | ||||
| 		address minter; | ||||
| 		uint256 lpFirstDeposited; | ||||
| 		uint256 creationBlock; | ||||
| 		uint256 lpRewardDebt; | ||||
| 		uint256 endBlock; | ||||
| 		uint256 lpAmount; | ||||
| 	} | ||||
| 
 | ||||
| 	function deposit(uint256 lpAmount, uint256 durationWeeks) | ||||
| 		external | ||||
| 		returns (uint256 bondingShareId); | ||||
| 
 | ||||
| 	function removeLiquidity(uint256 lpAmount, uint256 bondId) external; | ||||
| 
 | ||||
| 	function holderTokens(address) external view returns (uint256[] memory); | ||||
| 
 | ||||
| 	function totalLP() external view returns (uint256); | ||||
| 
 | ||||
| 	function totalSupply() external view returns (uint256); | ||||
| 
 | ||||
| 	function getBond(uint256 bondId) external returns (Bond memory bond); | ||||
| } | ||||
| 
 | ||||
| interface IUbiquityMetaPool { | ||||
| 	function add_liquidity(uint256[2] memory _amounts, uint256 _min_mint_amount) | ||||
| 		external | ||||
| 		returns (uint256); | ||||
| 
 | ||||
| 	function remove_liquidity_one_coin( | ||||
| 		uint256 lpAmount, | ||||
| 		int128 i, | ||||
| 		uint256 min_amount | ||||
| 	) external returns (uint256); | ||||
| } | ||||
| 
 | ||||
| interface I3Pool { | ||||
| 	function add_liquidity( | ||||
| 		uint256[3] calldata _amounts, | ||||
| 		uint256 _min_mint_amount | ||||
| 	) external; | ||||
| 
 | ||||
| 	function remove_liquidity_one_coin( | ||||
| 		uint256 lpAmount, | ||||
| 		int128 i, | ||||
| 		uint256 min_amount | ||||
| 	) external; | ||||
| } | ||||
| 
 | ||||
| interface IUbiquityAlgorithmicDollarManager { | ||||
| 	function dollarTokenAddress() external returns (address); | ||||
| 
 | ||||
| 	function stableSwapMetaPoolAddress() external returns (address); | ||||
| 
 | ||||
| 	function bondingContractAddress() external returns (address); | ||||
| 
 | ||||
| 	function bondingShareAddress() external returns (address); | ||||
| } | ||||
							
								
								
									
										222
									
								
								contracts/mainnet/connectors/ubiquity/main.sol
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										222
									
								
								contracts/mainnet/connectors/ubiquity/main.sol
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,222 @@ | |||
| // SPDX-License-Identifier: MIT | ||||
| pragma solidity ^0.7.6; | ||||
| pragma abicoder v2; | ||||
| 
 | ||||
| /** | ||||
|  * @title Ubiquity. | ||||
|  * @dev Ubiquity Dollar (uAD). | ||||
|  */ | ||||
| 
 | ||||
| import { TokenInterface } from "../../common/interfaces.sol"; | ||||
| import { IUbiquityBondingV2, IUbiquityMetaPool, I3Pool } from "./interfaces.sol"; | ||||
| import { Helpers } from "./helpers.sol"; | ||||
| import { Events } from "./events.sol"; | ||||
| 
 | ||||
| abstract contract UbiquityResolver is Helpers, Events { | ||||
| 	/** | ||||
| 	 * @dev Deposit into Ubiquity protocol | ||||
| 	 * @notice 3POOL (DAI / USDC / USDT) => METAPOOL (3CRV / uAD) => uAD3CRV-f => Ubiquity BondingShare | ||||
| 	 * @notice STEP 1 : 3POOL (DAI / USDC / USDT) => 3CRV | ||||
| 	 * @notice STEP 2 : METAPOOL(3CRV / UAD) => uAD3CRV-f | ||||
| 	 * @notice STEP 3 : uAD3CRV-f => Ubiquity BondingShare | ||||
| 	 * @param token Token deposited : DAI, USDC, USDT, 3CRV, uAD or uAD3CRV-f | ||||
| 	 * @param amount Amount of tokens to deposit (For max: `uint256(-1)`) | ||||
| 	 * @param durationWeeks Duration in weeks tokens will be locked (4-208) | ||||
| 	 * @param getId ID to retrieve amt. | ||||
| 	 * @param setId ID stores the bonding share id of tokens deposited. | ||||
| 	 */ | ||||
| 	function deposit( | ||||
| 		address token, | ||||
| 		uint256 amount, | ||||
| 		uint256 durationWeeks, | ||||
| 		uint256 getId, | ||||
| 		uint256 setId | ||||
| 	) | ||||
| 		external | ||||
| 		payable | ||||
| 		returns (string memory _eventName, bytes memory _eventParam) | ||||
| 	{ | ||||
| 		address UAD3CRVf = getUADCRV3(); | ||||
| 		bool[6] memory tok = [ | ||||
| 			token == DAI, // 0 | ||||
| 			token == USDC, // 1 | ||||
| 			token == USDT, // 2 | ||||
| 			token == CRV3, // 3 | ||||
| 			token == getUAD(), // 4 | ||||
| 			token == UAD3CRVf // 5 | ||||
| 		]; | ||||
| 
 | ||||
| 		require( | ||||
| 			// DAI / USDC / USDT / CRV3 / UAD / UAD3CRVF | ||||
| 			tok[0] || tok[1] || tok[2] || tok[3] || tok[4] || tok[5], | ||||
| 			"Invalid token: must be DAI, USDC, USDT, 3CRV, uAD or uAD3CRV-f" | ||||
| 		); | ||||
| 
 | ||||
| 		uint256 _amount = getUint(getId, amount); | ||||
| 		uint256 lpAmount; | ||||
| 
 | ||||
| 		// Full balance if amount = -1 | ||||
| 		if (_amount == uint256(-1)) { | ||||
| 			_amount = getTokenBal(TokenInterface(token)); | ||||
| 		} | ||||
| 
 | ||||
| 		// STEP 1 : SwapTo3CRV : Deposit DAI, USDC or USDT into 3Pool to get 3Crv LPs | ||||
| 		// DAI / USDC / USDT | ||||
| 		if (tok[0] || tok[1] || tok[2]) { | ||||
| 			uint256[3] memory amounts1; | ||||
| 
 | ||||
| 			if (tok[0]) amounts1[0] = _amount; | ||||
| 			else if (tok[1]) amounts1[1] = _amount; | ||||
| 			else if (tok[2]) amounts1[2] = _amount; | ||||
| 
 | ||||
| 			approve(TokenInterface(token), Pool3, _amount); | ||||
| 			I3Pool(Pool3).add_liquidity(amounts1, 0); | ||||
| 		} | ||||
| 
 | ||||
| 		// STEP 2 : ProvideLiquidityToMetapool : Deposit in uAD3CRV pool to get uAD3CRV-f LPs | ||||
| 		// DAI / USDC / USDT / CRV3 / UAD | ||||
| 		if (tok[0] || tok[1] || tok[2] || tok[3] || tok[4]) { | ||||
| 			uint256[2] memory amounts2; | ||||
| 			address token2 = token; | ||||
| 			uint256 _amount2; | ||||
| 
 | ||||
| 			if (tok[4]) { | ||||
| 				_amount2 = _amount; | ||||
| 				amounts2[0] = _amount2; | ||||
| 			} else { | ||||
| 				if (tok[3]) { | ||||
| 					_amount2 = _amount; | ||||
| 				} else { | ||||
| 					token2 = CRV3; | ||||
| 					_amount2 = getTokenBal(TokenInterface(token2)); | ||||
| 				} | ||||
| 				amounts2[1] = _amount2; | ||||
| 			} | ||||
| 
 | ||||
| 			approve(TokenInterface(token2), UAD3CRVf, _amount2); | ||||
| 			lpAmount = IUbiquityMetaPool(UAD3CRVf).add_liquidity(amounts2, 0); | ||||
| 		} | ||||
| 
 | ||||
| 		// STEP 3 : Farm/ApeIn : Deposit uAD3CRV-f LPs into UbiquityBondingV2 and get Ubiquity Bonding Shares | ||||
| 		// UAD3CRVF | ||||
| 		if (tok[5]) { | ||||
| 			lpAmount = _amount; | ||||
| 		} | ||||
| 
 | ||||
| 		address bonding = ubiquityManager.bondingContractAddress(); | ||||
| 		approve(TokenInterface(UAD3CRVf), bonding, lpAmount); | ||||
| 		uint256 bondingShareId = IUbiquityBondingV2(bonding).deposit( | ||||
| 			lpAmount, | ||||
| 			durationWeeks | ||||
| 		); | ||||
| 
 | ||||
| 		setUint(setId, bondingShareId); | ||||
| 
 | ||||
| 		_eventName = "LogDeposit(address,address,uint256,uint256,uint256,uint256,uint256,uint256)"; | ||||
| 		_eventParam = abi.encode( | ||||
| 			address(this), | ||||
| 			token, | ||||
| 			amount, | ||||
| 			bondingShareId, | ||||
| 			lpAmount, | ||||
| 			durationWeeks, | ||||
| 			getId, | ||||
| 			setId | ||||
| 		); | ||||
| 	} | ||||
| 
 | ||||
| 	/** | ||||
| 	 * @dev Withdraw from Ubiquity protocol | ||||
| 	 * @notice Ubiquity BondingShare => uAD3CRV-f => METAPOOL (3CRV / uAD) => 3POOL (DAI / USDC / USDT) | ||||
| 	 * @notice STEP 1 : Ubiquity BondingShare  => uAD3CRV-f | ||||
| 	 * @notice STEP 2 : uAD3CRV-f => METAPOOL(3CRV / UAD) | ||||
| 	 * @notice STEP 3 : 3CRV => 3POOL (DAI / USDC / USDT) | ||||
| 	 * @param bondingShareId Bonding Share Id to withdraw | ||||
| 	 * @param token Token to withdraw to : DAI, USDC, USDT, 3CRV, uAD or uAD3CRV-f | ||||
| 	 * @param getId ID | ||||
| 	 * @param setId ID | ||||
| 	 */ | ||||
| 	function withdraw( | ||||
| 		uint256 bondingShareId, | ||||
| 		address token, | ||||
| 		uint256 getId, | ||||
| 		uint256 setId | ||||
| 	) | ||||
| 		external | ||||
| 		payable | ||||
| 		returns (string memory _eventName, bytes memory _eventParam) | ||||
| 	{ | ||||
| 		address UAD3CRVf = getUADCRV3(); | ||||
| 		bool[6] memory tok = [ | ||||
| 			token == DAI, // 0 | ||||
| 			token == USDC, // 1 | ||||
| 			token == USDT, // 2 | ||||
| 			token == CRV3, // 3 | ||||
| 			token == getUAD(), // 4 | ||||
| 			token == UAD3CRVf // 5 | ||||
| 		]; | ||||
| 
 | ||||
| 		require( | ||||
| 			// DAI / USDC / USDT / CRV3 / UAD / UAD3CRVF | ||||
| 			tok[0] || tok[1] || tok[2] || tok[3] || tok[4] || tok[5], | ||||
| 			"Invalid token: must be DAI, USDC, USDT, 3CRV, uAD or uAD3CRV-f" | ||||
| 		); | ||||
| 
 | ||||
| 		uint256 _bondingShareId = getUint(getId, bondingShareId); | ||||
| 
 | ||||
| 		// Get Bond | ||||
| 		IUbiquityBondingV2.Bond memory bond = IUbiquityBondingV2( | ||||
| 			ubiquityManager.bondingShareAddress() | ||||
| 		).getBond(_bondingShareId); | ||||
| 
 | ||||
| 		require(address(this) == bond.minter, "Not bond owner"); | ||||
| 
 | ||||
| 		// STEP 1 : Withdraw Ubiquity Bonding Shares to get back uAD3CRV-f LPs | ||||
| 		address bonding = ubiquityManager.bondingContractAddress(); | ||||
| 		IUbiquityBondingV2(bonding).removeLiquidity( | ||||
| 			bond.lpAmount, | ||||
| 			_bondingShareId | ||||
| 		); | ||||
| 
 | ||||
| 		// STEP 2 : Withdraw uAD3CRV-f LPs to get back uAD or 3Crv | ||||
| 		// DAI / USDC / USDT / CRV3 / UAD | ||||
| 		if (tok[0] || tok[1] || tok[2] || tok[3] || tok[4]) { | ||||
| 			uint256 amount2 = getTokenBal(TokenInterface(UAD3CRVf)); | ||||
| 			IUbiquityMetaPool(UAD3CRVf).remove_liquidity_one_coin( | ||||
| 				amount2, | ||||
| 				tok[4] ? 0 : 1, | ||||
| 				0 | ||||
| 			); | ||||
| 		} | ||||
| 
 | ||||
| 		// STEP 3 : Withdraw  3Crv LPs from 3Pool to get back DAI, USDC or USDT | ||||
| 		// DAI / USDC / USDT | ||||
| 		if (tok[0] || tok[1] || tok[2]) { | ||||
| 			uint256 amount1 = getTokenBal(TokenInterface(CRV3)); | ||||
| 			I3Pool(Pool3).remove_liquidity_one_coin( | ||||
| 				amount1, | ||||
| 				tok[0] ? 0 : (tok[1] ? 1 : 2), | ||||
| 				0 | ||||
| 			); | ||||
| 		} | ||||
| 
 | ||||
| 		uint256 amount = getTokenBal(TokenInterface(token)); | ||||
| 
 | ||||
| 		setUint(setId, amount); | ||||
| 		_eventName = "LogWithdraw(address,uint256,uint256,uint256,address,uint256,uint256,uint256)"; | ||||
| 		_eventParam = abi.encode( | ||||
| 			address(this), | ||||
| 			_bondingShareId, | ||||
| 			bond.lpAmount, | ||||
| 			bond.endBlock, | ||||
| 			token, | ||||
| 			amount, | ||||
| 			getId, | ||||
| 			setId | ||||
| 		); | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| contract ConnectV2Ubiquity is UbiquityResolver { | ||||
| 	string public constant name = "Ubiquity-v1"; | ||||
| } | ||||
|  | @ -57,6 +57,9 @@ | |||
|     "hardhat": "^2.7.0", | ||||
|     "hardhat-deploy": "^0.9.14", | ||||
|     "hardhat-deploy-ethers": "^0.3.0-beta.11", | ||||
|     "prettier": "^2.4.1", | ||||
|     "prettier-plugin-solidity": "^1.0.0-beta.18", | ||||
|     "solhint": "^3.3.6", | ||||
|     "husky": "^7.0.4", | ||||
|     "solidity-coverage": "0.7.17", | ||||
|     "ts-node": "^10.4.0", | ||||
|  |  | |||
							
								
								
									
										451
									
								
								test/ubiquity/ubiquity.test.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										451
									
								
								test/ubiquity/ubiquity.test.js
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,451 @@ | |||
| const { expect } = require("chai"); | ||||
| const hre = require("hardhat"); | ||||
| const { waffle, ethers } = hre; | ||||
| const { provider } = waffle; | ||||
| const { BigNumber, utils } = ethers; | ||||
| 
 | ||||
| const deployAndEnableConnector = require("../../scripts/deployAndEnableConnector.js"); | ||||
| const buildDSAv2 = require("../../scripts/buildDSAv2"); | ||||
| const encodeSpells = require("../../scripts/encodeSpells"); | ||||
| const addresses = require("../../scripts/constant/addresses"); | ||||
| const abis = require("../../scripts/constant/abis"); | ||||
| const impersonate = require("../../scripts/impersonate"); | ||||
| const { forkReset, sendEth, mineNBlock } = require("./utils"); | ||||
| 
 | ||||
| const connectV2UbiquityArtifacts = require("../../artifacts/contracts/mainnet/connectors/ubiquity/main.sol/ConnectV2Ubiquity.json"); | ||||
| 
 | ||||
| const { abi: implementationsABI } = require("../../scripts/constant/abi/core/InstaImplementations.json"); | ||||
| const implementationsMappingAddr = "0xCBA828153d3a85b30B5b912e1f2daCac5816aE9D"; | ||||
| 
 | ||||
| describe("Ubiquity", function () { | ||||
|   const ubiquityTest = "UBIQUITY-TEST-A"; | ||||
| 
 | ||||
|   const BOND = "0x2dA07859613C14F6f05c97eFE37B9B4F212b5eF5"; | ||||
|   const UAD = "0x0F644658510c95CB46955e55D7BA9DDa9E9fBEc6"; | ||||
|   const DAI = "0x6B175474E89094C44Da98b954EedeAC495271d0F"; | ||||
|   const USDC = "0xA0b86991c6218b36c1d19D4a2e9Eb0cE3606eB48"; | ||||
|   const USDT = "0xdAC17F958D2ee523a2206206994597C13D831ec7"; | ||||
|   const CRV3 = "0x6c3F90f043a72FA612cbac8115EE7e52BDe6E490"; | ||||
|   const POOL3 = "0xbEbc44782C7dB0a1A60Cb6fe97d0b483032FF1C7"; | ||||
|   const UAD3CRVF = "0x20955CB69Ae1515962177D164dfC9522feef567E"; | ||||
| 
 | ||||
|   const ethWhaleAddress = "0xc02aaa39b223fe8d0a0e5c4f27ead9083c756cc2"; | ||||
|   const uadWhaleAddress = "0xefC0e701A824943b469a694aC564Aa1efF7Ab7dd"; | ||||
| 
 | ||||
|   const blockFork = 13097100; | ||||
|   const one = BigNumber.from(10).pow(18); | ||||
|   const onep = BigNumber.from(10).pow(6); | ||||
|   const ABI = [ | ||||
|     "function balanceOf(address owner) view returns (uint256)", | ||||
|     "function allowance(address owner, address spender) external view returns (uint256)", | ||||
|     "function transfer(address to, uint amount) returns (boolean)", | ||||
|     "function remove_liquidity_one_coin(uint256 _burn_amount, int128 i, uint256 _min_received) external returns (uint256)", | ||||
|     "function add_liquidity(uint256[3],uint256) returns (uint256)", | ||||
|     "function approve(address, uint256) external", | ||||
|     "function holderTokens(address) view returns (uint256[])", | ||||
|     "function getBond(uint256) view returns (tuple(address,uint256,uint256,uint256,uint256,uint256))" | ||||
|   ]; | ||||
|   let dsa; | ||||
|   let POOL3Contract; | ||||
|   let CRV3Contract; | ||||
|   let uAD3CRVfContract; | ||||
|   let uADContract; | ||||
|   let DAIContract; | ||||
|   let USDCContract; | ||||
|   let USDTContract; | ||||
|   let BONDContract; | ||||
|   let instaIndex; | ||||
|   let instaConnectorsV2; | ||||
|   let connector; | ||||
|   let instaImplementationsMapping; | ||||
|   let InstaAccountV2DefaultImpl; | ||||
| 
 | ||||
|   let uadWhale; | ||||
| 
 | ||||
|   const bondingShare = async function (address) { | ||||
|     let lpAmount = BigNumber.from(0); | ||||
|     let lpAmountTotal = BigNumber.from(0); | ||||
|     let bondId = -1; | ||||
| 
 | ||||
|     const bondIds = await BONDContract.holderTokens(address); | ||||
|     const bondN = bondIds?.length || 0; | ||||
| 
 | ||||
|     if (bondN) { | ||||
|       for await (_bondId of bondIds) { | ||||
|         lpAmountTotal = lpAmountTotal.add((await BONDContract.getBond(_bondId))[5]); | ||||
|       } | ||||
|       bondId = Number(bondIds[bondN - 1]); | ||||
|       lpAmount = (await BONDContract.getBond(bondId))[5]; | ||||
|     } | ||||
|     return { bondId, bondN, lpAmount, lpAmountTotal }; | ||||
|   }; | ||||
| 
 | ||||
|   const depositAndGetOneBond = async function () { | ||||
|     await dsaDepositUAD3CRVf(100); | ||||
|     dsa.cast( | ||||
|       ...encodeSpells([ | ||||
|         { | ||||
|           connector: ubiquityTest, | ||||
|           method: "deposit", | ||||
|           args: [UAD3CRVF, one.mul(100), 1, 0, 0] | ||||
|         } | ||||
|       ]), | ||||
|       uadWhaleAddress | ||||
|     ); | ||||
|   }; | ||||
| 
 | ||||
|   before(async () => { | ||||
|     // await forkReset(blockFork);
 | ||||
| 
 | ||||
|     [uadWhale] = await impersonate([uadWhaleAddress]); | ||||
|     const [ethWhale] = await impersonate([ethWhaleAddress]); | ||||
| 
 | ||||
|     await sendEth(ethWhale, uadWhaleAddress, 100); | ||||
|     POOL3Contract = new ethers.Contract(POOL3, ABI, uadWhale); | ||||
|     CRV3Contract = new ethers.Contract(CRV3, ABI, uadWhale); | ||||
|     uAD3CRVfContract = new ethers.Contract(UAD3CRVF, ABI, uadWhale); | ||||
|     uADContract = new ethers.Contract(UAD, ABI, uadWhale); | ||||
|     DAIContract = new ethers.Contract(DAI, ABI, uadWhale); | ||||
|     USDCContract = new ethers.Contract(USDC, ABI, uadWhale); | ||||
|     USDTContract = new ethers.Contract(USDT, ABI, uadWhale); | ||||
|     BONDContract = new ethers.Contract(BOND, ABI, uadWhale); | ||||
|     dsa = (await buildDSAv2(uadWhaleAddress)).connect(uadWhale); | ||||
|     await sendEth(ethWhale, dsa.address, 100); | ||||
|     await sendEth(ethWhale, uadWhaleAddress, 100); | ||||
| 
 | ||||
|     instaIndex = new ethers.Contract(addresses.core.instaIndex, abis.core.instaIndex, ethWhale); | ||||
| 
 | ||||
|     const masterAddress = await instaIndex.master(); | ||||
|     const [master] = await impersonate([masterAddress]); | ||||
|     await sendEth(ethWhale, masterAddress, 100); | ||||
| 
 | ||||
|     instaConnectorsV2 = new ethers.Contract(addresses.core.connectorsV2, abis.core.connectorsV2); | ||||
| 
 | ||||
|     instaImplementationsMapping = await ethers.getContractAt(implementationsABI, implementationsMappingAddr); | ||||
|     InstaAccountV2DefaultImpl = await ethers.getContractFactory("InstaDefaultImplementation"); | ||||
|     instaAccountV2DefaultImpl = await InstaAccountV2DefaultImpl.deploy(addresses.core.instaIndex); | ||||
|     await instaAccountV2DefaultImpl.deployed(); | ||||
|     await ( | ||||
|       await instaImplementationsMapping.connect(master).setDefaultImplementation(instaAccountV2DefaultImpl.address) | ||||
|     ).wait(); | ||||
| 
 | ||||
|     connector = await deployAndEnableConnector({ | ||||
|       connectorName: ubiquityTest, | ||||
|       contractArtifact: connectV2UbiquityArtifacts, | ||||
|       signer: master, | ||||
|       connectors: instaConnectorsV2 | ||||
|     }); | ||||
|   }); | ||||
| 
 | ||||
|   const logAll = async function () { | ||||
|     console.log("dsa            eth", utils.formatEther(await ethers.provider.getBalance(dsa.address))); | ||||
|     console.log("dsa            dai", utils.formatEther(await DAIContract.balanceOf(dsa.address))); | ||||
|     console.log("dsa           usdc", utils.formatUnits(await USDCContract.balanceOf(dsa.address), 6)); | ||||
|     console.log("dsa           usdt", utils.formatUnits(await USDTContract.balanceOf(dsa.address), 6)); | ||||
|     console.log("dsa            uad", utils.formatEther(await uADContract.balanceOf(dsa.address))); | ||||
|     console.log("dsa           3CRV", utils.formatEther(await CRV3Contract.balanceOf(dsa.address))); | ||||
|     console.log("dsa      uad3CRV-f", utils.formatEther(await uAD3CRVfContract.balanceOf(dsa.address))); | ||||
|     const { bondId, bondN, lpAmount, lpAmountTotal } = await bondingShare(dsa.address); | ||||
|     console.log("dsa        n bonds", utils.formatEther(lpAmountTotal), bondN); | ||||
|     console.log("dsa      last bond", utils.formatEther(lpAmount), bondId); | ||||
|   }; | ||||
| 
 | ||||
|   afterEach(logAll); | ||||
| 
 | ||||
|   const dsaDepositUAD3CRVf = async (amount) => { | ||||
|     await uAD3CRVfContract.transfer(dsa.address, one.mul(amount)); | ||||
|   }; | ||||
| 
 | ||||
|   const dsaDepositUAD = async (amount) => { | ||||
|     await uAD3CRVfContract.remove_liquidity_one_coin(one.mul(amount).mul(110).div(100), 0, one.mul(amount)); | ||||
|     await uADContract.transfer(dsa.address, one.mul(amount)); | ||||
|   }; | ||||
| 
 | ||||
|   const dsaDepositCRV3 = async (amount) => { | ||||
|     await uAD3CRVfContract.remove_liquidity_one_coin(one.mul(amount).mul(110).div(100), 1, one.mul(amount)); | ||||
|     await CRV3Contract.transfer(dsa.address, one.mul(amount)); | ||||
|   }; | ||||
| 
 | ||||
|   const dsaDepositDAI = async (amount) => { | ||||
|     await uAD3CRVfContract.remove_liquidity_one_coin( | ||||
|       one.mul(amount).mul(120).div(100), | ||||
|       1, | ||||
|       one.mul(amount).mul(110).div(100) | ||||
|     ); | ||||
|     await POOL3Contract.remove_liquidity_one_coin(one.mul(amount).mul(110).div(100), 0, one.mul(amount)); | ||||
|     await DAIContract.transfer(dsa.address, one.mul(amount)); | ||||
|   }; | ||||
|   const dsaDepositUSDC = async (amount) => { | ||||
|     await uAD3CRVfContract.remove_liquidity_one_coin( | ||||
|       one.mul(amount).mul(120).div(100), | ||||
|       1, | ||||
|       one.mul(amount).mul(110).div(100) | ||||
|     ); | ||||
|     await POOL3Contract.remove_liquidity_one_coin(one.mul(amount).mul(110).div(100), 1, onep.mul(amount)); | ||||
|     await USDCContract.transfer(dsa.address, onep.mul(amount)); | ||||
|   }; | ||||
|   const dsaDepositUSDT = async (amount) => { | ||||
|     await uAD3CRVfContract.remove_liquidity_one_coin( | ||||
|       one.mul(amount).mul(120).div(100), | ||||
|       1, | ||||
|       one.mul(amount).mul(110).div(100) | ||||
|     ); | ||||
|     await POOL3Contract.remove_liquidity_one_coin(one.mul(amount).mul(110).div(100), 2, onep.mul(amount)); | ||||
|     await USDTContract.transfer(dsa.address, onep.mul(amount)); | ||||
|   }; | ||||
| 
 | ||||
|   describe("Deposit", function () { | ||||
|     it("should deposit uAD3CRVf to get Ubiquity Bonding Shares", async function () { | ||||
|       await logAll(); | ||||
|       await dsaDepositUAD3CRVf(100); | ||||
|       expect((await bondingShare(dsa.address)).lpAmount).to.be.equal(0); | ||||
|       await expect( | ||||
|         dsa.cast( | ||||
|           ...encodeSpells([ | ||||
|             { | ||||
|               connector: ubiquityTest, | ||||
|               method: "deposit", | ||||
|               args: [UAD3CRVF, one.mul(100), 4, 0, 0] | ||||
|             } | ||||
|           ]), | ||||
|           uadWhaleAddress | ||||
|         ) | ||||
|       ).to.be.not.reverted; | ||||
|       expect((await bondingShare(dsa.address)).lpAmount).to.be.gt(0); | ||||
|     }); | ||||
| 
 | ||||
|     it("should deposit uAD to get Ubiquity Bonding Shares", async function () { | ||||
|       await dsaDepositUAD(100); | ||||
|       await expect( | ||||
|         dsa.cast( | ||||
|           ...encodeSpells([ | ||||
|             { | ||||
|               connector: ubiquityTest, | ||||
|               method: "deposit", | ||||
|               args: [UAD, one.mul(100), 4, 0, 0] | ||||
|             } | ||||
|           ]), | ||||
|           uadWhaleAddress | ||||
|         ) | ||||
|       ).to.be.not.reverted; | ||||
|       expect((await bondingShare(dsa.address)).lpAmount).to.be.gt(0); | ||||
|     }); | ||||
| 
 | ||||
|     it("should deposit 3CRV to get Ubiquity Bonding Shares", async function () { | ||||
|       await dsaDepositCRV3(100); | ||||
|       await expect( | ||||
|         dsa.cast( | ||||
|           ...encodeSpells([ | ||||
|             { | ||||
|               connector: ubiquityTest, | ||||
|               method: "deposit", | ||||
|               args: [CRV3, one.mul(100), 4, 0, 0] | ||||
|             } | ||||
|           ]), | ||||
|           uadWhaleAddress | ||||
|         ) | ||||
|       ).to.be.not.reverted; | ||||
|       expect((await bondingShare(dsa.address)).lpAmount).to.be.gt(0); | ||||
|     }); | ||||
| 
 | ||||
|     it("should deposit DAI to get Ubiquity Bonding Shares", async function () { | ||||
|       await dsaDepositDAI(100); | ||||
|       await expect( | ||||
|         dsa.cast( | ||||
|           ...encodeSpells([ | ||||
|             { | ||||
|               connector: ubiquityTest, | ||||
|               method: "deposit", | ||||
|               args: [DAI, one.mul(100), 4, 0, 0] | ||||
|             } | ||||
|           ]), | ||||
|           uadWhaleAddress | ||||
|         ) | ||||
|       ).to.be.not.reverted; | ||||
|       expect((await bondingShare(dsa.address)).lpAmount).to.be.gt(0); | ||||
|     }); | ||||
| 
 | ||||
|     it("should deposit USDC to get Ubiquity Bonding Shares", async function () { | ||||
|       await dsaDepositUSDC(100); | ||||
|       await expect( | ||||
|         dsa.cast( | ||||
|           ...encodeSpells([ | ||||
|             { | ||||
|               connector: ubiquityTest, | ||||
|               method: "deposit", | ||||
|               args: [USDC, onep.mul(100), 4, 0, 0] | ||||
|             } | ||||
|           ]), | ||||
|           uadWhaleAddress | ||||
|         ) | ||||
|       ).to.be.not.reverted; | ||||
|       expect((await bondingShare(dsa.address)).lpAmount).to.be.gt(0); | ||||
|     }); | ||||
| 
 | ||||
|     it("should deposit USDT to get Ubiquity Bonding Shares", async function () { | ||||
|       await dsaDepositUSDT(100); | ||||
|       await expect( | ||||
|         dsa.cast( | ||||
|           ...encodeSpells([ | ||||
|             { | ||||
|               connector: ubiquityTest, | ||||
|               method: "deposit", | ||||
|               args: [USDT, onep.mul(100), 4, 0, 0] | ||||
|             } | ||||
|           ]), | ||||
|           uadWhaleAddress | ||||
|         ) | ||||
|       ).to.be.not.reverted; | ||||
|       expect((await bondingShare(dsa.address)).lpAmount).to.be.gt(0); | ||||
|     }); | ||||
|   }); | ||||
| 
 | ||||
|   describe("Withdraw", function () { | ||||
|     let bondId = -1; | ||||
| 
 | ||||
|     before(async () => { | ||||
|       await depositAndGetOneBond(); | ||||
|       await depositAndGetOneBond(); | ||||
|       await depositAndGetOneBond(); | ||||
|       await depositAndGetOneBond(); | ||||
|       await depositAndGetOneBond(); | ||||
|       await depositAndGetOneBond(); | ||||
|       ({ bondId } = await bondingShare(dsa.address)); | ||||
| 
 | ||||
|       await logAll(); | ||||
|       console.log("Mining 50 000 blocks for more than one week, please wait..."); | ||||
|       await mineNBlock(50000); | ||||
|     }); | ||||
| 
 | ||||
|     it("Should deposit and withdraw DAI", async function () { | ||||
|       await expect( | ||||
|         dsa.cast( | ||||
|           ...encodeSpells([ | ||||
|             { | ||||
|               connector: ubiquityTest, | ||||
|               method: "withdraw", | ||||
|               args: [bondId, DAI, 0, 0] | ||||
|             } | ||||
|           ]), | ||||
|           uadWhaleAddress | ||||
|         ) | ||||
|       ).to.be.not.reverted; | ||||
|     }); | ||||
| 
 | ||||
|     it("Should deposit and withdraw USDC", async function () { | ||||
|       // await expect(
 | ||||
|       dsa.cast( | ||||
|         ...encodeSpells([ | ||||
|           { | ||||
|             connector: ubiquityTest, | ||||
|             method: "withdraw", | ||||
|             args: [bondId - 1, USDC, 0, 0] | ||||
|           } | ||||
|         ]), | ||||
|         uadWhaleAddress | ||||
|       ); | ||||
|       // ).to.be.not.reverted;
 | ||||
|     }); | ||||
| 
 | ||||
|     it("Should deposit and withdraw USDT", async function () { | ||||
|       await expect( | ||||
|         dsa.cast( | ||||
|           ...encodeSpells([ | ||||
|             { | ||||
|               connector: ubiquityTest, | ||||
|               method: "withdraw", | ||||
|               args: [bondId - 2, USDT, 0, 0] | ||||
|             } | ||||
|           ]), | ||||
|           uadWhaleAddress | ||||
|         ) | ||||
|       ).to.be.not.reverted; | ||||
|     }); | ||||
| 
 | ||||
|     it("Should deposit and withdraw UAD", async function () { | ||||
|       await expect( | ||||
|         dsa.cast( | ||||
|           ...encodeSpells([ | ||||
|             { | ||||
|               connector: ubiquityTest, | ||||
|               method: "withdraw", | ||||
|               args: [bondId - 3, UAD, 0, 0] | ||||
|             } | ||||
|           ]), | ||||
|           uadWhaleAddress | ||||
|         ) | ||||
|       ).to.be.not.reverted; | ||||
|     }); | ||||
| 
 | ||||
|     it("Should deposit and withdraw CRV3", async function () { | ||||
|       await expect( | ||||
|         dsa.cast( | ||||
|           ...encodeSpells([ | ||||
|             { | ||||
|               connector: ubiquityTest, | ||||
|               method: "withdraw", | ||||
|               args: [bondId - 4, CRV3, 0, 0] | ||||
|             } | ||||
|           ]), | ||||
|           uadWhaleAddress | ||||
|         ) | ||||
|       ).to.be.not.reverted; | ||||
|     }); | ||||
| 
 | ||||
|     it("Should deposit and withdraw UAD3CRVF", async function () { | ||||
|       await expect( | ||||
|         dsa.cast( | ||||
|           ...encodeSpells([ | ||||
|             { | ||||
|               connector: ubiquityTest, | ||||
|               method: "withdraw", | ||||
|               args: [bondId - 5, UAD3CRVF, 0, 0] | ||||
|             } | ||||
|           ]), | ||||
|           uadWhaleAddress | ||||
|         ) | ||||
|       ).to.be.not.reverted; | ||||
|     }); | ||||
|   }); | ||||
| 
 | ||||
|   describe("DSA wallet setup", function () { | ||||
|     it("Should have contracts deployed.", async function () { | ||||
|       expect(POOL3Contract.address).to.be.properAddress; | ||||
|       expect(CRV3Contract.address).to.be.properAddress; | ||||
|       expect(uADContract.address).to.be.properAddress; | ||||
|       expect(uAD3CRVfContract.address).to.be.properAddress; | ||||
|       expect(DAIContract.address).to.be.properAddress; | ||||
|       expect(USDCContract.address).to.be.properAddress; | ||||
|       expect(USDTContract.address).to.be.properAddress; | ||||
|       expect(BONDContract.address).to.be.properAddress; | ||||
|       expect(instaIndex.address).to.be.properAddress; | ||||
|       expect(instaConnectorsV2.address).to.be.properAddress; | ||||
|       expect(connector.address).to.be.properAddress; | ||||
|       expect(dsa.address).to.be.properAddress; | ||||
|     }); | ||||
|     it("Should deposit uAD3CRVf into DSA wallet", async function () { | ||||
|       await dsaDepositUAD3CRVf(100); | ||||
|       expect(await uAD3CRVfContract.balanceOf(dsa.address)).to.be.gte(one.mul(100)); | ||||
|     }); | ||||
|     it("Should deposit uAD into DSA wallet", async function () { | ||||
|       await dsaDepositUAD(100); | ||||
|       expect(await uADContract.balanceOf(dsa.address)).to.be.gte(one.mul(100)); | ||||
|     }); | ||||
|     it("Should deposit 3CRV into DSA wallet", async function () { | ||||
|       await dsaDepositCRV3(100); | ||||
|       expect(await CRV3Contract.balanceOf(dsa.address)).to.be.gte(one.mul(100)); | ||||
|     }); | ||||
|     it("Should deposit DAI into DSA wallet", async function () { | ||||
|       await dsaDepositDAI(100); | ||||
|       expect(await DAIContract.balanceOf(dsa.address)).to.be.gte(one.mul(100)); | ||||
|     }); | ||||
|     it("Should deposit USDC into DSA wallet", async function () { | ||||
|       await dsaDepositUSDC(100); | ||||
|       expect(await USDCContract.balanceOf(dsa.address)).to.be.gte(onep.mul(100)); | ||||
|     }); | ||||
|     it("Should deposit USDT into DSA wallet", async function () { | ||||
|       await dsaDepositUSDT(100); | ||||
|       expect(await USDTContract.balanceOf(dsa.address)).to.be.gte(onep.mul(100)); | ||||
|     }); | ||||
|   }); | ||||
| }); | ||||
							
								
								
									
										56
									
								
								test/ubiquity/utils.js
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										56
									
								
								test/ubiquity/utils.js
									
									
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,56 @@ | |||
| const hre = require("hardhat"); | ||||
| const hardhatConfig = require("../../hardhat.config"); | ||||
| 
 | ||||
| async function forkReset(blockNumber) { | ||||
|   await hre.network.provider.request({ | ||||
|     method: "hardhat_reset", | ||||
|     params: [ | ||||
|       { | ||||
|         forking: { | ||||
|           jsonRpcUrl: hardhatConfig.networks.hardhat.forking.url, | ||||
|           blockNumber | ||||
|         } | ||||
|       } | ||||
|     ] | ||||
|   }); | ||||
| } | ||||
| 
 | ||||
| async function mineBlock(timestamp) { | ||||
|   await network.provider.request({ | ||||
|     method: "evm_mine", | ||||
|     params: [timestamp] | ||||
|   }); | ||||
| } | ||||
| 
 | ||||
| async function sendEth(from, to, amount) { | ||||
|   await from.sendTransaction({ | ||||
|     to: to, | ||||
|     value: ethers.BigNumber.from(10).pow(18).mul(amount) | ||||
|   }); | ||||
| } | ||||
| 
 | ||||
| async function mineNBlock(blockCount, secondsBetweenBlock) { | ||||
|   const blockBefore = await ethers.provider.getBlock("latest"); | ||||
|   const maxMinedBlockPerBatch = 1000; | ||||
|   let blockToMine = blockCount; | ||||
|   let blockTime = blockBefore.timestamp; | ||||
|   while (blockToMine > maxMinedBlockPerBatch) { | ||||
|     // eslint-disable-next-line @typescript-eslint/no-loop-func
 | ||||
|     const minings = [...Array(maxMinedBlockPerBatch).keys()].map((_v, i) => { | ||||
|       const newTs = blockTime + i + (secondsBetweenBlock || 1); | ||||
|       return mineBlock(newTs); | ||||
|     }); | ||||
|     // eslint-disable-next-line no-await-in-loop
 | ||||
|     await Promise.all(minings); | ||||
|     blockToMine -= maxMinedBlockPerBatch; | ||||
|     blockTime = blockTime + maxMinedBlockPerBatch - 1 + maxMinedBlockPerBatch * (secondsBetweenBlock || 1); | ||||
|   } | ||||
|   const minings = [...Array(blockToMine).keys()].map((_v, i) => { | ||||
|     const newTs = blockTime + i + (secondsBetweenBlock || 1); | ||||
|     return mineBlock(newTs); | ||||
|   }); | ||||
|   // eslint-disable-next-line no-await-in-loop
 | ||||
|   await Promise.all(minings); | ||||
| } | ||||
| 
 | ||||
| module.exports = { forkReset, sendEth, mineNBlock }; | ||||
		Loading…
	
		Reference in New Issue
	
	Block a user
	 Thrilok kumar
						Thrilok kumar