mirror of
https://github.com/Instadapp/aave-protocol-v2.git
synced 2024-07-29 21:47:30 +00:00
632 lines
18 KiB
TypeScript
632 lines
18 KiB
TypeScript
import rawBRE from "@nomiclabs/buidler";
|
|
import {MockContract} from "ethereum-waffle";
|
|
import {
|
|
deployLendingPoolAddressesProvider,
|
|
deployMintableErc20,
|
|
deployLendingPoolAddressesProviderRegistry,
|
|
deployFeeProvider,
|
|
deployLendingPoolConfigurator,
|
|
deployLendingPool,
|
|
deployPriceOracle,
|
|
getLendingPoolConfiguratorProxy,
|
|
deployMockAggregator,
|
|
deployChainlinkProxyPriceProvider,
|
|
deployLendingRateOracle,
|
|
deployDefaultReserveInterestRateStrategy,
|
|
deployLendingPoolLiquidationManager,
|
|
deployMockOneSplit,
|
|
deployOneSplitAdapter,
|
|
deployTokenDistributor,
|
|
deployInitializableAdminUpgradeabilityProxy,
|
|
deployMockFlashLoanReceiver,
|
|
deployWalletBalancerProvider,
|
|
getFeeProvider,
|
|
getLendingPool,
|
|
insertContractAddressInDb,
|
|
deployAaveProtocolTestHelpers,
|
|
getEthersSigners,
|
|
registerContractInJsonDb,
|
|
} from "../helpers/contracts-helpers";
|
|
import {LendingPoolAddressesProvider} from "../types/LendingPoolAddressesProvider";
|
|
import {Wallet, ContractTransaction, ethers, Signer} from "ethers";
|
|
import {
|
|
TokenContractId,
|
|
eContractid,
|
|
iAssetBase,
|
|
tEthereumAddress,
|
|
iAssetAggregatorBase,
|
|
IMarketRates,
|
|
iMultiPoolsAssets,
|
|
AavePools,
|
|
IReserveParams,
|
|
} from "../helpers/types";
|
|
import {MintableErc20} from "../types/MintableErc20";
|
|
import {
|
|
MOCK_USD_PRICE_IN_WEI,
|
|
ALL_ASSETS_INITIAL_PRICES,
|
|
MOCK_ETH_ADDRESS,
|
|
USD_ADDRESS,
|
|
MOCK_CHAINLINK_AGGREGATORS_PRICES,
|
|
LENDING_RATE_ORACLE_RATES_COMMON,
|
|
getReservesConfigByPool,
|
|
getFeeDistributionParamsCommon,
|
|
ZERO_ADDRESS,
|
|
} from "../helpers/constants";
|
|
import {PriceOracle} from "../types/PriceOracle";
|
|
import {MockAggregator} from "../types/MockAggregator";
|
|
import {LendingRateOracle} from "../types/LendingRateOracle";
|
|
import {LendingPool} from "../types/LendingPool";
|
|
import {LendingPoolConfigurator} from "../types/LendingPoolConfigurator";
|
|
import { initializeMakeSuite } from './helpers/make-suite';
|
|
|
|
const deployAllMockTokens = async (deployer: Signer) => {
|
|
const tokens: {[symbol: string]: MockContract | MintableErc20} = {};
|
|
|
|
|
|
const protoConfigData = getReservesConfigByPool(AavePools.proto);
|
|
const secondaryConfigData = getReservesConfigByPool(AavePools.secondary);
|
|
|
|
|
|
for (const tokenSymbol of Object.keys(TokenContractId)) {
|
|
|
|
if (tokenSymbol !== "ETH") {
|
|
|
|
let decimals = 18;
|
|
|
|
let configData = (<any>protoConfigData)[tokenSymbol];
|
|
|
|
if(!configData){
|
|
configData = (<any>secondaryConfigData)[tokenSymbol]
|
|
}
|
|
|
|
if(!configData){
|
|
decimals = 18;
|
|
}
|
|
|
|
tokens[tokenSymbol] = await deployMintableErc20([
|
|
tokenSymbol,
|
|
tokenSymbol,
|
|
configData ? configData.reserveDecimals : 18,
|
|
]);
|
|
await registerContractInJsonDb(
|
|
tokenSymbol.toUpperCase(),
|
|
tokens[tokenSymbol]
|
|
);
|
|
}
|
|
}
|
|
|
|
return tokens;
|
|
};
|
|
|
|
const setInitialAssetPricesInOracle = async (
|
|
prices: iAssetBase<tEthereumAddress>,
|
|
assetsAddresses: iAssetBase<tEthereumAddress>,
|
|
priceOracleInstance: PriceOracle
|
|
) => {
|
|
for (const [assetSymbol, price] of Object.entries(prices) as [
|
|
string,
|
|
string
|
|
][]) {
|
|
const assetAddressIndex = Object.keys(assetsAddresses).findIndex(
|
|
(value) => value === assetSymbol
|
|
);
|
|
const [, assetAddress] = (Object.entries(assetsAddresses) as [
|
|
string,
|
|
string
|
|
][])[assetAddressIndex];
|
|
await waitForTx(
|
|
await priceOracleInstance.setAssetPrice(assetAddress, price)
|
|
);
|
|
}
|
|
};
|
|
|
|
const deployAllMockAggregators = async (
|
|
initialPrices: iAssetAggregatorBase<string>
|
|
) => {
|
|
const aggregators: {[tokenSymbol: string]: MockAggregator} = {};
|
|
for (const tokenContractName of Object.keys(initialPrices)) {
|
|
if (tokenContractName !== "ETH") {
|
|
const priceIndex = Object.keys(initialPrices).findIndex(
|
|
(value) => value === tokenContractName
|
|
);
|
|
const [, price] = (Object.entries(initialPrices) as [string, string][])[
|
|
priceIndex
|
|
];
|
|
aggregators[tokenContractName] = await deployMockAggregator(price);
|
|
}
|
|
}
|
|
return aggregators;
|
|
};
|
|
|
|
const getPairsTokenAggregator = (
|
|
allAssetsAddresses: {
|
|
[tokenSymbol: string]: tEthereumAddress;
|
|
},
|
|
aggregatorsAddresses: {[tokenSymbol: string]: tEthereumAddress}
|
|
): [string[], string[]] => {
|
|
const {ETH, ...assetsAddressesWithoutEth} = allAssetsAddresses;
|
|
|
|
const pairs = Object.entries(assetsAddressesWithoutEth).map(
|
|
([tokenSymbol, tokenAddress]) => {
|
|
if (tokenSymbol !== "ETH") {
|
|
const aggregatorAddressIndex = Object.keys(
|
|
aggregatorsAddresses
|
|
).findIndex((value) => value === tokenSymbol);
|
|
const [, aggregatorAddress] = (Object.entries(aggregatorsAddresses) as [
|
|
string,
|
|
tEthereumAddress
|
|
][])[aggregatorAddressIndex];
|
|
return [tokenAddress, aggregatorAddress];
|
|
}
|
|
}
|
|
);
|
|
|
|
const mappedPairs = pairs.map(([asset]) => asset);
|
|
const mappedAggregators = pairs.map(([, source]) => source);
|
|
|
|
return [mappedPairs, mappedAggregators];
|
|
};
|
|
|
|
const setInitialMarketRatesInRatesOracle = async (
|
|
marketRates: iMultiPoolsAssets<IMarketRates>,
|
|
assetsAddresses: {[x: string]: tEthereumAddress},
|
|
lendingRateOracleInstance: LendingRateOracle
|
|
) => {
|
|
for (const [assetSymbol, {borrowRate}] of Object.entries(marketRates) as [
|
|
string,
|
|
IMarketRates
|
|
][]) {
|
|
const assetAddressIndex = Object.keys(assetsAddresses).findIndex(
|
|
(value) => value === assetSymbol
|
|
);
|
|
const [, assetAddress] = (Object.entries(assetsAddresses) as [
|
|
string,
|
|
string
|
|
][])[assetAddressIndex];
|
|
await lendingRateOracleInstance.setMarketBorrowRate(
|
|
assetAddress,
|
|
borrowRate
|
|
);
|
|
}
|
|
};
|
|
|
|
const initReserves = async (
|
|
reservesParams: iMultiPoolsAssets<IReserveParams>,
|
|
tokenAddresses: {[symbol: string]: tEthereumAddress},
|
|
lendingPoolAddressesProvider: LendingPoolAddressesProvider,
|
|
lendingPool: LendingPool,
|
|
lendingPoolConfigurator: LendingPoolConfigurator,
|
|
aavePool: AavePools
|
|
) => {
|
|
if (aavePool !== AavePools.proto && aavePool !== AavePools.secondary) {
|
|
console.log(`Invalid Aave pool ${aavePool}`);
|
|
process.exit(1);
|
|
}
|
|
|
|
for (let [assetSymbol, {reserveDecimals}] of Object.entries(
|
|
reservesParams
|
|
) as [string, IReserveParams][]) {
|
|
const assetAddressIndex = Object.keys(tokenAddresses).findIndex(
|
|
(value) => value === assetSymbol
|
|
);
|
|
const [, tokenAddress] = (Object.entries(tokenAddresses) as [
|
|
string,
|
|
string
|
|
][])[assetAddressIndex];
|
|
|
|
const {isActive: reserveInitialized} = await lendingPool.getReserveConfigurationData(
|
|
tokenAddress
|
|
);
|
|
|
|
if (reserveInitialized) {
|
|
console.log(
|
|
`Reserve ${assetSymbol} is already active, skipping configuration`
|
|
);
|
|
continue;
|
|
}
|
|
|
|
try {
|
|
const reserveParamIndex = Object.keys(reservesParams).findIndex(
|
|
(value) => value === assetSymbol
|
|
);
|
|
const [
|
|
,
|
|
{
|
|
baseVariableBorrowRate,
|
|
variableRateSlope1,
|
|
variableRateSlope2,
|
|
stableRateSlope1,
|
|
stableRateSlope2,
|
|
},
|
|
] = (Object.entries(reservesParams) as [string, IReserveParams][])[
|
|
reserveParamIndex
|
|
];
|
|
const rateStrategyContract = await deployDefaultReserveInterestRateStrategy(
|
|
[
|
|
lendingPoolAddressesProvider.address,
|
|
baseVariableBorrowRate,
|
|
variableRateSlope1,
|
|
variableRateSlope2,
|
|
stableRateSlope1,
|
|
stableRateSlope2,
|
|
]
|
|
);
|
|
|
|
if (process.env.POOL === AavePools.secondary) {
|
|
if (assetSymbol.search("UNI") === -1) {
|
|
assetSymbol = `Uni${assetSymbol}`;
|
|
} else {
|
|
assetSymbol = assetSymbol.replace(/_/g, "").replace("UNI", "Uni");
|
|
}
|
|
}
|
|
|
|
await lendingPoolConfigurator.initReserveWithData(
|
|
tokenAddress,
|
|
`Aave Interest bearing ${assetSymbol}`,
|
|
`a${assetSymbol}`,
|
|
reserveDecimals,
|
|
rateStrategyContract.address
|
|
);
|
|
} catch (e) {
|
|
console.log(
|
|
`Reserve initialization for ${assetSymbol} failed with error ${e}. Skipped.`
|
|
);
|
|
}
|
|
}
|
|
};
|
|
|
|
const enableReservesToBorrow = async (
|
|
reservesParams: iMultiPoolsAssets<IReserveParams>,
|
|
tokenAddresses: {[symbol: string]: tEthereumAddress},
|
|
lendingPool: LendingPool,
|
|
lendingPoolConfigurator: LendingPoolConfigurator
|
|
) => {
|
|
for (const [
|
|
assetSymbol,
|
|
{borrowingEnabled, stableBorrowRateEnabled},
|
|
] of Object.entries(reservesParams) as [string, IReserveParams][]) {
|
|
if (!borrowingEnabled) continue;
|
|
try {
|
|
const assetAddressIndex = Object.keys(tokenAddresses).findIndex(
|
|
(value) => value === assetSymbol
|
|
);
|
|
const [, tokenAddress] = (Object.entries(tokenAddresses) as [
|
|
string,
|
|
string
|
|
][])[assetAddressIndex];
|
|
const {borrowingEnabled: borrowingAlreadyEnabled} = await lendingPool.getReserveConfigurationData(
|
|
tokenAddress
|
|
);
|
|
|
|
if (borrowingAlreadyEnabled) {
|
|
console.log(
|
|
`Reserve ${assetSymbol} is already enabled for borrowing, skipping`
|
|
);
|
|
continue;
|
|
}
|
|
|
|
await lendingPoolConfigurator.enableBorrowingOnReserve(
|
|
tokenAddress,
|
|
stableBorrowRateEnabled
|
|
);
|
|
} catch (e) {
|
|
console.log(
|
|
`Enabling reserve for borrowings for ${assetSymbol} failed with error ${e}. Skipped.`
|
|
);
|
|
}
|
|
}
|
|
};
|
|
|
|
const enableReservesAsCollateral = async (
|
|
reservesParams: iMultiPoolsAssets<IReserveParams>,
|
|
tokenAddresses: {[symbol: string]: tEthereumAddress},
|
|
lendingPool: LendingPool,
|
|
lendingPoolConfigurator: LendingPoolConfigurator
|
|
) => {
|
|
for (const [
|
|
assetSymbol,
|
|
{baseLTVAsCollateral, liquidationBonus, liquidationThreshold},
|
|
] of Object.entries(reservesParams) as [string, IReserveParams][]) {
|
|
if (baseLTVAsCollateral === "-1") continue;
|
|
|
|
const assetAddressIndex = Object.keys(tokenAddresses).findIndex(
|
|
(value) => value === assetSymbol
|
|
);
|
|
const [, tokenAddress] = (Object.entries(tokenAddresses) as [
|
|
string,
|
|
string
|
|
][])[assetAddressIndex];
|
|
const {usageAsCollateralEnabled: alreadyEnabled} = await lendingPool.getReserveConfigurationData(
|
|
tokenAddress
|
|
);
|
|
|
|
if (alreadyEnabled) {
|
|
console.log(
|
|
`Reserve ${assetSymbol} is already enabled as collateral, skipping`
|
|
);
|
|
continue;
|
|
}
|
|
|
|
try {
|
|
await lendingPoolConfigurator.enableReserveAsCollateral(
|
|
tokenAddress,
|
|
baseLTVAsCollateral,
|
|
liquidationThreshold,
|
|
liquidationBonus
|
|
);
|
|
} catch (e) {
|
|
console.log(
|
|
`Enabling reserve as collateral for ${assetSymbol} failed with error ${e}. Skipped.`
|
|
);
|
|
}
|
|
}
|
|
};
|
|
|
|
export const waitForTx = async (tx: ContractTransaction) => await tx.wait();
|
|
|
|
const buildTestEnv = async (deployer: Signer, secondaryWallet: Signer) => {
|
|
console.time("setup");
|
|
const lendingPoolManager = await deployer.getAddress();
|
|
|
|
const mockTokens = await deployAllMockTokens(deployer);
|
|
|
|
const addressesProvider = await deployLendingPoolAddressesProvider();
|
|
await waitForTx(
|
|
await addressesProvider.setLendingPoolManager(lendingPoolManager)
|
|
);
|
|
|
|
const addressesProviderRegistry = await deployLendingPoolAddressesProviderRegistry();
|
|
await waitForTx(
|
|
await addressesProviderRegistry.registerAddressesProvider(
|
|
addressesProvider.address,
|
|
0
|
|
)
|
|
);
|
|
|
|
const feeProviderImpl = await deployFeeProvider();
|
|
await waitForTx(
|
|
await addressesProvider.setFeeProviderImpl(feeProviderImpl.address)
|
|
);
|
|
const feeProviderProxy = await getFeeProvider(
|
|
await addressesProvider.getFeeProvider()
|
|
);
|
|
await insertContractAddressInDb(
|
|
eContractid.FeeProvider,
|
|
feeProviderProxy.address
|
|
);
|
|
|
|
const lendingPoolConfiguratorImpl = await deployLendingPoolConfigurator();
|
|
await waitForTx(
|
|
await addressesProvider.setLendingPoolConfiguratorImpl(
|
|
lendingPoolConfiguratorImpl.address
|
|
)
|
|
);
|
|
const lendingPoolConfiguratorProxy = await getLendingPoolConfiguratorProxy(
|
|
await addressesProvider.getLendingPoolConfigurator()
|
|
);
|
|
await insertContractAddressInDb(
|
|
eContractid.LendingPoolConfigurator,
|
|
lendingPoolConfiguratorProxy.address
|
|
);
|
|
|
|
|
|
const lendingPoolImpl = await deployLendingPool();
|
|
|
|
console.log("Deployed lending pool, address:", lendingPoolImpl.address)
|
|
await waitForTx(
|
|
await addressesProvider.setLendingPoolImpl(lendingPoolImpl.address)
|
|
);
|
|
|
|
console.log("Added pool to addresses provider")
|
|
|
|
const address = await addressesProvider.getLendingPool()
|
|
console.log("Address is ", address)
|
|
const lendingPoolProxy = await getLendingPool(
|
|
address
|
|
);
|
|
|
|
console.log("implementation set, address:", lendingPoolProxy.address)
|
|
|
|
await insertContractAddressInDb(
|
|
eContractid.LendingPool,
|
|
lendingPoolProxy.address
|
|
);
|
|
|
|
const fallbackOracle = await deployPriceOracle();
|
|
await waitForTx(await fallbackOracle.setEthUsdPrice(MOCK_USD_PRICE_IN_WEI));
|
|
await setInitialAssetPricesInOracle(
|
|
ALL_ASSETS_INITIAL_PRICES,
|
|
{
|
|
ETH: MOCK_ETH_ADDRESS,
|
|
DAI: mockTokens.DAI.address,
|
|
TUSD: mockTokens.TUSD.address,
|
|
USDC: mockTokens.USDC.address,
|
|
USDT: mockTokens.USDT.address,
|
|
SUSD: mockTokens.SUSD.address,
|
|
LEND: mockTokens.LEND.address,
|
|
BAT: mockTokens.BAT.address,
|
|
REP: mockTokens.REP.address,
|
|
MKR: mockTokens.MKR.address,
|
|
LINK: mockTokens.LINK.address,
|
|
KNC: mockTokens.KNC.address,
|
|
WBTC: mockTokens.WBTC.address,
|
|
MANA: mockTokens.MANA.address,
|
|
ZRX: mockTokens.ZRX.address,
|
|
SNX: mockTokens.SNX.address,
|
|
BUSD: mockTokens.BUSD.address,
|
|
|
|
USD: USD_ADDRESS,
|
|
|
|
UNI_DAI_ETH: mockTokens.UNI_DAI_ETH.address,
|
|
UNI_USDC_ETH: mockTokens.UNI_USDC_ETH.address,
|
|
UNI_SETH_ETH: mockTokens.UNI_SETH_ETH.address,
|
|
UNI_LEND_ETH: mockTokens.UNI_LEND_ETH.address,
|
|
UNI_MKR_ETH: mockTokens.UNI_MKR_ETH.address,
|
|
UNI_LINK_ETH: mockTokens.UNI_LINK_ETH.address,
|
|
},
|
|
fallbackOracle
|
|
);
|
|
|
|
const mockAggregators = await deployAllMockAggregators(
|
|
MOCK_CHAINLINK_AGGREGATORS_PRICES
|
|
);
|
|
|
|
const allTokenAddresses = Object.entries(mockTokens).reduce(
|
|
(
|
|
accum: {[tokenSymbol: string]: tEthereumAddress},
|
|
[tokenSymbol, tokenContract]
|
|
) => ({
|
|
...accum,
|
|
[tokenSymbol]: tokenContract.address,
|
|
}),
|
|
{}
|
|
);
|
|
const allAggregatorsAddresses = Object.entries(mockAggregators).reduce(
|
|
(
|
|
accum: {[tokenSymbol: string]: tEthereumAddress},
|
|
[tokenSymbol, aggregator]
|
|
) => ({
|
|
...accum,
|
|
[tokenSymbol]: aggregator.address,
|
|
}),
|
|
{}
|
|
);
|
|
|
|
const [tokens, aggregators] = getPairsTokenAggregator(
|
|
allTokenAddresses,
|
|
allAggregatorsAddresses
|
|
);
|
|
|
|
const chainlinkProxyPriceProvider = await deployChainlinkProxyPriceProvider([
|
|
tokens,
|
|
aggregators,
|
|
fallbackOracle.address,
|
|
]);
|
|
await waitForTx(
|
|
await addressesProvider.setPriceOracle(chainlinkProxyPriceProvider.address)
|
|
);
|
|
|
|
const lendingRateOracle = await deployLendingRateOracle();
|
|
await waitForTx(
|
|
await addressesProvider.setLendingRateOracle(lendingRateOracle.address)
|
|
);
|
|
|
|
const {USD, ...tokensAddressesWithoutUsd} = allTokenAddresses;
|
|
const allReservesAddresses = {
|
|
ETH: MOCK_ETH_ADDRESS,
|
|
...tokensAddressesWithoutUsd,
|
|
};
|
|
await setInitialMarketRatesInRatesOracle(
|
|
LENDING_RATE_ORACLE_RATES_COMMON,
|
|
allReservesAddresses,
|
|
lendingRateOracle
|
|
);
|
|
|
|
const {
|
|
UNI_DAI_ETH,
|
|
UNI_USDC_ETH,
|
|
UNI_SETH_ETH,
|
|
UNI_LINK_ETH,
|
|
UNI_MKR_ETH,
|
|
UNI_LEND_ETH,
|
|
...protoPoolReservesAddresses
|
|
} = <{[symbol: string]: tEthereumAddress}>allReservesAddresses;
|
|
|
|
const reservesParams = getReservesConfigByPool(AavePools.proto);
|
|
|
|
console.log("Initialize configuration")
|
|
await initReserves(
|
|
reservesParams,
|
|
protoPoolReservesAddresses,
|
|
addressesProvider,
|
|
lendingPoolProxy,
|
|
lendingPoolConfiguratorProxy,
|
|
AavePools.proto
|
|
);
|
|
await enableReservesToBorrow(
|
|
reservesParams,
|
|
protoPoolReservesAddresses,
|
|
lendingPoolProxy,
|
|
lendingPoolConfiguratorProxy
|
|
);
|
|
await enableReservesAsCollateral(
|
|
reservesParams,
|
|
protoPoolReservesAddresses,
|
|
lendingPoolProxy,
|
|
lendingPoolConfiguratorProxy
|
|
);
|
|
|
|
const liquidationManager = await deployLendingPoolLiquidationManager();
|
|
await waitForTx(
|
|
await addressesProvider.setLendingPoolLiquidationManager(
|
|
liquidationManager.address
|
|
)
|
|
);
|
|
|
|
const {receivers, percentages} = getFeeDistributionParamsCommon(
|
|
lendingPoolManager
|
|
);
|
|
|
|
await deployMockOneSplit(tokensAddressesWithoutUsd.LEND);
|
|
const oneSplitAdapter = await deployOneSplitAdapter();
|
|
const tokenDistributorImpl = await deployTokenDistributor();
|
|
const tokenDistributorProxy = await deployInitializableAdminUpgradeabilityProxy();
|
|
const implementationParams = tokenDistributorImpl.interface.functions.initialize.encode(
|
|
[
|
|
ZERO_ADDRESS,
|
|
tokensAddressesWithoutUsd.LEND,
|
|
oneSplitAdapter.address,
|
|
receivers,
|
|
percentages,
|
|
Object.values(tokensAddressesWithoutUsd),
|
|
]
|
|
);
|
|
await waitForTx(
|
|
await tokenDistributorProxy.initialize(
|
|
tokenDistributorImpl.address,
|
|
await secondaryWallet.getAddress(),
|
|
implementationParams
|
|
)
|
|
);
|
|
await waitForTx(
|
|
await addressesProvider.setTokenDistributor(tokenDistributorProxy.address)
|
|
);
|
|
|
|
await insertContractAddressInDb(
|
|
eContractid.TokenDistributor,
|
|
tokenDistributorProxy.address
|
|
);
|
|
|
|
const mockFlashLoanReceiver = await deployMockFlashLoanReceiver(
|
|
addressesProvider.address
|
|
);
|
|
await insertContractAddressInDb(
|
|
eContractid.MockFlashLoanReceiver,
|
|
mockFlashLoanReceiver.address
|
|
);
|
|
|
|
await deployWalletBalancerProvider(addressesProvider.address);
|
|
|
|
const testHelpers = await deployAaveProtocolTestHelpers(
|
|
addressesProvider.address
|
|
);
|
|
|
|
await insertContractAddressInDb(
|
|
eContractid.AaveProtocolTestHelpers,
|
|
testHelpers.address
|
|
);
|
|
|
|
console.timeEnd("setup");
|
|
};
|
|
|
|
before(async () => {
|
|
await rawBRE.run("set-bre");
|
|
const [deployer, secondaryWallet] = await getEthersSigners();
|
|
console.log("-> Deploying test environment...");
|
|
await buildTestEnv(deployer, secondaryWallet);
|
|
await initializeMakeSuite();
|
|
console.log("\n***************");
|
|
console.log("Setup and snapshot finished");
|
|
console.log("***************\n");
|
|
});
|