Skip to content

Commit 9963d4a

Browse files
authored
feat: reimplement validator rewarder (#69)
This PR reimplements validator rewarder contract according to a fixed token increments. Please checkout the discussion here for more details about the current implementation: recallnet/ipc#151 (comment) With these changes there won't be any "state" in the rewarder contract. The overall architecture remains the same. That is the validator rewarder is being called by the IPC when there is a new claim for rewards. We assume that IPC guarantees that there won't be any "double claiming". IPC notifies the validator rewarder with the checkpoint height, validator's address and the number of blocks that _this_ validator has committed in the checkpoint. Each checkpoint contains 600 blocks. Which means 200 new tokens (1 token per 3 blocks) are minted in each checkpoint. We just calculate the validator's share of these 200 new tokens i.e. the ratio of blocks that they have committed to total blocks in the checkpoint. We mint this amount to validator's address. --------- Signed-off-by: avichalp <[email protected]>
1 parent 4f7eda5 commit 9963d4a

6 files changed

+151
-317
lines changed

foundry.toml

+1
Original file line numberDiff line numberDiff line change
@@ -2,6 +2,7 @@
22
src = "src"
33
out = "out"
44
libs = ["lib"]
5+
fs_permissions = [{ access = "read", path = "./out" }]
56
test = "test"
67
build_info = true
78
extra_output = ["storageLayout"]

src/token/ValidatorRewarder.sol

+38-75
Original file line numberDiff line numberDiff line change
@@ -32,28 +32,22 @@ contract ValidatorRewarder is IValidatorRewarder, UUPSUpgradeable, OwnableUpgrad
3232
/// @notice The token that this rewarder mints
3333
Recall public token;
3434

35-
/// @notice The latest checkpoint height that rewards can be claimed for
36-
/// @dev Using uint64 to match Filecoin's epoch height type and save gas when interacting with the network
37-
uint64 public latestClaimedCheckpoint;
38-
3935
/// @notice The bottomup checkpoint period for the subnet.
4036
/// @dev The checkpoint period is set when the subnet is created.
4137
uint256 public checkpointPeriod;
4238

43-
/// @notice The supply of RECALL tokens at each checkpoint
44-
mapping(uint64 checkpointHeight => uint256 totalSupply) public checkpointToSupply;
45-
46-
/// @notice The inflation rate for the subnet
47-
/// @dev The rate is expressed as a decimal*1e18.
48-
/// @dev For example 5% APY is 0.0000928276004952% yield per checkpoint period.
49-
/// @dev This is expressed as 928_276_004_952 or 0.000000928276004952*1e18.
50-
uint256 public constant INFLATION_RATE = 928_276_004_952;
39+
/// @notice The number of blocks required to generate 1 new token (with 18 decimals)
40+
uint256 public constant BLOCKS_PER_TOKEN = 3;
5141

5242
// ========== EVENTS & ERRORS ==========
5343

5444
event ActiveStateChange(bool active, address account);
5545
event SubnetUpdated(SubnetID subnet, uint256 checkpointPeriod);
56-
event CheckpointClaimed(uint64 indexed checkpointHeight, address indexed validator, uint256 amount);
46+
/// @notice Emitted when a validator claims their rewards for a checkpoint
47+
/// @param checkpointHeight The height of the checkpoint for which rewards are claimed
48+
/// @param validator The address of the validator claiming rewards
49+
/// @param amount The amount of tokens claimed as reward
50+
event RewardsClaimed(uint64 indexed checkpointHeight, address indexed validator, uint256 amount);
5751

5852
error SubnetMismatch(SubnetID id);
5953
error InvalidClaimNotifier(address notifier);
@@ -136,83 +130,52 @@ contract ValidatorRewarder is IValidatorRewarder, UUPSUpgradeable, OwnableUpgrad
136130
revert InvalidClaimNotifier(msg.sender);
137131
}
138132

139-
// When the supply for the checkpoint is 0, it means that this is the first claim
140-
// for this checkpoint.
141-
// In this case we will set the supply for the checkpoint and
142-
// calculate the inflation and mint the rewards to the rewarder and the first claimant.
143-
// Otherwise, we know the supply for the checkpoint.
144-
// We will calculate the rewards and transfer them to the other claimants for this checkpoint.
145-
uint256 supplyAtCheckpoint = checkpointToSupply[claimedCheckpointHeight];
146-
if (supplyAtCheckpoint == 0) {
147-
// Check that the checkpoint height is valid.
148-
if (!validateCheckpointHeight(claimedCheckpointHeight)) {
149-
revert InvalidCheckpointHeight(claimedCheckpointHeight);
150-
}
151-
152-
// Get the current supply of RECALL tokens
153-
uint256 currentSupply = token.totalSupply();
154-
155-
// Set the supply for the checkpoint and update latest claimed checkpoint
156-
checkpointToSupply[claimedCheckpointHeight] = currentSupply;
157-
latestClaimedCheckpoint = claimedCheckpointHeight;
158-
159-
// Calculate rewards
160-
uint256 supplyDelta = calculateInflationForCheckpoint(currentSupply);
161-
uint256 validatorShare = calculateValidatorShare(data.blocksCommitted, supplyDelta);
162-
163-
// Perform external interactions after state updates
164-
token.mint(address(this), supplyDelta - validatorShare);
165-
token.mint(data.validator, validatorShare);
166-
emit CheckpointClaimed(claimedCheckpointHeight, data.validator, validatorShare);
167-
} else {
168-
// Calculate the supply delta for the checkpoint
169-
uint256 supplyDelta = calculateInflationForCheckpoint(supplyAtCheckpoint);
170-
// Calculate the validator's share of the supply delta
171-
uint256 validatorShare = calculateValidatorShare(data.blocksCommitted, supplyDelta);
172-
// Transfer the validator's share of the supply delta to the validator
173-
token.safeTransfer(data.validator, validatorShare);
174-
emit CheckpointClaimed(claimedCheckpointHeight, data.validator, validatorShare);
133+
// Check that the checkpoint height is valid
134+
if (!validateCheckpointHeight(claimedCheckpointHeight)) {
135+
revert InvalidCheckpointHeight(claimedCheckpointHeight);
175136
}
137+
138+
// Calculate rewards for this checkpoint
139+
uint256 newTokens = calculateNewTokensForCheckpoint();
140+
uint256 validatorShare = calculateValidatorShare(data.blocksCommitted, newTokens);
141+
142+
// Mint the validator's share
143+
token.mint(data.validator, validatorShare);
144+
emit RewardsClaimed(claimedCheckpointHeight, data.validator, validatorShare);
176145
}
177146

178147
// ========== INTERNAL FUNCTIONS ==========
179148

180-
/// @notice The internal method to calculate the supply delta for a checkpoint
181-
/// @param supply The token supply at the checkpoint
182-
/// @return The supply delta, i.e. the amount of new tokens minted for the checkpoint
183-
function calculateInflationForCheckpoint(uint256 supply) internal pure returns (uint256) {
184-
UD60x18 supplyFixed = ud(supply);
185-
UD60x18 inflationRateFixed = ud(INFLATION_RATE);
186-
UD60x18 result = supplyFixed.mul(inflationRateFixed);
187-
return result.unwrap();
149+
/// @notice Calculates the total number of new tokens to be minted for a checkpoint
150+
/// @return The number of new tokens to be minted (in base units with 18 decimals)
151+
function calculateNewTokensForCheckpoint() internal view returns (uint256) {
152+
UD60x18 blocksPerToken = ud(BLOCKS_PER_TOKEN);
153+
UD60x18 period = ud(checkpointPeriod);
154+
UD60x18 oneToken = ud(1 ether);
155+
156+
// Calculate (checkpointPeriod * 1 ether) / BLOCKS_PER_TOKEN using fixed-point math
157+
return period.mul(oneToken).div(blocksPerToken).unwrap();
188158
}
189159

190-
/// @notice The internal method to calculate the validator's share of the supply delta
160+
/// @notice The internal method to calculate the validator's share of the new tokens
191161
/// @param blocksCommitted The number of blocks committed by the validator
192-
/// @param supplyDelta The supply delta, i.e. the amount of new tokens minted for the checkpoint
193-
/// @return The validator's share of the supply delta
194-
function calculateValidatorShare(uint256 blocksCommitted, uint256 supplyDelta) internal view returns (uint256) {
195-
UD60x18 blocksFixed = ud(blocksCommitted);
196-
UD60x18 deltaFixed = ud(supplyDelta);
197-
UD60x18 periodFixed = ud(checkpointPeriod);
198-
UD60x18 share = blocksFixed.div(periodFixed);
199-
UD60x18 result = share.mul(deltaFixed);
162+
/// @param totalNewTokens The total number of new tokens for the checkpoint
163+
/// @return The validator's share of the new tokens
164+
function calculateValidatorShare(uint256 blocksCommitted, uint256 totalNewTokens) internal view returns (uint256) {
165+
UD60x18 blocks = ud(blocksCommitted);
166+
UD60x18 tokens = ud(totalNewTokens);
167+
UD60x18 period = ud(checkpointPeriod);
168+
UD60x18 share = blocks.div(period);
169+
UD60x18 result = share.mul(tokens);
200170
return result.unwrap();
201171
}
202172

203173
/// @notice Validates that the claimed checkpoint height is valid
204174
/// @param claimedCheckpointHeight The height of the checkpoint that the validator is claiming for
205175
/// @return True if the checkpoint height is valid, false otherwise
206-
/// @dev When the latest claimable checkpoint is not set (0), it means that _this_ is the first ever claim.
207-
/// @dev In this case, we need to ensure the first claim is at the first checkpoint period.
208-
/// @dev Otherwise, we must ensure that the claimed checkpoint is the next expected checkpoint.
176+
/// @dev Ensures the checkpoint height is a multiple of the checkpoint period
209177
function validateCheckpointHeight(uint64 claimedCheckpointHeight) internal view returns (bool) {
210-
if (latestClaimedCheckpoint == 0) {
211-
// First claim must be at the first checkpoint period
212-
return claimedCheckpointHeight == checkpointPeriod;
213-
}
214-
// Subsequent claims must be at the next checkpoint
215-
return claimedCheckpointHeight == latestClaimedCheckpoint + checkpointPeriod;
178+
return claimedCheckpointHeight > 0 && claimedCheckpointHeight % checkpointPeriod == 0;
216179
}
217180

218181
/// @dev Function that should revert when `msg.sender` is not authorized to upgrade the contract

test/ValidatorGater.t.sol

+2-1
Original file line numberDiff line numberDiff line change
@@ -83,9 +83,10 @@ contract ValidatorGaterTest is Test {
8383
assertFalse(gater.isAllow(validator1, 101)); // Above range
8484
}
8585

86-
function testFailUnauthorizedApprove() public {
86+
function testRevertWhenUnauthorizedApprove() public {
8787
// Non-owner should not be able to approve a validator
8888
vm.prank(validator2); // validator2 is not the owner
89+
vm.expectRevert(abi.encodeWithSignature("OwnableUnauthorizedAccount(address)", validator2));
8990
gater.approve(validator1, 10, 100);
9091
}
9192

0 commit comments

Comments
 (0)