The SFC (Special Fee Contract) maintains a list of validation stakers and delegators. It distributes the rewards, based on information written into contract by Lachesis node.
The essential:
- SFC contract only maintains the list of delegators and stakers, and doesn’t calculate reward weights or validators groups. The reward weights and validators group are written into SFC contract storage by consensus (i.e. golang code) for performance reasons. SFC calculates the rewards, depending on epoch snapshots written by consensus.
- Calculation of reward is O(1) for each epoch. Staker/delegator may specify a range of epochs he claims reward for, but not lower than
prev reward epoch + 1
and not higher thanlast sealed epoch snapshot
. - Each delegation is a separate object. Delegator claims rewards for each delegation independently of other delegations and stakers. Staker has a counter of delegated tokens to it.
- Staker/delegator cannot withdraw his stake any time. He must deactivate the stake (prepare to withdraw) in advance of N days.
- If validator is confirmed to be a cheater, then only a flag is written into SFC storage by consensus. SFC checks this flag and doesn't withdraw stake/delegation during
withdraw
operation if flag is set. - Validation staker is allowed to increase his stake, but not decrease (only allowed to withdraw the whole stake).
- Install solc 0.5.12
solc -o $PWD/build --optimize --optimize-runs=2000 --abi --bin-runtime --allow-paths $PWD/contracts --overwrite $PWD/contracts/sfc/Staker.sol
# compilation. don't forget --bin-runtime flag, if contract is pre-deployed!cat build/TestStakers.bin-runtime
# paste this into GetTestContractBinV1 in go-lachesis repocat build/TestStakers.abi
# use this ABI to call contract methods
- Install nodejs 10.5.0
npm install -g [email protected]
# install truffle v5.1.4npm update
npm test
If everything is allright, it should output something along this:
> [email protected] test fantom-sfc
> truffle test
Using network 'test'.
Compiling your contracts...
===========================
> Compiling ./contracts/ownership/Ownable.sol
> Compiling ./contracts/sfc/Migrations.sol
> Compiling ./contracts/sfc/SafeMath.sol
> Compiling ./contracts/sfc/Staker.sol
> Compiling ./contracts/sfc/StakerConstants.sol
> Compiling ./contracts/test/LegacyStaker.sol
> Compiling ./contracts/test/UnitTestStakers.sol
> Compiling ./contracts/upgradeability/Address.sol
> Compiling ./contracts/upgradeability/BaseUpgradeabilityProxy.sol
> Compiling ./contracts/upgradeability/Proxy.sol
> Compiling ./contracts/upgradeability/UpgradeabilityProxy.sol
> Compiling ./contracts/version/Version.sol
Contract: SFC
Locking stake tests
✓ should start "locked stake" feature (736ms)
✓ should calc raw ValidatorEpochReward correctly after locked up started (1076ms)
✓ should lock stake (1208ms)
✓ should lock stake with right duration (513ms)
✓ should not call prepareToWithdrawStake, until locked time is pass (455ms)
✓ should not call prepareToWithdrawStakePartial, until locked time is pass (477ms)
✓ should lock delegation (1427ms)
✓ should lock delegation with right duration (758ms)
✓ should subtract penalty if prepareToWithdrawDelegation will call earlier than locked time is pass (842ms)
✓ should subtract penalty if prepareToWithdrawDelegationPartial will call earlier than locked time is pass (886ms)
Contract: SFC
Methods tests
✓ checking Staker parameters (165ms)
✓ checking createStake function (525ms)
✓ checking increaseStake function (249ms)
✓ checking createDelegation function (436ms)
✓ checking calcRawValidatorEpochReward function (363ms)
✓ checking epoch snapshot logic (163ms)
✓ checking calcValidatorEpochReward function (453ms)
✓ checking calcDelegationEpochReward function (418ms)
✓ checking claimDelegationRewards function (647ms)
✓ checking bonded ratio (107ms)
✓ checking claimValidatorRewards function (388ms)
✓ checking prepareToWithdrawStake function (198ms)
✓ checking withdrawStake function (922ms)
✓ checking prepareToWithdrawDelegation function (330ms)
✓ checking withdrawDelegation function (1497ms)
25 passing (18s)