2020-10-19 16:17:27 +00:00
|
|
|
// running `npx hardhat test` automatically makes use of hardhat-waffle plugin
|
2020-08-17 10:17:04 +00:00
|
|
|
// => only dependency we need is "chai"
|
2020-10-15 18:04:08 +00:00
|
|
|
const {expect} = require("chai");
|
2020-10-19 16:17:27 +00:00
|
|
|
const hre = require("hardhat");
|
|
|
|
const {ethers} = hre;
|
2020-08-17 10:17:04 +00:00
|
|
|
const GelatoCoreLib = require("@gelatonetwork/core");
|
2020-10-07 17:45:46 +00:00
|
|
|
//const { sleep } = GelatoCoreLib;
|
2020-08-17 10:17:04 +00:00
|
|
|
|
|
|
|
// Constants
|
|
|
|
const ETH = "0xEeeeeEeeeEeEeeEeEeEeeEEEeeeeEeeeeeeeEEeE";
|
|
|
|
|
|
|
|
// Contracts
|
|
|
|
const InstaIndex = require("../pre-compiles/InstaIndex.json");
|
|
|
|
const InstaList = require("../pre-compiles/InstaList.json");
|
|
|
|
const InstaConnectors = require("../pre-compiles/InstaConnectors.json");
|
|
|
|
const InstaAccount = require("../pre-compiles/InstaAccount.json");
|
|
|
|
const ConnectAuth = require("../pre-compiles/ConnectAuth.json");
|
|
|
|
const ConnectBasic = require("../pre-compiles/ConnectBasic.json");
|
2020-09-23 10:27:33 +00:00
|
|
|
const ProviderModuleDSA_ABI = require("../pre-compiles/ProviderModuleDSA_ABI.json");
|
2020-08-17 10:17:04 +00:00
|
|
|
|
|
|
|
describe("DSA setup with Gelato Tests", function () {
|
|
|
|
this.timeout(50000);
|
2020-10-19 16:17:27 +00:00
|
|
|
if (hre.network.name !== "hardhat") {
|
|
|
|
console.error("Test Suite is meant to be run on hardhat only");
|
2020-08-17 10:17:04 +00:00
|
|
|
process.exit(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Wallet to use for local testing
|
|
|
|
let userWallet;
|
|
|
|
let userAddress;
|
|
|
|
let dsaAddress;
|
|
|
|
|
|
|
|
// Deployed instances
|
|
|
|
let instaIndex;
|
|
|
|
let instaList;
|
|
|
|
let instaConnectors;
|
|
|
|
let instaAccount;
|
|
|
|
let gelatoCore;
|
2020-09-23 10:27:33 +00:00
|
|
|
let providerModuleDSA;
|
2020-08-17 10:17:04 +00:00
|
|
|
|
|
|
|
// Contracts to deploy and use for local testing
|
|
|
|
let dsa;
|
|
|
|
|
|
|
|
// Other variables
|
|
|
|
let dsaVersion;
|
|
|
|
let dsaID;
|
|
|
|
|
|
|
|
before(async function () {
|
|
|
|
// Get Test Wallet for local testnet
|
|
|
|
[userWallet] = await ethers.getSigners();
|
|
|
|
userAddress = await userWallet.getAddress();
|
|
|
|
|
|
|
|
// ===== DSA LOCAL SETUP ==================
|
|
|
|
instaIndex = await ethers.getContractAt(
|
|
|
|
InstaIndex.abi,
|
2020-10-19 16:17:27 +00:00
|
|
|
hre.network.config.InstaIndex
|
2020-08-17 10:17:04 +00:00
|
|
|
);
|
|
|
|
instaList = await ethers.getContractAt(
|
|
|
|
InstaList.abi,
|
2020-10-19 16:17:27 +00:00
|
|
|
hre.network.config.InstaList
|
2020-08-17 10:17:04 +00:00
|
|
|
);
|
|
|
|
instaConnectors = await ethers.getContractAt(
|
|
|
|
InstaConnectors.abi,
|
2020-10-19 16:17:27 +00:00
|
|
|
hre.network.config.InstaConnectors
|
2020-08-17 10:17:04 +00:00
|
|
|
);
|
|
|
|
instaAccount = await ethers.getContractAt(
|
|
|
|
InstaAccount.abi,
|
2020-10-19 16:17:27 +00:00
|
|
|
hre.network.config.InstaAccount
|
2020-08-17 10:17:04 +00:00
|
|
|
);
|
|
|
|
|
|
|
|
dsaVersion = await instaAccount.version();
|
|
|
|
dsaID = await instaList.accounts();
|
|
|
|
|
|
|
|
// Deploy DSA and get and verify ID of newly deployed DSA
|
|
|
|
await expect(instaIndex.build(userAddress, 1, userAddress)).to.emit(
|
|
|
|
instaIndex,
|
|
|
|
"LogAccountCreated"
|
|
|
|
);
|
|
|
|
await expect(await instaList.accounts()).to.be.equal(dsaID.add(1));
|
|
|
|
dsaID = dsaID.add(1);
|
|
|
|
|
|
|
|
// Instantiate the DSA
|
|
|
|
dsaAddress = await instaList.accountAddr(dsaID);
|
|
|
|
dsa = await ethers.getContractAt(InstaAccount.abi, dsaAddress);
|
|
|
|
|
|
|
|
// ===== GELATO LOCAL SETUP ==================
|
|
|
|
gelatoCore = await ethers.getContractAt(
|
|
|
|
GelatoCoreLib.GelatoCore.abi,
|
2020-10-19 16:17:27 +00:00
|
|
|
hre.network.config.GelatoCore
|
2020-08-17 10:17:04 +00:00
|
|
|
);
|
2020-09-23 10:27:33 +00:00
|
|
|
providerModuleDSA = await ethers.getContractAt(
|
|
|
|
ProviderModuleDSA_ABI,
|
2020-10-19 16:17:27 +00:00
|
|
|
hre.network.config.ProviderModuleDSA
|
2020-08-17 10:17:04 +00:00
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
it("#1: Forks InstaDapp Mainnet config", async function () {
|
|
|
|
expect(await instaIndex.list()).to.be.equal(instaList.address);
|
|
|
|
expect(dsaVersion).to.be.equal(1);
|
|
|
|
expect(await instaIndex.connectors(dsaVersion)).to.be.equal(
|
|
|
|
instaConnectors.address
|
|
|
|
);
|
2020-10-19 16:17:27 +00:00
|
|
|
expect(await instaConnectors.connectors(hre.network.config.ConnectAuth)).to
|
2020-08-17 10:17:04 +00:00
|
|
|
.be.true;
|
2020-10-19 16:17:27 +00:00
|
|
|
expect(await instaConnectors.connectors(hre.network.config.ConnectBasic)).to
|
2020-08-17 10:17:04 +00:00
|
|
|
.be.true;
|
2020-10-19 16:17:27 +00:00
|
|
|
expect(await instaConnectors.connectors(hre.network.config.ConnectMaker)).to
|
2020-08-17 10:17:04 +00:00
|
|
|
.be.true;
|
2020-10-19 16:17:27 +00:00
|
|
|
expect(await instaConnectors.connectors(hre.network.config.ConnectCompound))
|
2020-08-17 10:17:04 +00:00
|
|
|
.to.be.true;
|
|
|
|
});
|
|
|
|
|
|
|
|
it("#2: Deploys a DSA with user as authority", async function () {
|
|
|
|
expect(await dsa.isAuth(userAddress)).to.be.true;
|
|
|
|
});
|
|
|
|
|
|
|
|
it("#3: Let's User deposit and withdraw funds from DSA", async function () {
|
|
|
|
// Send withdraw TX via DSA.cast delegatecall
|
|
|
|
const gasLimit = ethers.BigNumber.from(1000000);
|
|
|
|
const gasPrice = ethers.utils.parseUnits("20", "gwei");
|
|
|
|
const gasCostMax = gasLimit.mul(gasPrice);
|
|
|
|
|
|
|
|
// Deposit funds into DSA
|
|
|
|
const initialWalletBalance = await userWallet.getBalance();
|
|
|
|
expect(await ethers.provider.getBalance(dsaAddress)).to.be.equal(0);
|
|
|
|
await userWallet.sendTransaction({
|
|
|
|
to: dsaAddress,
|
|
|
|
value: ethers.utils.parseEther("1"),
|
|
|
|
gasLimit,
|
|
|
|
gasPrice,
|
|
|
|
});
|
|
|
|
expect(await userWallet.getBalance()).to.be.lt(
|
|
|
|
initialWalletBalance.sub(ethers.utils.parseEther("1"))
|
|
|
|
);
|
|
|
|
expect(await ethers.provider.getBalance(dsaAddress)).to.be.equal(
|
|
|
|
ethers.utils.parseEther("1")
|
|
|
|
);
|
|
|
|
|
|
|
|
// Encode Payloads for ConnectBasic.withdraw
|
2020-10-19 16:17:27 +00:00
|
|
|
const withdrawData = await hre.run("abi-encode-withselector", {
|
2020-08-17 10:17:04 +00:00
|
|
|
abi: ConnectBasic.abi,
|
|
|
|
functionname: "withdraw",
|
|
|
|
inputs: [ETH, ethers.utils.parseEther("1"), userAddress, 0, 0],
|
|
|
|
});
|
|
|
|
|
|
|
|
await expect(
|
2020-10-19 16:17:27 +00:00
|
|
|
dsa.cast([hre.network.config.ConnectBasic], [withdrawData], userAddress, {
|
2020-08-17 10:17:04 +00:00
|
|
|
gasLimit,
|
|
|
|
gasPrice,
|
|
|
|
})
|
|
|
|
)
|
|
|
|
.to.emit(dsa, "LogCast")
|
|
|
|
.withArgs(userAddress, userAddress, 0);
|
|
|
|
|
|
|
|
expect(await ethers.provider.getBalance(dsaAddress)).to.be.equal(0);
|
|
|
|
expect(await userWallet.getBalance()).to.be.gte(
|
|
|
|
initialWalletBalance.sub(gasCostMax.mul(2))
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
it("#4: Enables GelatoCore as a User of the DSA", async function () {
|
|
|
|
expect(await dsa.isAuth(gelatoCore.address)).to.be.false;
|
|
|
|
|
|
|
|
// Encode Payloads for ConnectAuth.addModule
|
2020-10-19 16:17:27 +00:00
|
|
|
const addAuthData = await hre.run("abi-encode-withselector", {
|
2020-08-17 10:17:04 +00:00
|
|
|
abi: ConnectAuth.abi,
|
|
|
|
functionname: "add",
|
|
|
|
inputs: [gelatoCore.address],
|
|
|
|
});
|
|
|
|
|
|
|
|
await expect(
|
2020-10-19 16:17:27 +00:00
|
|
|
dsa.cast([hre.network.config.ConnectAuth], [addAuthData], userAddress)
|
2020-08-17 10:17:04 +00:00
|
|
|
)
|
|
|
|
.to.emit(dsa, "LogCast")
|
|
|
|
.withArgs(userAddress, userAddress, 0);
|
|
|
|
|
|
|
|
expect(await dsa.isAuth(gelatoCore.address)).to.be.true;
|
|
|
|
});
|
|
|
|
|
2020-09-23 10:27:33 +00:00
|
|
|
it("#5: ConnectGelato is deployed and whitelisted on mainnet", async function () {
|
|
|
|
expect(
|
2020-10-19 16:17:27 +00:00
|
|
|
await instaConnectors.isConnector([hre.network.config.ConnectGelato])
|
2020-09-23 10:27:33 +00:00
|
|
|
).to.be.true;
|
2020-08-17 10:17:04 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
it("#6: Gelato ProviderModuleDSA returns correct execPayload", async function () {
|
|
|
|
// Deposit 1 ETH into DSA
|
|
|
|
await userWallet.sendTransaction({
|
|
|
|
to: dsaAddress,
|
|
|
|
value: ethers.utils.parseEther("1"),
|
|
|
|
});
|
|
|
|
expect(await ethers.provider.getBalance(dsaAddress)).to.be.equal(
|
|
|
|
ethers.utils.parseEther("1")
|
|
|
|
);
|
|
|
|
|
|
|
|
// We withdraw to otherWallet to ignore gasUsed during test
|
2020-10-15 18:04:08 +00:00
|
|
|
const {1: otherWallet} = await ethers.getSigners();
|
2020-08-17 10:17:04 +00:00
|
|
|
|
|
|
|
// Instantiate Gelato ConnectBasic.withdraw Task
|
|
|
|
const withdrawFromDSATask = new GelatoCoreLib.Task({
|
|
|
|
actions: [
|
|
|
|
new GelatoCoreLib.Action({
|
2020-10-19 16:17:27 +00:00
|
|
|
addr: hre.network.config.ConnectBasic,
|
|
|
|
data: await hre.run("abi-encode-withselector", {
|
2020-08-17 10:17:04 +00:00
|
|
|
abi: ConnectBasic.abi,
|
|
|
|
functionname: "withdraw",
|
|
|
|
inputs: [
|
|
|
|
ETH,
|
|
|
|
ethers.utils.parseEther("1"),
|
|
|
|
await otherWallet.getAddress(),
|
|
|
|
0,
|
|
|
|
0,
|
|
|
|
],
|
|
|
|
}),
|
|
|
|
operation: GelatoCoreLib.Operation.Delegatecall, // placeholder
|
|
|
|
}),
|
|
|
|
],
|
|
|
|
});
|
|
|
|
|
|
|
|
// otherWallet needs to be an authority to qualify as withdraw to address.
|
2020-10-19 16:17:27 +00:00
|
|
|
const addAuthData = await hre.run("abi-encode-withselector", {
|
2020-08-17 10:17:04 +00:00
|
|
|
abi: ConnectAuth.abi,
|
|
|
|
functionname: "add",
|
|
|
|
inputs: [await otherWallet.getAddress()],
|
|
|
|
});
|
|
|
|
await dsa.cast(
|
2020-10-19 16:17:27 +00:00
|
|
|
[hre.network.config.ConnectAuth],
|
2020-08-17 10:17:04 +00:00
|
|
|
[addAuthData],
|
|
|
|
userAddress
|
|
|
|
);
|
|
|
|
|
|
|
|
const [execPayload] = await providerModuleDSA.execPayload(
|
|
|
|
0, // placeholder
|
|
|
|
ethers.constants.AddressZero, // placeholder
|
|
|
|
ethers.constants.AddressZero, // placeholder
|
|
|
|
withdrawFromDSATask,
|
|
|
|
0 // placeholder
|
|
|
|
);
|
|
|
|
|
|
|
|
await expect(() =>
|
|
|
|
userWallet.sendTransaction({
|
|
|
|
to: dsaAddress,
|
|
|
|
data: execPayload,
|
|
|
|
})
|
|
|
|
).to.changeBalance(otherWallet, ethers.utils.parseEther("1"));
|
|
|
|
expect(await ethers.provider.getBalance(dsaAddress)).to.be.equal(0);
|
|
|
|
});
|
|
|
|
});
|