const { expect } = require("chai"); const hre = require("hardhat"); const { web3, deployments, waffle, ethers } = hre; const { provider, deployContract } = waffle; const deployAndEnableConnector = require("../../scripts/deployAndEnableConnector.js"); const buildDSAv2 = require("../../scripts/buildDSAv2"); const encodeSpells = require("../../scripts/encodeSpells.js"); const encodeFlashcastData = require("../../scripts/encodeFlashcastData.js"); const getMasterSigner = require("../../scripts/getMasterSigner"); const addLiquidity = require("../../scripts/addLiquidity"); const addresses = require("../../scripts/constant/addresses"); const abis = require("../../scripts/constant/abis"); const constants = require("../../scripts/constant/constant"); const tokens = require("../../scripts/constant/tokens"); const { abi: nftManagerAbi, } = require("@uniswap/v3-periphery/artifacts/contracts/NonfungiblePositionManager.sol/NonfungiblePositionManager.json"); const connectV2UniswapV3Artifacts = require("../../artifacts/contracts/mainnet/connectors/uniswap/v3/main.sol/ConnectV2UniswapV3.json"); const { eth } = require("../../scripts/constant/tokens"); const { BigNumber } = require("ethers"); const FeeAmount = { LOW: 500, MEDIUM: 3000, HIGH: 10000, }; const TICK_SPACINGS = { 500: 10, 3000: 60, 10000: 200, }; const USDT_ADDR = "0xdac17f958d2ee523a2206206994597c13d831ec7"; const DAI_ADDR = "0x6b175474e89094c44da98b954eedeac495271d0f"; let tokenIds = []; let liquidities = []; const abiCoder = ethers.utils.defaultAbiCoder; describe("UniswapV3", function() { const connectorName = "UniswapV3-v1"; let dsaWallet0; let masterSigner; let instaConnectorsV2; let connector; let nftManager; const wallets = provider.getWallets(); const [wallet0, wallet1, wallet2, wallet3] = wallets; before(async () => { await hre.network.provider.request({ method: "hardhat_reset", params: [ { forking: { jsonRpcUrl: hre.config.networks.hardhat.forking.url, blockNumber: 13005785, }, }, ], }); masterSigner = await getMasterSigner(wallet3); instaConnectorsV2 = await ethers.getContractAt( abis.core.connectorsV2, addresses.core.connectorsV2 ); nftManager = await ethers.getContractAt( nftManagerAbi, "0xC36442b4a4522E871399CD717aBDD847Ab11FE88" ); connector = await deployAndEnableConnector({ connectorName, contractArtifact: connectV2UniswapV3Artifacts, signer: masterSigner, connectors: instaConnectorsV2, }); console.log("Connector address", connector.address); }); it("Should have contracts deployed.", async function() { expect(!!instaConnectorsV2.address).to.be.true; expect(!!connector.address).to.be.true; expect(!!masterSigner.address).to.be.true; }); describe("DSA wallet setup", function() { it("Should build DSA v2", async function() { dsaWallet0 = await buildDSAv2(wallet0.address); expect(!!dsaWallet0.address).to.be.true; }); it("Deposit ETH & DAI 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") ); await addLiquidity( "dai", dsaWallet0.address, ethers.utils.parseEther("100000") ); }); it("Deposit ETH & USDT 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") ); await addLiquidity( "usdt", dsaWallet0.address, ethers.utils.parseEther("100000") ); }); }); describe("Main", function() { it("Should mint successfully", async function() { const ethAmount = ethers.utils.parseEther("0.1"); // 1 ETH const daiAmount = ethers.utils.parseEther("400"); // 1 ETH const usdtAmount = ethers.utils.parseEther("400") / Math.pow(10, 12); // 1 ETH const ethAddress = "0xeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeee"; const getIds = ["0", "0"]; const setId = "0"; const spells = [ { connector: connectorName, method: "mint", args: [ ethAddress, DAI_ADDR, FeeAmount.MEDIUM, getMinTick(TICK_SPACINGS[FeeAmount.MEDIUM]), getMaxTick(TICK_SPACINGS[FeeAmount.MEDIUM]), ethAmount, daiAmount, "500000000000000000", getIds, setId, ], }, { connector: connectorName, method: "mint", args: [ DAI_ADDR, USDT_ADDR, FeeAmount.MEDIUM, getMinTick(TICK_SPACINGS[FeeAmount.MEDIUM]), getMaxTick(TICK_SPACINGS[FeeAmount.MEDIUM]), daiAmount, usdtAmount, "300000000000000000", getIds, setId, ], }, { connector: connectorName, method: "mint", args: [ ethAddress, USDT_ADDR, FeeAmount.MEDIUM, getMinTick(TICK_SPACINGS[FeeAmount.MEDIUM]), getMaxTick(TICK_SPACINGS[FeeAmount.MEDIUM]), ethAmount, usdtAmount, "300000000000000000", getIds, setId, ], }, ]; const tx = await dsaWallet0 .connect(wallet0) .cast(...encodeSpells(spells), wallet1.address); let receipt = await tx.wait(); let castEvent = new Promise((resolve, reject) => { dsaWallet0.on( "LogCast", ( origin, sender, value, targetNames, targets, eventNames, eventParams, event ) => { const params = abiCoder.decode( ["uint256", "uint256", "uint256", "uint256", "int24", "int24"], eventParams[0] ); const params1 = abiCoder.decode( ["uint256", "uint256", "uint256", "uint256", "int24", "int24"], eventParams[2] ); tokenIds.push(params[0]); tokenIds.push(params1[0]); liquidities.push(params[1]); event.removeListener(); resolve({ eventNames, }); } ); setTimeout(() => { reject(new Error("timeout")); }, 60000); }); let event = await castEvent; const data = await nftManager.positions(tokenIds[0]); expect(data.liquidity).to.be.equals(liquidities[0]); }).timeout(10000000000); it("Should deposit successfully", async function() { const daiAmount = ethers.utils.parseEther("400"); // 1 ETH const ethAmount = ethers.utils.parseEther("0.1"); // 1 ETH const usdtAmount = ethers.utils.parseEther("400") / Math.pow(10, 12); // 1 ETH const ethAddress = "0xeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeeee"; const getIds = ["0", "0"]; const setId = "0"; const spells = [ { connector: connectorName, method: "deposit", args: [ tokenIds[0], daiAmount, ethAmount, "500000000000000000", getIds, setId, ], }, ]; const tx = await dsaWallet0 .connect(wallet0) .cast(...encodeSpells(spells), wallet1.address); const receipt = await tx.wait(); let castEvent = new Promise((resolve, reject) => { dsaWallet0.on( "LogCast", ( origin, sender, value, targetNames, targets, eventNames, eventParams, event ) => { const params = abiCoder.decode( ["uint256", "uint256", "uint256", "uint256"], eventParams[0] ); liquidities[0] = liquidities[0].add(params[1]); event.removeListener(); resolve({ eventNames, }); } ); setTimeout(() => { reject(new Error("timeout")); }, 60000); }); let event = await castEvent; const data = await nftManager.positions(tokenIds[0]); expect(data.liquidity).to.be.equals(liquidities[0]); }); it("Should withdraw successfully", async function() { const getId = "0"; const setIds = ["0", "0"]; const data = await nftManager.positions(tokenIds[0]); let data1 = await nftManager.positions(tokenIds[1]); const spells = [ { connector: connectorName, method: "withdraw", args: [tokenIds[0], data.liquidity, 0, 0, getId, setIds], }, { connector: connectorName, method: "withdraw", args: [0, data1.liquidity, 0, 0, getId, setIds], }, ]; const tx = await dsaWallet0 .connect(wallet0) .cast(...encodeSpells(spells), wallet1.address); const receipt = await tx.wait(); data1 = await nftManager.positions(tokenIds[1]); expect(data1.liquidity.toNumber()).to.be.equals(0); }); it("Should collect successfully", async function() { const ethAmount = ethers.utils.parseEther("0.2"); // 1 ETH const daiAmount = ethers.utils.parseEther("800"); // 1 ETH const getIds = ["0", "0"]; const setIds = ["0", "0"]; const spells = [ { connector: connectorName, method: "collect", args: [tokenIds[0], daiAmount, ethAmount, getIds, setIds], }, ]; const tx = await dsaWallet0 .connect(wallet0) .cast(...encodeSpells(spells), wallet1.address); const receipt = await tx.wait(); }); it("Should burn successfully", async function() { const spells = [ { connector: connectorName, method: "burn", args: [tokenIds[0]], }, ]; const tx = await dsaWallet0 .connect(wallet0) .cast(...encodeSpells(spells), wallet1.address); const receipt = await tx.wait(); }); }); }); const getMinTick = (tickSpacing) => Math.ceil(-887272 / tickSpacing) * tickSpacing; const getMaxTick = (tickSpacing) => Math.floor(887272 / tickSpacing) * tickSpacing;