const { assertRevert } = require('@aragon/contract-helpers-test/assertThrow') const { bn, bigExp, assertBn, MAX_UINT64 } = require('@aragon/contract-helpers-test/numbers') const { deploy } = require('../helpers/deploy')(artifacts) const { approveAndStake, approveStakeAndLock } = require('../helpers/helpers')(artifacts) const { DEFAULT_STAKE_AMOUNT, DEFAULT_LOCK_AMOUNT, EMPTY_DATA, ZERO_ADDRESS } = require('../helpers/constants') const { STAKING_ERRORS } = require('../helpers/errors') contract('Staking app, Locking', ([owner, user1, user2]) => { let staking, lockManager beforeEach(async () => { const deployment = await deploy(owner) staking = deployment.staking lockManager = deployment.lockManager }) // it('allows new manager and locks amount', async () => { // await approveStakeAndLock({ staking, manager: user1, from: owner }) // // check lock values // const { _amount, _allowance } = await staking.getLock(owner, user1) // assertBn(_amount, DEFAULT_LOCK_AMOUNT, "locked amount should match") // assertBn(_allowance, DEFAULT_LOCK_AMOUNT, "locked allowance should match") // assertBn(await staking.unlockedBalanceOf(owner), DEFAULT_STAKE_AMOUNT.sub(DEFAULT_LOCK_AMOUNT), "Unlocked balance should match") // const { staked, locked } = await staking.getBalancesOf(owner) // assertBn(staked, DEFAULT_STAKE_AMOUNT, "Staked balance should match") // assertBn(locked, DEFAULT_LOCK_AMOUNT, "Locked balance should match") // }) it('fails locking 0 tokens', async () => { await approveAndStake({ staking, from: owner }) await assertRevert(staking.allowManagerAndLock(0, user1, 1, EMPTY_DATA), STAKING_ERRORS.ERROR_AMOUNT_ZERO) }) // it('fails locking without enough allowance', async () => { // await approveAndStake({ staking, from: owner }) // await assertRevert(staking.allowManagerAndLock(2, user1, 1, EMPTY_DATA), STAKING_ERRORS.ERROR_NOT_ENOUGH_ALLOWANCE) // }) // it('fails locking more tokens than staked', async () => { // await approveAndStake({ staking, from: owner }) // await assertRevert(staking.allowManagerAndLock(DEFAULT_STAKE_AMOUNT.add(bn(1)), user1, DEFAULT_STAKE_AMOUNT.add(bn(1)), EMPTY_DATA), STAKING_ERRORS.ERROR_NOT_ENOUGH_BALANCE) // }) // it('fails locking if already locked', async () => { // await approveStakeAndLock({ staking, manager: user1, from: owner }) // await approveAndStake({ staking, from: owner }) // await assertRevert(staking.allowManagerAndLock(DEFAULT_STAKE_AMOUNT, user1, DEFAULT_STAKE_AMOUNT, "0x02"), STAKING_ERRORS.ERROR_LOCK_ALREADY_EXISTS) // }) // it('fails unstaking locked tokens', async () => { // await approveStakeAndLock({ staking, manager: user1, from: owner }) // await assertRevert(staking.unstake(DEFAULT_STAKE_AMOUNT, EMPTY_DATA), STAKING_ERRORS.ERROR_NOT_ENOUGH_BALANCE) // }) // it('creates a new allowance', async () => { // await staking.allowManager(user1, DEFAULT_LOCK_AMOUNT, EMPTY_DATA) // const { _allowance } = await staking.getLock(owner, user1) // assertBn(_allowance, DEFAULT_LOCK_AMOUNT, "allowed amount should match") // }) // it('creates a new allowance and then lock manager locks', async () => { // await approveAndStake({ staking, from: owner }) // await staking.allowManager(user1, DEFAULT_LOCK_AMOUNT, EMPTY_DATA) // await staking.lock(owner, user1, DEFAULT_LOCK_AMOUNT, { from: user1 }) // // check lock values // const { _amount, _allowance } = await staking.getLock(owner, user1) // assertBn(_amount, DEFAULT_LOCK_AMOUNT, "locked amount should match") // assertBn(_allowance, DEFAULT_LOCK_AMOUNT, "locked allowance should match") // assertBn(await staking.unlockedBalanceOf(owner), DEFAULT_STAKE_AMOUNT.sub(DEFAULT_LOCK_AMOUNT), "Unlocked balance should match") // }) // it('fails creating allowance of 0 tokens', async () => { // await assertRevert(staking.allowManager(user1, 0, EMPTY_DATA), STAKING_ERRORS.ERROR_AMOUNT_ZERO) // }) // it('fails creating allowance if lock exists', async () => { // await approveStakeAndLock({ staking, manager: user1, from: owner }) // await assertRevert(staking.allowManager(user1, 1, EMPTY_DATA), STAKING_ERRORS.ERROR_LOCK_ALREADY_EXISTS) // }) // it('increases allowance of existing lock', async () => { // await approveStakeAndLock({ staking, manager: user1, from: owner }) // await staking.increaseLockAllowance(user1, DEFAULT_LOCK_AMOUNT) // const { _allowance } = await staking.getLock(owner, user1) // assertBn(_allowance, DEFAULT_LOCK_AMOUNT.mul(bn(2)), "allowed amount should match") // }) // it('fails increasing allowance of non-existing', async () => { // await assertRevert(staking.increaseLockAllowance(user1, 1), STAKING_ERRORS.ERROR_LOCK_DOES_NOT_EXIST) // }) // it('fails increasing allowance of existing lock by 0', async () => { // await approveStakeAndLock({ staking, manager: user1, from: owner }) // await assertRevert(staking.increaseLockAllowance(user1, 0), STAKING_ERRORS.ERROR_AMOUNT_ZERO) // }) // it('fails increasing allowance of existing lock if not owner or manager', async () => { // await approveStakeAndLock({ staking, manager: user1, from: owner }) // await assertRevert(staking.increaseLockAllowance(user1, 1, { from: user2 }), STAKING_ERRORS.ERROR_LOCK_DOES_NOT_EXIST) // }) // it('decreases allowance of existing lock by the owner', async () => { // await approveAndStake({ staking, from: owner }) // await staking.allowManagerAndLock(DEFAULT_LOCK_AMOUNT, user1, DEFAULT_LOCK_AMOUNT.add(bn(1)), EMPTY_DATA) // await staking.decreaseLockAllowance(owner, user1, 1, { from: owner }) // const { _allowance } = await staking.getLock(owner, user1) // assertBn(_allowance, DEFAULT_LOCK_AMOUNT, "allowed amount should match") // }) // it('decreases allowance of existing lock by manager', async () => { // await approveAndStake({ staking, from: owner }) // await staking.allowManagerAndLock(DEFAULT_LOCK_AMOUNT, user1, DEFAULT_LOCK_AMOUNT.add(bn(1)), EMPTY_DATA) // await staking.decreaseLockAllowance(owner, user1, 1, { from: user1 }) // const { _allowance } = await staking.getLock(owner, user1) // assertBn(_allowance, DEFAULT_LOCK_AMOUNT, "allowed amount should match") // }) // it('fails decreasing allowance of existing lock by 0', async () => { // await approveStakeAndLock({ staking, manager: user1, from: owner }) // await assertRevert(staking.decreaseLockAllowance(owner, user1, 0), STAKING_ERRORS.ERROR_AMOUNT_ZERO) // }) // it('fails decreasing allowance of existing lock to 0', async () => { // await approveStakeAndLock({ staking, manager: user1, from: owner }) // await staking.unlock(owner, user1, DEFAULT_LOCK_AMOUNT, { from: user1 }) // await assertRevert(staking.decreaseLockAllowance(owner, user1, DEFAULT_LOCK_AMOUNT), STAKING_ERRORS.ERROR_ALLOWANCE_ZERO) // }) // it('fails decreasing allowance to less than lock', async () => { // await approveAndStake({ staking, from: owner }) // await staking.allowManagerAndLock(DEFAULT_LOCK_AMOUNT, user1, DEFAULT_LOCK_AMOUNT.add(bn(1)), EMPTY_DATA) // await assertRevert(staking.decreaseLockAllowance(owner, user1, 2), STAKING_ERRORS.ERROR_NOT_ENOUGH_ALLOWANCE) // }) // it('fails decreasing allowance by 3rd party', async () => { // await approveAndStake({ staking, from: owner }) // await staking.allowManagerAndLock(DEFAULT_LOCK_AMOUNT, user1, DEFAULT_LOCK_AMOUNT.add(bn(1)), EMPTY_DATA) // await assertRevert(staking.decreaseLockAllowance(owner, user1, 1, { from: user2 }), STAKING_ERRORS.ERROR_CANNOT_CHANGE_ALLOWANCE) // }) // it('increases amount of existing lock', async () => { // await approveStakeAndLock({ staking, manager: user1, from: owner }) // await approveAndStake({ staking, from: owner }) // await staking.increaseLockAllowance(user1, DEFAULT_LOCK_AMOUNT) // await staking.lock(owner, user1, DEFAULT_LOCK_AMOUNT) // const { _amount } = await staking.getLock(owner, user1) // assertBn(_amount, DEFAULT_LOCK_AMOUNT.mul(bn(2)), "locked amount should match") // }) // it('fails increasing lock with 0 tokens', async () => { // await approveStakeAndLock({ staking, manager: user1, from: owner }) // await approveAndStake({ staking, from: owner }) // await assertRevert(staking.lock(owner, user1, 0), STAKING_ERRORS.ERROR_AMOUNT_ZERO) // }) // it('fails increasing lock with more tokens than staked', async () => { // await approveStakeAndLock({ staking, manager: user1, from: owner }) // await approveAndStake({ staking, from: owner }) // await assertRevert(staking.lock(owner, user1, DEFAULT_STAKE_AMOUNT.mul(bn(2)).add(bn(1))), STAKING_ERRORS.ERROR_NOT_ENOUGH_BALANCE) // }) // it('fails increasing lock if not owner or manager', async () => { // await approveStakeAndLock({ staking, manager: user1, from: owner }) // await approveAndStake({ staking, from: owner }) // await assertRevert(staking.lock(owner, user1, 1, { from: user2 }), STAKING_ERRORS.ERROR_SENDER_NOT_ALLOWED) // }) // it('unlocks with only 1 lock, EOA manager', async () => { // await approveStakeAndLock({ staking, manager: user1, lockAmount: DEFAULT_LOCK_AMOUNT, from: owner }) // // unlock // await staking.unlockAndRemoveManager(owner, user1, { from: user1 }) // assertBn(await staking.unlockedBalanceOf(owner), DEFAULT_STAKE_AMOUNT, "Unlocked balance should match") // assertBn(await staking.lockedBalanceOf(owner), bn(0), "total locked doesn’t match") // }) // it('unlocks with more than 1 lock, EOA manager', async () => { // await approveStakeAndLock({ staking, manager: user1, from: owner }) // // lock again // await staking.allowManagerAndLock(DEFAULT_LOCK_AMOUNT, user2, DEFAULT_LOCK_AMOUNT, EMPTY_DATA) // const previousTotalLocked = await staking.lockedBalanceOf(owner) // // unlock // await staking.unlockAndRemoveManager(owner, user1, { from: user1 }) // assertBn(await staking.unlockedBalanceOf(owner), DEFAULT_STAKE_AMOUNT.sub(DEFAULT_LOCK_AMOUNT), "Unlocked balance should match") // assertBn(await staking.lockedBalanceOf(owner), previousTotalLocked.sub(bn(DEFAULT_LOCK_AMOUNT)), "total locked doesn’t match") // }) // it('unlocks completely, contract manager, called by owner', async () => { // await lockManager.setResult(true) // await approveStakeAndLock({ staking, manager: lockManager.address, from: owner }) // // unlock // await staking.unlockAndRemoveManager(owner, lockManager.address, { from: owner }) // assertBn(await staking.unlockedBalanceOf(owner), DEFAULT_STAKE_AMOUNT, "Unlocked balance should match") // assertBn(await staking.lockedBalanceOf(owner), bn(0), "total locked doesn’t match") // }) // it('unlocks completely, contract manager, called by manager', async () => { // await lockManager.setResult(true) // await approveStakeAndLock({ staking, manager: lockManager.address, from: owner }) // // unlock // await lockManager.unlockAndRemoveManager(staking.address, owner, lockManager.address) // assertBn(await staking.unlockedBalanceOf(owner), DEFAULT_STAKE_AMOUNT, "Unlocked balance should match") // assertBn(await staking.lockedBalanceOf(owner), bn(0), "total locked doesn’t match") // }) // it('unlocks completely, contract manager, called by manager, even if condition is not satisfied', async () => { // // not needed, is false by default // //await lockManager.setResult(false) // await approveStakeAndLock({ staking, manager: lockManager.address, from: owner }) // // unlock // await lockManager.unlockAndRemoveManager(staking.address, owner, lockManager.address) // assertBn(await staking.unlockedBalanceOf(owner), DEFAULT_STAKE_AMOUNT, "Unlocked balance should match") // assertBn(await staking.lockedBalanceOf(owner), bn(0), "total locked doesn’t match") // }) // it('fails calling canUnlock, EOA manager', async () => { // await approveStakeAndLock({ staking, manager: user1, from: owner }) // // call canUnlock // await assertRevert(staking.canUnlock(owner, owner, user1, 0)) // no reason: it’s trying to call an EOA // }) // it('can unlock if amount is zero', async () => { // await staking.allowManager(user1, DEFAULT_LOCK_AMOUNT, EMPTY_DATA, { from: owner }) // assert.isTrue(await staking.canUnlock(owner, owner, user1, 0)) // }) // it('fails to unlock if it cannot unlock, EOA manager', async () => { // await approveStakeAndLock({ staking, manager: user1, from: owner }) // // tries to unlock // await assertRevert(staking.unlockAndRemoveManager(owner, user1)) // no reason: it’s trying to call an EOA // }) // it('fails to unlock if can not unlock, contract manager, called by owner', async () => { // // not needed, is false by default // // await lockManager.setResult(false) // await approveStakeAndLock({ staking, manager: lockManager.address, from: owner }) // // tries to unlock // await assertRevert(staking.unlockAndRemoveManager(owner, lockManager.address, { from: owner }), STAKING_ERRORS.ERROR_CANNOT_UNLOCK) // }) // it('fails to unlock if, contract manager, called by 3rd party (even if condition is true)', async () => { // await lockManager.setResult(true) // await approveStakeAndLock({ staking, manager: lockManager.address, from: owner }) // // tries to unlock // await assertRevert(staking.unlockAndRemoveManager(owner, lockManager.address, { from: user1 }), STAKING_ERRORS.ERROR_CANNOT_UNLOCK) // }) // it('transfers (slash) and unlocks (everything else) in one transaction', async () => { // const totalLock = bigExp(120, 18) // const transferAmount = bigExp(40, 18) // await approveStakeAndLock({ staking, manager: user1, allowanceAmount: totalLock, lockAmount: totalLock, stakeAmount: totalLock, from: owner }) // // unlock and transfer // await staking.slashAndUnlock(owner, user2, totalLock.sub(transferAmount), transferAmount, { from: user1 }) // assertBn(await staking.unlockedBalanceOf(owner), totalLock.sub(transferAmount), "Unlocked balance should match") // assertBn(await staking.lockedBalanceOf(owner), bn(0), "total locked doesn’t match") // // lock manager // assertBn(await staking.unlockedBalanceOf(user1), bn(0), "Unlocked balance should match") // assertBn(await staking.lockedBalanceOf(user1), bn(0), "total locked doesn’t match") // // recipient // assertBn(await staking.unlockedBalanceOf(user2), transferAmount, "Unlocked balance should match") // assertBn(await staking.lockedBalanceOf(user2), bn(0), "total locked doesn’t match") // }) // it('transfers (slash) and unlocks in one transaction', async () => { // const totalLock = bigExp(120, 18) // const transferAmount = bigExp(40, 18) // const decreaseAmount = bigExp(60, 18) // await approveStakeAndLock({ staking, manager: user1, allowanceAmount: totalLock, lockAmount: totalLock, stakeAmount: totalLock, from: owner }) // // unlock and transfer // await staking.slashAndUnlock(owner, user2, decreaseAmount, transferAmount, { from: user1 }) // assertBn(await staking.unlockedBalanceOf(owner), decreaseAmount, "Unlocked balance should match") // assertBn(await staking.lockedBalanceOf(owner), totalLock.sub(decreaseAmount).sub(transferAmount), "total locked doesn’t match") // // lock manager // assertBn(await staking.unlockedBalanceOf(user1), bn(0), "Unlocked balance should match") // assertBn(await staking.lockedBalanceOf(user1), bn(0), "total locked doesn’t match") // // recipient // assertBn(await staking.unlockedBalanceOf(user2), transferAmount, "Unlocked balance should match") // assertBn(await staking.lockedBalanceOf(user2), bn(0), "total locked doesn’t match") // }) // it('fails to transfer (slash) and unlocks in one transaction if unlock amount is zero', async () => { // const totalLock = bigExp(120, 18) // const transferAmount = bigExp(40, 18) // const decreaseAmount = bigExp(0, 18) // await approveStakeAndLock({ staking, manager: user1, allowanceAmount: totalLock, lockAmount: totalLock, stakeAmount: totalLock, from: owner }) // // unlock and transfer // await assertRevert(staking.slashAndUnlock(owner, user2, decreaseAmount, transferAmount, { from: user1 }), STAKING_ERRORS.ERROR_AMOUNT_ZERO) // }) // it('fails to transfer (slash) and unlock in one transaction if not owner nor manager', async () => { // const totalLock = bigExp(120, 18) // const transferAmount = bigExp(40, 18) // const decreaseAmount = bigExp(60, 18) // await approveStakeAndLock({ staking, manager: user1, allowanceAmount: totalLock, lockAmount: totalLock, stakeAmount: totalLock, from: owner }) // // unlock and transfer // await assertRevert(staking.slashAndUnlock(owner, user2, decreaseAmount, transferAmount, { from: user2 }), STAKING_ERRORS.ERROR_NOT_ENOUGH_LOCK) // }) // it('change lock amount', async () => { // await approveStakeAndLock({ staking, manager: lockManager.address, from: owner }) // const { _amount: amount1 } = await staking.getLock(owner, lockManager.address) // assertBn(amount1, bn(DEFAULT_LOCK_AMOUNT), "Amount should match") // assertBn(await staking.unlockedBalanceOf(owner), DEFAULT_STAKE_AMOUNT.sub(DEFAULT_LOCK_AMOUNT), "Unlocked balance should match") // // change amount // const unlockAmount = DEFAULT_LOCK_AMOUNT.div(bn(2)) // await lockManager.unlock(staking.address, owner, unlockAmount) // const { _amount: amount2 } = await staking.getLock(owner, lockManager.address) // assertBn(amount2, unlockAmount, "Amount should match") // assertBn(await staking.unlockedBalanceOf(owner), DEFAULT_STAKE_AMOUNT.sub(unlockAmount), "Unlocked balance should match") // }) // it('fails to change lock amount to zero', async () => { // await approveStakeAndLock({ staking, manager: lockManager.address, from: owner }) // // try to change amount // await assertRevert(lockManager.unlock(staking.address, owner, 0), STAKING_ERRORS.ERROR_AMOUNT_ZERO) // }) // it('fails to change lock amount to greater than before', async () => { // await approveStakeAndLock({ staking, manager: lockManager.address, from: owner }) // // try to change amount // await assertRevert(lockManager.unlock(staking.address, owner, DEFAULT_LOCK_AMOUNT.add(bn(1))), STAKING_ERRORS.ERROR_NOT_ENOUGH_LOCK) // }) // it('change lock manager', async () => { // await approveStakeAndLock({ staking, manager: user1, from: owner }) // assert.equal(await staking.canUnlock(user1, owner, user1, 0), true, "User 1 can unlock") // assert.equal(await staking.canUnlock(user2, owner, user1, 0), false, "User 2 can not unlock") // await assertRevert(staking.canUnlock(user2, owner, user2, 0), STAKING_ERRORS.ERROR_LOCK_DOES_NOT_EXIST) // it doesn’t exist // // change manager // await staking.setLockManager(owner, user2, { from: user1 }) // await assertRevert(staking.canUnlock(user1, owner, user1, 0), STAKING_ERRORS.ERROR_LOCK_DOES_NOT_EXIST) // it doesn’t exist // assert.equal(await staking.canUnlock(user1, owner, user2, 0), false, "User 1 can not unlock") // assert.equal(await staking.canUnlock(user2, owner, user2, 0), true, "User 2 can unlock") // }) // it('fails to change lock manager if it doesn’t exist', async () => { // await assertRevert(staking.setLockManager(owner, user2, { from: user1 }), STAKING_ERRORS.ERROR_LOCK_DOES_NOT_EXIST) // }) })