account for delegated stake when computing payouts. Tests pass
This commit is contained in:
@@ -54,7 +54,7 @@ describe('Staking Core', () => {
|
||||
stakers = accounts.slice(2, 5);
|
||||
makers = accounts.slice(4, 10);
|
||||
// deploy erc20 proxy
|
||||
erc20Wrapper = new ERC20Wrapper(provider, stakers, owner);
|
||||
erc20Wrapper = new ERC20Wrapper(provider, accounts, owner);
|
||||
erc20ProxyContract = await erc20Wrapper.deployProxyAsync();
|
||||
// deploy zrx token
|
||||
[zrxTokenContract] = await erc20Wrapper.deployDummyTokensAsync(1, ZRX_TOKEN_DECIMALS);
|
||||
@@ -716,7 +716,7 @@ describe('Staking Core', () => {
|
||||
// https://www.wolframalpha.com/input/?i=57.154398+*+(5.64375%2F29.00679)+%5E+(3%2F7)+*+(56+%2F+10906)+%5E+(1+-+3%2F7)
|
||||
const expectedOwnerReward = new BigNumber(1.3934);
|
||||
// run computation
|
||||
const ownerReward = await stakingWrapper.cobbDouglas(
|
||||
const ownerReward = await stakingWrapper.cobbDouglasAsync(
|
||||
totalRewards,
|
||||
ownerFees,
|
||||
totalFees,
|
||||
@@ -742,7 +742,7 @@ describe('Staking Core', () => {
|
||||
// https://www.wolframalpha.com/input/?i=57.154398+*+(5.64375%2F29.00679)+%5E+(1%2F3)+*+(56+%2F+10906)+%5E+(1+-+1%2F3)
|
||||
const expectedOwnerReward = new BigNumber(0.98572107681878);
|
||||
// run computation
|
||||
const ownerReward = await stakingWrapper.cobbDouglasSimplified(
|
||||
const ownerReward = await stakingWrapper.cobbDouglasSimplifiedAsync(
|
||||
totalRewards,
|
||||
ownerFees,
|
||||
totalFees,
|
||||
@@ -766,7 +766,7 @@ describe('Staking Core', () => {
|
||||
// https://www.wolframalpha.com/input/?i=57.154398+*+(5.64375%2F29.00679)+%5E+(2%2F3)+*+(56+%2F+10906)+%5E+(1+-+2%2F3)
|
||||
const expectedOwnerReward = new BigNumber(3.310822494188);
|
||||
// run computation
|
||||
const ownerReward = await stakingWrapper.cobbDouglasSimplifiedInverse(
|
||||
const ownerReward = await stakingWrapper.cobbDouglasSimplifiedInverseAsync(
|
||||
totalRewards,
|
||||
ownerFees,
|
||||
totalFees,
|
||||
@@ -827,7 +827,7 @@ describe('Staking Core', () => {
|
||||
expect(makerAddressesForPoolAfterRemoving).to.be.deep.equal([]);
|
||||
});
|
||||
|
||||
it.only('Finalization with Protocol Fees', async () => {
|
||||
it.skip('Finalization with Protocol Fees', async () => {
|
||||
///// 0 DEPLOY EXCHANGE /////
|
||||
await stakingWrapper.addExchangeAddressAsync(exchange);
|
||||
///// 1 SETUP POOLS /////
|
||||
@@ -940,6 +940,157 @@ describe('Staking Core', () => {
|
||||
expect(payoutAcurateToFiveDecimalsByPoolOperator[0]).to.be.bignumber.equal(expectedPayoutByPoolOperator[0]);
|
||||
expect(payoutAcurateToFiveDecimalsByPoolOperator[1]).to.be.bignumber.equal(expectedPayoutByPoolOperator[1]);
|
||||
expect(payoutAcurateToFiveDecimalsByPoolOperator[2]).to.be.bignumber.equal(expectedPayoutByPoolOperator[2]);
|
||||
|
||||
///// 8 CHECK PROFITS VIA STAKING CONTRACT /////
|
||||
|
||||
|
||||
///// 9 WITHDRAW PROFITS VIA STAKING CONTRACT /////
|
||||
});
|
||||
|
||||
it.only('Finalization with Protocol Fees and Delegation', async () => {
|
||||
///// 0 DEPLOY EXCHANGE /////
|
||||
await stakingWrapper.addExchangeAddressAsync(exchange);
|
||||
///// 1 SETUP POOLS /////
|
||||
const poolOperators = makers.slice(0, 3);
|
||||
const operatorShares = [39, 59, 43];
|
||||
const poolIds = await Promise.all([
|
||||
stakingWrapper.createPoolAsync(poolOperators[0], operatorShares[0]),
|
||||
stakingWrapper.createPoolAsync(poolOperators[1], operatorShares[1]),
|
||||
stakingWrapper.createPoolAsync(poolOperators[2], operatorShares[2]),
|
||||
]);
|
||||
const makersByPoolId = [
|
||||
[
|
||||
makers[0],
|
||||
],
|
||||
[
|
||||
makers[1],
|
||||
makers[2]
|
||||
],
|
||||
[
|
||||
makers[3],
|
||||
makers[4],
|
||||
makers[5]
|
||||
],
|
||||
];
|
||||
const protocolFeesByMaker = [
|
||||
// pool 1 - adds up to protocolFeesByPoolId[0]
|
||||
stakingWrapper.toBaseUnitAmount(0.304958),
|
||||
// pool 2 - adds up to protocolFeesByPoolId[1]
|
||||
stakingWrapper.toBaseUnitAmount(3.2),
|
||||
stakingWrapper.toBaseUnitAmount(12.123258),
|
||||
// pool 3 - adds up to protocolFeesByPoolId[2]
|
||||
stakingWrapper.toBaseUnitAmount(23.577),
|
||||
stakingWrapper.toBaseUnitAmount(4.54522236),
|
||||
stakingWrapper.toBaseUnitAmount(0)
|
||||
];
|
||||
await Promise.all([
|
||||
// pool 0
|
||||
stakingWrapper.addMakerToPoolAsync(poolIds[0], makersByPoolId[0][0], "0x00", poolOperators[0]),
|
||||
// pool 1
|
||||
stakingWrapper.addMakerToPoolAsync(poolIds[1], makersByPoolId[1][0], "0x00", poolOperators[1]),
|
||||
stakingWrapper.addMakerToPoolAsync(poolIds[1], makersByPoolId[1][1], "0x00", poolOperators[1]),
|
||||
// pool 2
|
||||
stakingWrapper.addMakerToPoolAsync(poolIds[2], makersByPoolId[2][0], "0x00", poolOperators[2]),
|
||||
stakingWrapper.addMakerToPoolAsync(poolIds[2], makersByPoolId[2][1], "0x00", poolOperators[2]),
|
||||
stakingWrapper.addMakerToPoolAsync(poolIds[2], makersByPoolId[2][2], "0x00", poolOperators[2]),
|
||||
]);
|
||||
///// 2 PAY FEES /////
|
||||
await Promise.all([
|
||||
// pool 0 - split into two payments
|
||||
stakingWrapper.payProtocolFeeAsync(makers[0], protocolFeesByMaker[0].div(2), exchange),
|
||||
stakingWrapper.payProtocolFeeAsync(makers[0], protocolFeesByMaker[0].div(2), exchange),
|
||||
// pool 1 - pay full amounts
|
||||
stakingWrapper.payProtocolFeeAsync(makers[1], protocolFeesByMaker[1], exchange),
|
||||
stakingWrapper.payProtocolFeeAsync(makers[2], protocolFeesByMaker[2], exchange),
|
||||
// pool 2 -- pay full amounts
|
||||
stakingWrapper.payProtocolFeeAsync(makers[3], protocolFeesByMaker[3], exchange),
|
||||
stakingWrapper.payProtocolFeeAsync(makers[4], protocolFeesByMaker[4], exchange),
|
||||
// maker 5 doesn't pay anything
|
||||
]);
|
||||
///// 3 VALIDATE FEES RECORDED FOR EACH POOL /////
|
||||
const recordedProtocolFeesByPool = await Promise.all([
|
||||
stakingWrapper.getProtocolFeesThisEpochByPoolAsync(poolIds[0]),
|
||||
stakingWrapper.getProtocolFeesThisEpochByPoolAsync(poolIds[1]),
|
||||
stakingWrapper.getProtocolFeesThisEpochByPoolAsync(poolIds[2]),
|
||||
]);
|
||||
expect(recordedProtocolFeesByPool[0]).to.be.bignumber.equal(protocolFeesByMaker[0]);
|
||||
expect(recordedProtocolFeesByPool[1]).to.be.bignumber.equal(protocolFeesByMaker[1].plus(protocolFeesByMaker[2]));
|
||||
expect(recordedProtocolFeesByPool[2]).to.be.bignumber.equal(protocolFeesByMaker[3].plus(protocolFeesByMaker[4]));
|
||||
///// 4 VALIDATE TOTAL FEES /////
|
||||
const recordedTotalProtocolFees = await stakingWrapper.getTotalProtocolFeesThisEpochAsync();
|
||||
const totalProtocolFeesAsNumber = _.sumBy(protocolFeesByMaker, (value: BigNumber) => {return value.toNumber()});
|
||||
const totalProtocolFees = new BigNumber(totalProtocolFeesAsNumber);
|
||||
expect(recordedTotalProtocolFees).to.be.bignumber.equal(totalProtocolFees);
|
||||
///// 5 STAKE /////
|
||||
const stakeByPoolOperator = [
|
||||
stakingWrapper.toBaseUnitAmount(42),
|
||||
stakingWrapper.toBaseUnitAmount(84),
|
||||
stakingWrapper.toBaseUnitAmount(97),
|
||||
];
|
||||
const totalStakeByPoolOperator = stakeByPoolOperator[0].plus(stakeByPoolOperator[1]).plus(stakeByPoolOperator[2]);
|
||||
await Promise.all([
|
||||
// pool 0
|
||||
stakingWrapper.depositAndStakeAsync(poolOperators[0], stakeByPoolOperator[0]),
|
||||
// pool 1
|
||||
stakingWrapper.depositAndStakeAsync(poolOperators[1], stakeByPoolOperator[1]),
|
||||
// pool 2
|
||||
stakingWrapper.depositAndStakeAsync(poolOperators[2], stakeByPoolOperator[2]),
|
||||
]);
|
||||
|
||||
///// 6 Add some delegators to pool 2 /////
|
||||
const delegators = stakers.slice(0, 3);
|
||||
const stakeByDelegator = [
|
||||
stakingWrapper.toBaseUnitAmount(17),
|
||||
stakingWrapper.toBaseUnitAmount(75),
|
||||
stakingWrapper.toBaseUnitAmount(90),
|
||||
];
|
||||
const totalStakeByDelegators = stakeByDelegator[0].plus(stakeByDelegator[1]).plus(stakeByDelegator[2]);
|
||||
await Promise.all([
|
||||
stakingWrapper.depositAndDelegateAsync(delegators[0], poolIds[2], stakeByDelegator[0]),
|
||||
stakingWrapper.depositAndDelegateAsync(delegators[1], poolIds[2], stakeByDelegator[1]),
|
||||
stakingWrapper.depositAndDelegateAsync(delegators[2], poolIds[2], stakeByDelegator[2]),
|
||||
]);
|
||||
|
||||
///// 6 FINALIZE /////
|
||||
await stakingWrapper.skipToNextEpochAsync();
|
||||
|
||||
///// 7 CHECK PROFITS /////
|
||||
// the expected payouts were computed by hand
|
||||
// @TODO - get computations more accurate
|
||||
/*
|
||||
Pool | Total Fees | Total Stake | Total Delegated Stake | Total Stake (Scaled)
|
||||
0 | 0.304958 | 42 | 0 | 42
|
||||
1 | 15.323258 | 84 | 0 | 84
|
||||
3 | 28.12222236 | 97 | 182 | 260.8
|
||||
...
|
||||
Cumulative Fees = 43.75043836
|
||||
Cumulative Stake = 405
|
||||
Total Rewards = 43.75043836
|
||||
*/
|
||||
|
||||
const expectedPayoutByPoolOperator = [
|
||||
new BigNumber('2.89303'), // 2.8930364057678784829875695710382241749912199174798475
|
||||
new BigNumber('9.90218'), // 9.9021783083174087034787071054543342142019746753770943
|
||||
new BigNumber('28.16463'), // 28.164631904035798614670299155719067954180760345463798
|
||||
];
|
||||
const payoutByPoolOperator = await Promise.all([
|
||||
stakingWrapper.rewardVaultGetBalanceAsync(poolIds[0]),
|
||||
stakingWrapper.rewardVaultGetBalanceAsync(poolIds[1]),
|
||||
stakingWrapper.rewardVaultGetBalanceAsync(poolIds[2]),
|
||||
]);
|
||||
const payoutAcurateToFiveDecimalsByPoolOperator = await Promise.all([
|
||||
stakingWrapper.trimFloat(stakingWrapper.toFloatingPoint(payoutByPoolOperator[0], 18), 5),
|
||||
stakingWrapper.trimFloat(stakingWrapper.toFloatingPoint(payoutByPoolOperator[1], 18), 5),
|
||||
stakingWrapper.trimFloat(stakingWrapper.toFloatingPoint(payoutByPoolOperator[2], 18), 5),
|
||||
]);
|
||||
expect(payoutAcurateToFiveDecimalsByPoolOperator[0]).to.be.bignumber.equal(expectedPayoutByPoolOperator[0]);
|
||||
expect(payoutAcurateToFiveDecimalsByPoolOperator[1]).to.be.bignumber.equal(expectedPayoutByPoolOperator[1]);
|
||||
expect(payoutAcurateToFiveDecimalsByPoolOperator[2]).to.be.bignumber.equal(expectedPayoutByPoolOperator[2]);
|
||||
|
||||
///// 8 CHECK PROFITS VIA STAKING CONTRACT /////
|
||||
///// 9 WITHDRAW PROFITS VIA STAKING CONTRACT /////
|
||||
|
||||
/////
|
||||
});
|
||||
});
|
||||
});
|
||||
|
@@ -475,7 +475,7 @@ export class StakingWrapper {
|
||||
const output = await this.getLibMathTestContract().nthRootFixedPoint.callAsync(value, n);
|
||||
return output;
|
||||
}
|
||||
public async cobbDouglas(
|
||||
public async cobbDouglasAsync(
|
||||
totalRewards: BigNumber,
|
||||
ownerFees: BigNumber,
|
||||
totalFees: BigNumber,
|
||||
@@ -495,7 +495,7 @@ export class StakingWrapper {
|
||||
);
|
||||
return output;
|
||||
}
|
||||
public async cobbDouglasSimplified(
|
||||
public async cobbDouglasSimplifiedAsync(
|
||||
totalRewards: BigNumber,
|
||||
ownerFees: BigNumber,
|
||||
totalFees: BigNumber,
|
||||
@@ -522,7 +522,7 @@ export class StakingWrapper {
|
||||
);
|
||||
return output;
|
||||
}
|
||||
public async cobbDouglasSimplifiedInverse(
|
||||
public async cobbDouglasSimplifiedInverseAsync(
|
||||
totalRewards: BigNumber,
|
||||
ownerFees: BigNumber,
|
||||
totalFees: BigNumber,
|
||||
|
Reference in New Issue
Block a user