aave-protocol-v2/test-suites/test-aave/rewardAwareAToken.spec.ts
2021-06-03 12:47:34 +02:00

595 lines
18 KiB
TypeScript

import BigNumberJs from 'bignumber.js';
import { reverse } from 'dns';
import { BigNumber } from 'ethers';
import { formatEther, parseEther } from 'ethers/lib/utils';
import { MAX_UINT_AMOUNT } from '../../helpers/constants';
import { evmRevert, evmSnapshot, increaseTime } from '../../helpers/misc-utils';
import { makeSuite, SignerWithAddress, TestEnv } from './helpers/make-suite';
import { checkRewards } from './helpers/rewards-distribution/verify';
const chai = require('chai');
const { expect } = chai;
/**
* @dev REW is a mocked mintable token named RewardsToken.sol with an emission rate of 1 REW per second that can be deposited using a RewardsAwareAToken implementation named RewardsATokenMock.sol
* The distribution of REW happens at `claim`, but there is also a `updateMintableEmission` functions that updates the state of the distribution of 1 user but does not claim.
*/
makeSuite('Reward Aware AToken', (testEnv: TestEnv) => {
let initTimestamp;
let evmSnapshotId;
before('Initializing configuration', async () => {
// Sets BigNumber for this suite, instead of globally
BigNumberJs.config({ DECIMAL_PLACES: 0, ROUNDING_MODE: BigNumberJs.ROUND_DOWN });
});
after('Reset', () => {
// Reset BigNumber
BigNumberJs.config({ DECIMAL_PLACES: 20, ROUNDING_MODE: BigNumberJs.ROUND_HALF_UP });
});
beforeEach(async () => {
initTimestamp = await testEnv.rew.INIT_TIMESTAMP();
evmSnapshotId = await evmSnapshot();
});
afterEach(async () => {
await evmRevert(evmSnapshotId);
});
const mintAndDeposit = async (
key: SignerWithAddress,
shouldReward?: boolean,
amountSize?: BigNumber
) => {
const { rew, aRew, pool } = testEnv;
const amount = amountSize || parseEther('1');
const userATokenBalanceBeforeDeposit = await aRew.balanceOf(key.address);
// Mint REW to user
await rew.connect(key.signer).mint(amount);
// Approve and Deposit REW to pool and mint
await rew.connect(key.signer).approve(pool.address, amount);
const txDeposit = await pool.connect(key.signer).deposit(rew.address, amount, key.address, '0');
expect(Promise.resolve(txDeposit)).emit(aRew, 'Mint');
const userBalanceAfterDeposit = await rew.balanceOf(key.address);
const userATokenBalanceAfterDeposit = await aRew.balanceOf(key.address);
expect(userATokenBalanceAfterDeposit)
.to.be.eq(
userATokenBalanceBeforeDeposit.add(amount),
'User aToken balance should be equal the amount deposited'
)
.and.gt('0', 'User aToken balance should be greater than zero');
expect(userBalanceAfterDeposit).to.be.eq('0', 'Token balance should be zero');
if (!txDeposit.blockNumber) {
throw 'missing block number';
}
// Check all token rewards
await checkRewards(key, aRew.address, txDeposit.blockNumber, shouldReward);
};
const claim = async (user: SignerWithAddress, skipRewardChecks?: boolean) => {
skipRewardChecks = true;
const { rew, aRew } = testEnv;
// Claim all the rewards from aToken
const txClaim = await aRew.connect(user.signer).claim(rew.address);
await expect(Promise.resolve(txClaim)).to.emit(aRew, 'Claim');
// Calculate expected rewards
if (!txClaim.blockNumber) {
throw 'Block number missing from tx';
}
if (!skipRewardChecks) {
// Check all token rewards
await checkRewards(user, aRew.address, txClaim.blockNumber, true);
}
};
describe('Deposits: mints', async () => {
it('User1 deposits Reward Aware Token to Lending Pool', async () => {
const {
users: [user1],
} = testEnv;
await mintAndDeposit(user1);
});
it('Other users deposits Reward Aware Token to Lending Pool', async () => {
const {
users: [, user2, user3, user4],
} = testEnv;
await mintAndDeposit(user2);
await mintAndDeposit(user3);
await mintAndDeposit(user4);
});
it('User1 deposits multiple time Reward Aware Token to Lending Pool', async () => {
const {
users: [user1],
} = testEnv;
await mintAndDeposit(user1, false, parseEther('2'));
await mintAndDeposit(user1, true, parseEther('12'));
});
});
describe('Withdrawals: burns', async () => {
it('User1 deposits Reward Aware Token to Lending Pool', async () => {
const {
users: [user1],
aRew,
rew,
pool,
} = testEnv;
// Deposits
await mintAndDeposit(user1);
// Burn/Withdraw
await aRew.approve(pool.address, MAX_UINT_AMOUNT);
await pool.connect(user1.signer).withdraw(rew.address, MAX_UINT_AMOUNT, user1.address);
});
});
describe('Claim rewards', async () => {
it('User1 claims 100% portion of REW via the aToken contract', async () => {
const {
users: [user1],
rew,
aRew,
} = testEnv;
// User1 deposits
await mintAndDeposit(user1);
// Pass time to generate rewards
await increaseTime(1000);
// Check rewards for aToken at rew
const token = await aRew.getRewardsTokenAddress('0');
const aTokenRewards = await rew.getClaimableRewards(aRew.address);
// Check rewards for user at aRew
const userRewards = await aRew.getClaimableRewards(rew.address, user1.address);
// Expect user rewards to be the same as aToken rewards due 100%
expect(aTokenRewards).to.be.eq(
userRewards,
'Rewards should be the same due user holds 100% of RewardsAware distribution'
);
// Claims and check rewards
await claim(user1);
});
it('Two users with half the portion of REW via the aToken contract', async () => {
const {
users: [user1, user2],
rew,
aRew,
} = testEnv;
// User1 and user2 deposits
await mintAndDeposit(user1);
await mintAndDeposit(user2);
// Pass time to generate rewards
await increaseTime(1000);
// Check rewards for aToken at rew
const aTokenRewards = await rew.getClaimableRewards(aRew.address);
// Check rewards for user1 at aRew
const user1Rewards = await aRew.getClaimableRewards(rew.address, user1.address);
// Check rewards for user2 at aRew
const user2Rewards = await aRew.getClaimableRewards(rew.address, user2.address);
// Expect user rewards to be the same as aToken rewards
expect(aTokenRewards).to.be.eq(
user1Rewards.add(user2Rewards),
'Rewards should be the same of all users of RewardsAware distribution'
);
// Claims and check rewards
await claim(user1);
await claim(user2);
});
it('Four users with different portions of REW via the aToken contract', async () => {
const {
users: [user1, user2, user3, user4],
rew,
aRew,
} = testEnv;
// Deposits
await mintAndDeposit(user1, false, parseEther('1'));
await mintAndDeposit(user2, false, parseEther('2.5'));
await mintAndDeposit(user3, false, parseEther('4.7'));
await mintAndDeposit(user4, false, parseEther('0.31'));
// Pass time to generate rewards
await increaseTime(1000);
// Claims and check rewards
await claim(user1);
await claim(user2);
await claim(user3);
await claim(user4);
// Pass time to generate rewards
await increaseTime(2713);
// Claims and check rewards
await claim(user1);
await claim(user2);
await claim(user3);
await claim(user4);
});
it('Annoying case. Four users with different portions of REW via the aToken contract (withdraw before last claim)', async () => {
const {
users: [user1, user2, user3, user4],
rew,
aRew,
pool,
} = testEnv;
let users = [user1, user2]; //, user3, user4];
const printState = async () => {
let summedClaimable = BigNumber.from(0);
for (let i = 0; i < users.length; i++) {
let usr = users[i];
console.log(
`\tUser rewards: ${formatEther(
await rew.balanceOf(usr.address)
)}. User claimable rewards: ${formatEther(
await aRew.getClaimableRewards(rew.address, usr.address)
)}. User aRew balance: ${formatEther(await aRew.balanceOf(usr.address))}`
);
summedClaimable = summedClaimable.add(
await aRew.getClaimableRewards(rew.address, usr.address)
);
}
console.log(
`\taRew Total supply: ${formatEther(
await aRew.totalSupply()
)}. Rewards in contract: ${formatEther(
await rew.balanceOf(aRew.address)
)}. Summed claimable: ${formatEther(summedClaimable)}`
);
};
// Deposits
console.log(`Mint and deposit`);
await mintAndDeposit(user1, false, parseEther('1'));
await mintAndDeposit(user2, false, parseEther('2.5'));
//await mintAndDeposit(user3, false, parseEther('4.7'));
//await mintAndDeposit(user4, false, parseEther('0.31'));
// Pass time to generate rewards
console.log(`IncreaseTime by 1000`);
await increaseTime(1000);
await printState();
// Claims and check rewards
console.log(`Claim`);
await claim(user1);
await printState();
// Pass time to generate rewards
console.log(`Increase time by 2713`);
await increaseTime(2713);
console.log(`Withdraws`);
for (let i = 0; i < users.length; i++) {
await pool
.connect(users[i].signer)
.withdraw(rew.address, MAX_UINT_AMOUNT, users[i].address);
}
// Claims and check rewards
console.log(`Claim`);
for (let i = 0; i < users.length; i++) {
await claim(users[i]);
}
await printState();
});
it('Two users with half the portion of REW via the aToken contract, one burns and them claims', async () => {
const {
users: [user1, user2],
rew,
aRew,
pool,
} = testEnv;
// User1 and user2 deposits
await mintAndDeposit(user1, false, parseEther('1000'));
await mintAndDeposit(user2, false, parseEther('1000'));
// Pass time to generate rewards
await increaseTime(1000);
// Claims and check rewards
await claim(user1);
await claim(user2);
// Pass time to generate rewards
await increaseTime(1000);
// Burn/Withdraw to update current lifetime rewards state
await aRew.approve(pool.address, MAX_UINT_AMOUNT);
await pool.connect(user1.signer).withdraw(rew.address, MAX_UINT_AMOUNT, user1.address);
// Claims
await claim(user1);
await claim(user2);
});
});
describe('Getters', () => {
describe('getClaimableRewards', () => {
it('Rewards should be zero if user with no balance', async () => {
const {
users: [user1],
aRew,
rew,
} = testEnv;
const rewards = await aRew.getClaimableRewards(rew.address, user1.address);
expect(rewards).eq('0', 'Rewards should be zero');
});
it('Rewards should be available after time travel', async () => {
const {
users: [user1],
aRew,
rew,
} = testEnv;
await mintAndDeposit(user1);
// Pass time to generate rewards
await increaseTime(1000);
const rewards = await aRew.getClaimableRewards(rew.address, user1.address);
expect(rewards).gt('0', 'Rewards should be greater than zero');
});
});
describe('getUserLifetimeRewardsAccrued', () => {
it('User lifetime rewards should be zero if no deposit', async () => {
const {
users: [user1],
aRew,
rew,
} = testEnv;
const rewards = await aRew.getUserRewardsAccrued(rew.address, user1.address);
expect(rewards).eq('0', 'Rewards should be zero');
});
it('User lifetime rewards should be zero if deposit due state is not updated', async () => {
const {
users: [user1],
aRew,
rew,
} = testEnv;
await mintAndDeposit(user1);
const rewards = await aRew.getUserRewardsAccrued(rew.address, user1.address);
expect(rewards).eq('0', 'Rewards should be zero');
});
it('User should have some lifetime rewards if deposit again due state is updated', async () => {
const {
users: [user1],
aRew,
rew,
} = testEnv;
await mintAndDeposit(user1);
// Pass time to generate rewards
await increaseTime(1000);
await mintAndDeposit(user1, true);
const rewards = await aRew.getUserRewardsAccrued(rew.address, user1.address);
expect(rewards).gt('0', 'Rewards should be greater than zero');
});
it('User should have some lifetime rewards if claims due state is updated', async () => {
const {
users: [user1],
aRew,
rew,
} = testEnv;
await mintAndDeposit(user1);
// Pass time to generate rewards
await increaseTime(1000);
await claim(user1);
const rewards = await aRew.getUserRewardsAccrued(rew.address, user1.address);
expect(rewards).gt('0', 'Rewards should be greater than zero');
});
});
describe('getUserIndex', () => {
it('User index should be zero if no deposit', async () => {
const {
users: [user1],
aRew,
rew,
} = testEnv;
const rewards = await aRew.getUserIndex(rew.address, user1.address);
expect(rewards).eq('0', 'Rewards should be zero');
});
it('User lifetime rewards should be zero if deposit due state is not updated', async () => {
const {
users: [user1],
aRew,
rew,
} = testEnv;
await mintAndDeposit(user1);
const rewards = await aRew.getUserIndex(rew.address, user1.address);
expect(rewards).eq('0', 'Rewards should be zero');
});
it('User should have some lifetime rewards if deposit again due state is updated', async () => {
const {
users: [user1],
aRew,
rew,
} = testEnv;
await mintAndDeposit(user1);
// Pass time to generate rewards
await increaseTime(1000);
await mintAndDeposit(user1, true);
const rewards = await aRew.getUserIndex(rew.address, user1.address);
expect(rewards).gt('0', 'Rewards should be greater than zero');
});
it('User should have some lifetime rewards if claims due state is updated', async () => {
const {
users: [user1],
aRew,
rew,
} = testEnv;
await mintAndDeposit(user1);
// Pass time to generate rewards
await increaseTime(1000);
await claim(user1);
const rewards = await aRew.getUserIndex(rew.address, user1.address);
expect(rewards).gt('0', 'Rewards should be greater than zero');
});
});
describe('getUserClaimedRewards', () => {
it('User should NOT have claimed rewards if didnt claim', async () => {
const {
users: [user1],
aRew,
rew,
} = testEnv;
await mintAndDeposit(user1);
// Pass time to generate rewards
await increaseTime(1000);
const rewards = await aRew.getUserIndex(rew.address, user1.address);
expect(rewards).eq('0', 'Rewards should be zero');
});
it('User should have claimed rewards if claims', async () => {
const {
users: [user1],
aRew,
rew,
} = testEnv;
await mintAndDeposit(user1);
// Pass time to generate rewards
await increaseTime(1000);
await claim(user1);
const rewards = await aRew.getUserIndex(rew.address, user1.address);
expect(rewards).gt('0', 'Rewards should be greater than zero');
});
});
describe('getLifetimeRewards', () => {
it('The aToken Tifetime rewards should be zero if there is no deposits', async () => {
const { aRew, rew } = testEnv;
const rewards = await aRew.getLifetimeRewards(rew.address);
expect(rewards).eq('0', 'Rewards should be zero');
});
it('The aToken lifetime rewards should be zero at init', async () => {
const { aRew, rew } = testEnv;
const rewards = await aRew.getLifetimeRewards(rew.address);
expect(rewards).eq('0', 'Rewards should be zero');
});
it('The aToken lifetime rewards should update if there is further actions: deposit', async () => {
const {
users: [user1],
aRew,
rew,
} = testEnv;
await mintAndDeposit(user1);
// Pass time to generate rewards
await increaseTime(1000);
// Deposit again to update current lifetime rewards state
await mintAndDeposit(user1, true);
const rewards = await aRew.getLifetimeRewards(rew.address);
expect(rewards).gte('0', 'Rewards should be greater than zero');
});
it('The aToken lifetime rewards should update if there is further actions: claim', async () => {
const {
users: [user1],
aRew,
rew,
} = testEnv;
await mintAndDeposit(user1);
// Pass time to generate rewards
await increaseTime(1000);
// Claim to update current lifetime rewards state
await claim(user1);
const rewards = await aRew.getLifetimeRewards(rew.address);
expect(rewards).gte('0', 'Rewards should be greater than zero');
});
it('The aToken lifetime rewards should update if there is further actions: burn', async () => {
const {
users: [user1],
aRew,
rew,
pool,
} = testEnv;
await mintAndDeposit(user1);
// Pass time to generate rewards
await increaseTime(1000);
// Burn/Withdraw to update current lifetime rewards state
await aRew.approve(pool.address, MAX_UINT_AMOUNT);
await pool.connect(user1.signer).withdraw(rew.address, MAX_UINT_AMOUNT, user1.address);
const rewards = await aRew.getLifetimeRewards(rew.address);
expect(rewards).gte('0', 'Rewards should be greater than zero');
});
});
describe('getRewardsToken', () => {
it('The getter should return the current token reward address', async () => {
const { aRew, rew } = testEnv;
const rewardToken = await aRew.getRewardsTokenAddress(0);
expect(rewardToken).to.be.equal(rew.address);
});
});
});
});