Skip to content
This repository was archived by the owner on Mar 1, 2024. It is now read-only.

Rewards on curve #376

Open
wants to merge 20 commits into
base: main
Choose a base branch
from
Open
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
359 changes: 244 additions & 115 deletions contracts/staking/stakeManager/StakeManager.sol

Large diffs are not rendered by default.

8 changes: 6 additions & 2 deletions contracts/staking/stakeManager/StakeManagerStorage.sol
Original file line number Diff line number Diff line change
@@ -23,11 +23,13 @@ contract StakeManagerStorage is GovernanceLockable, RootChainable {
struct State {
uint256 amount;
uint256 stakerCount;
uint256 shares;
}

struct StateChange {
int256 amount;
int256 stakerCount;
int256 shares;
}

struct Validator {
@@ -43,12 +45,14 @@ contract StakeManagerStorage is GovernanceLockable, RootChainable {
uint256 lastCommissionUpdate;
uint256 delegatorsReward;
uint256 delegatedAmount;
uint256 initialRewardPerStake;
uint256 initialRewardPerShare;
}

uint256 constant MAX_COMMISION_RATE = 100;
uint256 constant MAX_PROPOSER_BONUS = 100;
uint256 constant CHK_REWARD_PRECISION = 100;
uint256 constant REWARD_PRECISION = 10**25;

uint256 internal constant INCORRECT_VALIDATOR_ID = 2**256 - 1;
uint256 internal constant INITIALIZED_AMOUNT = 1;

@@ -68,7 +72,7 @@ contract StakeManagerStorage is GovernanceLockable, RootChainable {
uint256 public checkPointBlockInterval;
uint256 public signerUpdateLimit;

uint256 public validatorThreshold; //128
uint256 public validatorThreshold; // 128
uint256 public totalStaked;
uint256 public NFTCounter;
uint256 public totalRewards;
15 changes: 13 additions & 2 deletions contracts/staking/stakeManager/StakeManagerStorageExtension.sol
Original file line number Diff line number Diff line change
@@ -1,17 +1,28 @@
pragma solidity 0.5.17;

contract StakeManagerStorageExtension {
uint256 public constant SHARES_PRECISION = 10**12; // will cause and overflow if ALL tokens will be staked at once, which will won't matter at this point anyway

struct StakeSharesState {
uint256 sharesPool;
uint256 stakePool;
uint256 shares;
}

address public eventsHub;
uint256 public rewardPerStake;
uint256 public rewardPerShare;
address public extensionCode;
address[] public signers;

uint256 constant CHK_REWARD_PRECISION = 100;
uint256 public prevBlockInterval;
// how much less reward per skipped checkpoint, 0 - 100%
uint256 public rewardDecreasePerCheckpoint;
// how many checkpoints to reward
uint256 public maxRewardedCheckpoints;
// increase / decrease value for faster or slower checkpoints, 0 - 100%
uint256 public checkpointRewardDelta;
// constant for stake shares curve
uint256 public sharesCurvature;
// validator Id => state
mapping(uint256 => StakeSharesState) public sharesState;
}
6,746 changes: 3,383 additions & 3,363 deletions package-lock.json

Large diffs are not rendered by default.

21 changes: 10 additions & 11 deletions package.json
Original file line number Diff line number Diff line change
@@ -27,8 +27,7 @@
"lint:sol:fix": "solium -d contracts/ --fix",
"lint:sol": "solium -d contracts/"
},
"pre-commit": [
],
"pre-commit": [],
"repository": {
"type": "git",
"url": "https://github.com/maticnetwork/contracts.git"
@@ -43,7 +42,7 @@
"mnemonics": "clock radar mass judge dismiss just intact mind resemble fringe diary casino"
},
"devDependencies": {
"@openzeppelin/test-helpers": "^0.5.5",
"@openzeppelin/test-helpers": "^0.5.10",
"babel-cli": "^6.26.0",
"babel-eslint": "10.0.1",
"babel-polyfill": "^6.26.0",
@@ -62,31 +61,31 @@
"eslint-plugin-promise": "4.0.1",
"eslint-plugin-standard": "4.0.0",
"eth-gas-reporter": "0.1.12",
"eth-sig-util": "^2.1.1",
"eth-sig-util": "^2.5.4",
"ethereumjs-wallet": "0.6.2",
"ethers": "4.0.13",
"moonwalker": "github:atvanguard/moonwalker#4c4592e2d6c1744fbcd9452cf0f00e9beb236b21",
"pre-commit": "^1.2.2",
"safe-buffer": "5.1.2",
"solidity-coverage": "^0.7.4",
"solidity-coverage": "^0.7.16",
"solium": "^1.1.8",
"truffle": "5.0.34",
"web3": "^1.0.0-beta.33",
"web3-eth-abi": "^1.0.0-beta.51",
"truffle-contract-size": "^1.0.1"
"truffle-contract-size": "^1.0.1",
"web3": "^1.3.5",
"web3-eth-abi": "^1.3.5"
},
"dependencies": {
"commander": "^4.0.1",
"ethereumjs-abi": "^0.6.5",
"ethereumjs-block": "2.1.0",
"ethereumjs-tx": "1.3.7",
"ethereumjs-util": "6.0.0",
"ganache-cli": "^6.8.0-istanbul.0",
"ganache-cli": "^6.12.2",
"glob": "^7.1.6",
"merkle-patricia-tree": "2.3.2",
"nunjucks": "^3.2.0",
"nunjucks": "^3.2.3",
"openzeppelin-solidity": "2.2.0",
"solidity-rlp": "^2.0.0",
"solidity-rlp": "^2.0.4",
"truffle-hdwallet-provider": "^1.0.17",
"truffle-plugin-verify": "^0.3.8"
}
54 changes: 29 additions & 25 deletions scripts/process-templates.js
Original file line number Diff line number Diff line change
@@ -1,39 +1,43 @@
const program = require("commander");
const nunjucks = require("nunjucks");
const glob = require("glob");
const fs = require("fs");
const path = require("path");
const program = require('commander')
const nunjucks = require('nunjucks')
const glob = require('glob')
const fs = require('fs')
const path = require('path')

program.version("0.0.1");
program.option("-c, --bor-chain-id <bor-chain-id>", "Bor chain id", "15001");
program.parse(process.argv);
program.version('0.0.1')
program.option('-c, --bor-chain-id <bor-chain-id>', 'Bor chain id', '15001')
program.parse(process.argv)

//joining path of directory
const directoryPath = path.join(__dirname, "..", "**/*.template");
//passsing directoryPath and callback function
glob(directoryPath, function (err, files) {
//handling error
// joining path of directory
const directoryPath = path.join(__dirname, '..', '**/*.template')
// passsing directoryPath and callback function
glob(directoryPath, function(err, files) {
// handling error
if (err) {
return console.log("Unable to scan directory: " + err);
return console.log('Unable to scan directory: ' + err)
}

//listing all files using forEach
files.forEach(function (file) {
// listing all files using forEach
files.forEach(function(file) {
if (file.includes('node_modules')) {
return
}

// Do whatever you want to do with the file
const borChainIdHex = parseInt(program.borChainId, 10)
.toString(16)
.toUpperCase();
.toUpperCase()

const data = {
borChainId: program.borChainId,
borChainIdHex:
borChainIdHex.length % 2 !== 0 ? `0${borChainIdHex}` : borChainIdHex,
};
borChainIdHex.length % 2 !== 0 ? `0${borChainIdHex}` : borChainIdHex
}

const templateString = fs.readFileSync(file).toString();
const resultString = nunjucks.renderString(templateString, data);
fs.writeFileSync(file.replace(".template", ""), resultString);
});
const templateString = fs.readFileSync(file).toString()
const resultString = nunjucks.renderString(templateString, data)
fs.writeFileSync(file.replace('.template', ''), resultString)
})

console.log("All template files have been processed.");
});
console.log('All template files have been processed.')
})
29 changes: 24 additions & 5 deletions test/helpers/deployer.js
Original file line number Diff line number Diff line change
@@ -29,6 +29,8 @@ class Deployer {
proxy.address
)

contracts.EventsHub.address = proxy.address

return contracts.EventsHub.at(proxy.address)
}

@@ -39,6 +41,9 @@ class Deployer {
this.eventsHub = await this.deployEventsHub(this.registry.address)
this.validatorShareFactory = await contracts.ValidatorShareFactory.new()
this.stakeToken = await contracts.TestToken.new('Stake Token', 'ST')

contracts.TestToken.address = this.stakeToken.address

this.stakingInfo = await contracts.StakingInfo.new(this.registry.address)
this.slashingManager = await contracts.SlashingManager.new(this.registry.address, this.stakingInfo.address, 'heimdall-P5rXwg')
this.rootChain = await this.deployRootChain()
@@ -95,7 +100,7 @@ class Deployer {
return _contracts
}

buildStakeManagerObject(stakeManager, governance) {
async buildStakeManagerObject(stakeManager, governance) {
stakeManager.updateDynastyValue = (val) => {
return governance.update(
stakeManager.address,
@@ -171,6 +176,15 @@ class Deployer {
stakeManager.contract.methods.updateCheckpointRewardParams(val1, val2, val3).encodeABI()
)
}

// built-in extensions code
const extensions = await contracts.StakeManagerExtension.at(stakeManager.address)
stakeManager.setSharesK = (...args) => {
return governance.update(
stakeManager.address,
extensions.contract.methods.setSharesK(...args).encodeABI()
)
}
}

async deployStakeManager(wallets) {
@@ -182,7 +196,12 @@ class Deployer {
this.validatorShare = await contracts.ValidatorShare.new()
this.rootChain = await this.deployRootChain()
this.stakingInfo = await contracts.StakingInfo.new(this.registry.address)

contracts.StakingInfo.address = this.stakingInfo.address

this.stakeToken = await contracts.TestToken.new('Stake Token', 'STAKE')
contracts.TestToken.address = this.stakeToken.address

this.stakingNFT = await contracts.StakingNFT.new('Matic Validator', 'MV')

let stakeManager = await contracts.StakeManagerTestable.new()
@@ -204,7 +223,7 @@ class Deployer {
).encodeABI())

this.stakeManager = await contracts.StakeManagerTestable.at(proxy.address)
this.buildStakeManagerObject(this.stakeManager, this.governance)
await this.buildStakeManagerObject(this.stakeManager, this.governance)
this.slashingManager = await contracts.SlashingManager.new(this.registry.address, this.stakingInfo.address, 'heimdall-P5rXwg')

await this.stakingNFT.transferOwnership(this.stakeManager.address)
@@ -460,7 +479,7 @@ class Deployer {
false /* isERC721 */
)
}
return { rootERC20, childToken, childTokenProxy }
return { rootERC20, childToken, childTokenProxy }
}

async deployMaticToken() {
@@ -608,9 +627,9 @@ class Deployer {
let proxy = await contracts.GnosisSafeProxy.new(gnosisSafe.address)
gnosisSafe = await contracts.GnosisSafe.at(proxy.address)
await gnosisSafe.setup(
[...signers], 2, utils.ZeroAddress, "0x", utils.ZeroAddress, utils.ZeroAddress, 0, utils.ZeroAddress
[...signers], 2, utils.ZeroAddress, '0x', utils.ZeroAddress, utils.ZeroAddress, 0, utils.ZeroAddress
)
return gnosisSafe
return gnosisSafe
}
}

35 changes: 35 additions & 0 deletions test/units/behaviors/stakeShares.behavior.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,35 @@
import { assertBigNumberEquality } from '../../helpers/utils'

export function shouldHaveCorrectStakeShares({ shares, validatorId, user } = {}) {
it(`should have correct validator shares`, async function() {
const sharesState = await this.stakeManager.sharesState(validatorId || this.validatorId)
// console.log('sharesState.shares', sharesState.shares.toString())
// console.log('sharesState.sharesPool', sharesState.sharesPool.toString())
// console.log('sharesState.stakePool', sharesState.stakePool.toString())

assertBigNumberEquality(shares || this.shares, sharesState.shares)
})

it(`should have correct shares pool`, async function() {
const sharesCurvature = await this.stakeManager.sharesCurvature()
const sharesPrecision = await this.stakeManager.SHARES_PRECISION()
const sharesState = await this.stakeManager.sharesState(validatorId || this.validatorId)

assertBigNumberEquality(sharesState.shares.add(sharesState.sharesPool), sharesCurvature.mul(sharesPrecision))
})

it(`should have correct stake pool`, async function() {
const sharesCurvature = await this.stakeManager.sharesCurvature()
const sharesState = await this.stakeManager.sharesState(validatorId || this.validatorId)
const stakedFor = await this.stakeManager.totalStakedFor(user || this.user)

assertBigNumberEquality(sharesState.stakePool.sub(stakedFor), sharesCurvature)
})
}

export function shouldHaveCorrectTotalStakeShares({ totalShares } = {}) {
it('should have correct total stake shares in the network', async function() {
const state = await this.stakeManager.validatorState()
assertBigNumberEquality(state.shares, totalShares || this.totalShares)
})
}
2 changes: 1 addition & 1 deletion test/units/staking/SlashingManager.test.js
Original file line number Diff line number Diff line change
@@ -13,7 +13,7 @@ import { buyVoucher } from './ValidatorShareHelper.js'
import { expectEvent, expectRevert } from '@openzeppelin/test-helpers'
import { BN } from 'ethereumjs-util'

contract('Slashing:validator', async function(accounts) {
contract.skip('Slashing:validator', async function(accounts) {
let stakeToken
let stakeManager
let slashingManager
140 changes: 70 additions & 70 deletions test/units/staking/ValidatorShare.test.js

Large diffs are not rendered by default.

2 changes: 1 addition & 1 deletion test/units/staking/deployment.js
Original file line number Diff line number Diff line change
@@ -21,7 +21,7 @@ export const walletAmounts = {
[wallets[2].getAddressString()]: {
amount: web3.utils.toWei('250'),
stakeAmount: web3.utils.toWei('150'),
restakeAmonut: web3.utils.toWei('100'),
restakeAmount: web3.utils.toWei('100'),
initialBalance: web3.utils.toWei('805')
},
[wallets[3].getAddressString()]: {
190 changes: 130 additions & 60 deletions test/units/staking/stakeManager/StakeManager.Staking.js
Original file line number Diff line number Diff line change
@@ -6,6 +6,9 @@ import {
} from '../../../helpers/utils.js'
import { expectEvent, expectRevert, BN } from '@openzeppelin/test-helpers'
import { wallets, walletAmounts, freshDeploy, approveAndStake } from '../deployment'
import { shouldHaveCorrectStakeShares, shouldHaveCorrectTotalStakeShares } from '../../behaviors/stakeShares.behavior'

const { toWei, toBN } = web3.utils

module.exports = function(accounts) {
let owner = accounts[0]
@@ -76,7 +79,7 @@ module.exports = function(accounts) {
})
}

function testStake(user, userPubkey, amount, stakeAmount, validatorId, fee) {
function testStake({ user, userPubkey, amount, stakeAmount, validatorId, fee, shares }) {
before('Approve', async function() {
this.user = user
this.fee = new BN(fee || this.defaultHeimdallFee)
@@ -114,6 +117,8 @@ module.exports = function(accounts) {
assertBigNumberEquality(stakedFor, stakeAmount)
})

shouldHaveCorrectStakeShares({ shares, validatorId, user })

it('must have correct total staked balance', async function() {
const stake = await this.stakeManager.currentValidatorSetTotalStake()
assertBigNumberEquality(stake, stakeAmount)
@@ -131,7 +136,7 @@ module.exports = function(accounts) {
})
}

function testRestake(user, amount, stakeAmount, restakeAmount, totalStaked) {
function testRestake({ user, amount, stakeAmount, restakeAmount, totalStaked, shares, validatorId }) {
before('Approve', async function() {
this.user = user

@@ -167,20 +172,23 @@ module.exports = function(accounts) {
const stakedFor = await this.stakeManager.totalStakedFor(user)
assertBigNumberEquality(stakedFor, stakeAmount)
})

shouldHaveCorrectStakeShares({ shares, validatorId, user })
}

describe('double stake', async function() {
before(freshDeploy)

describe('when stakes first time', function() {
const amounts = walletAmounts[wallets[1].getAddressString()]
testStake(
wallets[1].getChecksumAddressString(),
wallets[1].getPublicKeyString(),
amounts.amount,
amounts.stakeAmount,
1
)
testStake({
user: wallets[1].getChecksumAddressString(),
userPubkey: wallets[1].getPublicKeyString(),
amount: amounts.amount,
stakeAmount: amounts.stakeAmount,
validatorId: 1,
shares: '199999999999999999771428571428571'
})
})

describe('when stakes again', function() {
@@ -200,13 +208,15 @@ module.exports = function(accounts) {
before('Stake', doStake(wallets[2]))

describe('when restakes', function() {
testRestake(
wallets[2].getChecksumAddressString(),
amounts.restakeAmonut,
amounts.amount,
amounts.restakeAmonut,
amounts.amount
)
testRestake({
user: wallets[2].getChecksumAddressString(),
amount: amounts.restakeAmount,
stakeAmount: amounts.amount,
restakeAmount: amounts.restakeAmount,
totalStaked: amounts.amount,
validatorId: '1',
shares: '249999999999999999642857142857142'
})
})

describe('when stakes again', function() {
@@ -217,41 +227,51 @@ module.exports = function(accounts) {
web3.utils.toWei('150')
)
})
describe('when reStakes while on going auction', function() {
it('when auction is active', async function() {
let auctionBid = web3.utils.toWei('10000')
const auctionUser = wallets[4].getAddressString()
await this.stakeToken.mint(auctionUser, auctionBid)
await this.stakeToken.approve(this.stakeManager.address, auctionBid, {
from: auctionUser
})
const validatorId = await this.stakeManager.getValidatorId(wallets[2].getChecksumAddressString())
await this.stakeManager.startAuction(validatorId, auctionBid, false, wallets[4].getPublicKeyString(), {
from: auctionUser
})
testRestake(
wallets[2].getChecksumAddressString(),
amounts.restakeAmonut,
amounts.amount,
amounts.restakeAmonut,
amounts.amount
)
})

describe('when restakes while on going auction', function() {
before(freshDeploy)

const amounts = walletAmounts[wallets[2].getAddressString()]
before('Stake', doStake(wallets[2]))

before(async function() {
let auctionBid = web3.utils.toWei('10000')
const auctionUser = wallets[4].getAddressString()
await this.stakeToken.mint(auctionUser, auctionBid)
await this.stakeToken.approve(this.stakeManager.address, auctionBid, {
from: auctionUser
})
const validatorId = await this.stakeManager.getValidatorId(wallets[2].getChecksumAddressString())
await this.stakeManager.startAuction(validatorId, auctionBid, false, wallets[4].getPublicKeyString(), {
from: auctionUser
})
})

testRestake({
user: wallets[2].getChecksumAddressString(),
amount: amounts.restakeAmount,
stakeAmount: amounts.amount,
restakeAmount: amounts.restakeAmount,
totalStaked: amounts.amount,
validatorId: '1',
shares: '249999999999999999642857142857142'
})
})

describe('stake beyond validator threshold', async function() {
before(prepareForTest(2, 1))

describe('when user stakes', function() {
const amounts = walletAmounts[wallets[3].getAddressString()]
testStake(
wallets[3].getChecksumAddressString(),
wallets[3].getPublicKeyString(),
amounts.amount,
amounts.stakeAmount,
1
)
testStake({
user: wallets[3].getChecksumAddressString(),
userPubkey: wallets[3].getPublicKeyString(),
amount: amounts.amount,
stakeAmount: amounts.stakeAmount,
validatorId: 1,
shares: '299999999999999999485714285714285'
})
})

describe('when other user stakes beyond validator threshold', function() {
@@ -284,14 +304,15 @@ module.exports = function(accounts) {
describe('stake with heimdall fee', function() {
before(freshDeploy)

testStake(
wallets[0].getChecksumAddressString(),
wallets[0].getPublicKeyString(),
web3.utils.toWei('200'),
web3.utils.toWei('150'),
1,
web3.utils.toWei('50')
)
testStake({
user: wallets[0].getChecksumAddressString(),
userPubkey: wallets[0].getPublicKeyString(),
amount: web3.utils.toWei('200'),
stakeAmount: web3.utils.toWei('150'),
validatorId: 1,
shares: '149999999999999999871428571428571',
fee: web3.utils.toWei('50')
})
})

describe('when Alice stakes, change signer and stakes with old signer', function() {
@@ -364,6 +385,9 @@ module.exports = function(accounts) {
const balance = await this.stakeToken.balanceOf(user)
assertBigNumberEquality(balance, this.afterStakeBalance.add(this.reward))
})

shouldHaveCorrectStakeShares({ shares: '149999999999999999871428571428571', user })
shouldHaveCorrectTotalStakeShares({ totalShares: '149999999999999999871428571428571' })
})

describe('when user unstakes after 2 epochs', async function() {
@@ -387,8 +411,7 @@ module.exports = function(accounts) {
})

it('must unstake', async function() {
const validatorId = await this.stakeManager.getValidatorId(user)
this.receipt = await this.stakeManager.unstake(validatorId, {
this.receipt = await this.stakeManager.unstake(this.validatorId, {
from: user
})
})
@@ -421,6 +444,9 @@ module.exports = function(accounts) {
const balance = await this.stakeToken.balanceOf(user)
assertBigNumberEquality(balance, this.afterStakeBalance.add(this.reward))
})

shouldHaveCorrectStakeShares({ shares: '299999999999999999485714285714285', user })
shouldHaveCorrectTotalStakeShares({ totalShares: '299999999999999999485714285714285' })
})

describe('reverts', function() {
@@ -488,7 +514,7 @@ module.exports = function(accounts) {
})
})

describe('when user claims after 1 epoch and 1 dynasty passed', function() {
describe('when Alice claims after 1 epoch and 1 dynasty passed', function() {
let dynasties = 1
const Alice = wallets[2].getChecksumAddressString()

@@ -528,6 +554,9 @@ module.exports = function(accounts) {
const stake = await this.stakeManager.currentValidatorSetTotalStake()
assertBigNumberEquality(stake, walletAmounts[wallets[3].getAddressString()].stakeAmount)
})

shouldHaveCorrectStakeShares({ shares: '0', user: Alice })
shouldHaveCorrectTotalStakeShares({ totalShares: '299999999999999999485714285714285' })
})

describe('when user claims next epoch', function() {
@@ -596,6 +625,7 @@ module.exports = function(accounts) {

describe('when Alice claims', function() {
const user = Alice.getAddressString()
const reward = '2999999999999999998285'

it('must claim', async function() {
this.validatorId = await this.stakeManager.getValidatorId(user)
@@ -606,13 +636,13 @@ module.exports = function(accounts) {
})

it('must have correct reward', async function() {
assertBigNumberEquality(this.reward, web3.utils.toWei('3000'))
assertBigNumberEquality(this.reward, reward)
})

it('must emit ClaimRewards', async function() {
await expectEvent.inTransaction(this.receipt.tx, StakingInfo, 'ClaimRewards', {
validatorId: this.validatorId,
amount: web3.utils.toWei('3000'),
amount: reward,
totalAmount: await this.stakeManager.totalRewardsLiquidated()
})
})
@@ -621,10 +651,13 @@ module.exports = function(accounts) {
let balance = await this.stakeToken.balanceOf(user)
assertBigNumberEquality(balance, new BN(walletAmounts[user].initialBalance).add(this.reward).sub(this.defaultHeimdallFee))
})

shouldHaveCorrectStakeShares({ shares: '0', user: Alice.getAddressString(), validatorId: '1' })
})

describe('when Bob claims', function() {
const user = Bob.getAddressString()
const reward = '2999999999999999998285'

it('must claim', async function() {
this.validatorId = await this.stakeManager.getValidatorId(user)
@@ -635,13 +668,13 @@ module.exports = function(accounts) {
})

it('must have correct reward', async function() {
assertBigNumberEquality(this.reward, web3.utils.toWei('3000'))
assertBigNumberEquality(this.reward, reward)
})

it('must emit ClaimRewards', async function() {
await expectEvent.inTransaction(this.receipt.tx, StakingInfo, 'ClaimRewards', {
validatorId: this.validatorId,
amount: web3.utils.toWei('3000'),
amount: reward,
totalAmount: await this.stakeManager.totalRewardsLiquidated()
})
})
@@ -650,6 +683,8 @@ module.exports = function(accounts) {
let balance = await this.stakeToken.balanceOf(user)
assertBigNumberEquality(balance, new BN(walletAmounts[user].initialBalance).add(this.reward).sub(this.defaultHeimdallFee))
})

shouldHaveCorrectStakeShares({ shares: '0', user: Bob.getAddressString(), validatorId: '2' })
})

describe('afterwards verification', function() {
@@ -666,8 +701,11 @@ module.exports = function(accounts) {
it('Eve must have correct rewards', async function() {
const validatorId = await this.stakeManager.getValidatorId(Eve.getAddressString())
this.reward = await this.stakeManager.validatorReward(validatorId)
assertBigNumberEquality(this.reward, web3.utils.toWei('12000'))
assertBigNumberEquality(this.reward, '11999999999999999993142')
})

shouldHaveCorrectStakeShares({ shares: '99999999999999999942857142857142', user: Eve.getAddressString(), validatorId: '3' })
shouldHaveCorrectTotalStakeShares({ totalShares: '99999999999999999942857142857142' })
})
})
})
@@ -705,12 +743,12 @@ module.exports = function(accounts) {

for (let i = currentEpoch; i <= auctionPeriod; i++) {
await checkPoint(initialStakers, this.rootChainOwner, this.stakeManager)
console.log('checkpoint')
}
// without 10% proposer bonus
this.validatorReward = checkpointReward.mul(new BN(100 - proposerBonus)).div(new BN(100)).mul(new BN(auctionPeriod - currentEpoch))
this.validatorReward = this.validatorReward || new BN(0)
this.validatorId = '1'
this.user = initialStakers[0].getAddressString()
this.amount = web3.utils.toWei('100')

await this.stakeToken.mint(this.user, this.amount)
await this.stakeToken.approve(this.stakeManager.address, this.amount, {
@@ -766,29 +804,61 @@ module.exports = function(accounts) {
assertBigNumberEquality(reward, this.oldReward)
})
}

shouldHaveCorrectStakeShares()
shouldHaveCorrectTotalStakeShares()
}

describe('with delegation', function() {
describe('with rewards', function() {
before(function() {
this.shares = '10099999999999999365656714285714324'
this.totalShares = '11099999999999999359942428571428609'
this.amount = toWei('100')
this.validatorReward = new BN('8999999999999999948571')
})

testRestake(true, true)
})

describe('without rewards', function() {
before(function() {
this.shares = '1099999999999999993085714285714285'
this.totalShares = '2099999999999999987371428571428570'
this.amount = toWei('100')
})

testRestake(true, false)
})
})

describe('without delegation', function() {
describe('with rewards', function() {
before(function() {
this.shares = '10099999999999999365656714285714324'
this.totalShares = '11099999999999999359942428571428609'
this.amount = toWei('100')
this.validatorReward = new BN('8999999999999999948571')
})

testRestake(false, true)
})

describe('without rewards', function() {
before(function() {
this.shares = '1099999999999999993085714285714285'
this.totalShares = '2099999999999999987371428571428570'
this.amount = toWei('100')
})
testRestake(false, false)
})
})

describe('reverts', function() {
before(function() {
this.amount = toWei('100')
})

before(doDeploy(false))

it('when validatorId is incorrect', async function() {
952 changes: 526 additions & 426 deletions test/units/staking/stakeManager/StakeManager.test.js

Large diffs are not rendered by default.