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); }); }); }); });