Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Daily rewards for active nodes #1064

Merged
merged 26 commits into from
Dec 2, 2024
Merged
Changes from all commits
Commits
Show all changes
26 commits
Select commit Hold shift + click to select a range
1b73942
Initial draft for daily rewards for active nodes
Neeharika-Sompalli Oct 17, 2024
a3ee264
update HIP name
Neeharika-Sompalli Oct 17, 2024
cf69045
update HIP number
Neeharika-Sompalli Oct 17, 2024
bdd4e70
update HIP number
Neeharika-Sompalli Oct 17, 2024
b20a89f
update HIP number
Neeharika-Sompalli Oct 17, 2024
84da99d
update created date
Neeharika-Sompalli Oct 17, 2024
d44642a
update as per review comments
Neeharika-Sompalli Oct 17, 2024
7fc7325
Change to needs-tsc-approval
Neeharika-Sompalli Oct 17, 2024
81ccfe3
Update HIP/hip-1064.md
Neeharika-Sompalli Oct 18, 2024
e25468f
Update HIP/hip-1064.md
Neeharika-Sompalli Oct 18, 2024
082eb24
Update HIP/hip-1064.md
Neeharika-Sompalli Oct 18, 2024
bc391f4
Update HIP/hip-1064.md
Neeharika-Sompalli Oct 18, 2024
e1d5d6f
Update HIP/hip-1064.md
Neeharika-Sompalli Oct 18, 2024
aa5aea8
Update HIP/hip-1064.md
Neeharika-Sompalli Oct 18, 2024
e3ea9db
Update HIP/hip-1064.md
Neeharika-Sompalli Oct 18, 2024
11217cd
Update HIP/hip-1064.md
Neeharika-Sompalli Oct 18, 2024
9801f0c
Update HIP/hip-1064.md
Neeharika-Sompalli Oct 18, 2024
938e20e
Update HIP/hip-1064.md
Neeharika-Sompalli Oct 18, 2024
ec10c9f
review comments
Neeharika-Sompalli Oct 18, 2024
33f9478
review comments
Neeharika-Sompalli Oct 18, 2024
f845541
Update definition as per Leemon's suggestion
Neeharika-Sompalli Nov 7, 2024
72146fd
fix review comments
Neeharika-Sompalli Nov 7, 2024
1132190
fix review comments
Neeharika-Sompalli Nov 13, 2024
c52d0bb
fix review comments
Neeharika-Sompalli Nov 13, 2024
34fb4a6
Update hip-1064.md to review
mgarbs Nov 27, 2024
137f664
Update hip-1064.md to last call
mgarbs Nov 27, 2024
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
225 changes: 225 additions & 0 deletions HIP/hip-1064.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,225 @@
---
hip: 1064
title: Daily Rewards For Active Nodes
author: Neeharika Sompalli <@Neeharika-Sompalli>
working-group: Leemon Baird (@lbaird), Richard Bair <@rbair23>, Michael Tinker <@tinker-michaelj>
requested-by: Hashgraph
type: Standards Track
category: Core
needs-tsc-approval: Yes
status: Last Call
last-call-date-time: 2024-12-11T07:00:00Z
created: 2024-10-17
discussions-to: https://github.com/hashgraph/hedera-improvement-proposal/discussions/1063
updated: 2024-11-27
---

## Abstract

This HIP proposes a reward mechanism that will incentivize nodes to remain active on the network. The definition of
active node is described below. The minimum reward per day for nodes will be a configurable property decided by the
governance of the deployed network. This minimum reward may be adjusted based on fees already paid throughout the day.

### Terminology
jsync-swirlds marked this conversation as resolved.
Show resolved Hide resolved

<dl>
<dt>Judge</dt>
<dd>An event that wins the election to be made a judge. It must be a witness, and it will have tended to have been
gossiped to most of the other nodes quickly (otherwise it would have lost the election). An event reaches consensus
when it is an ancestor of all judges in a given round. The first round where that happens is its consensus round.
It’s a math theorem that every round is guaranteed to have at least one judge, and a math conjecture that every round
is guaranteed to have judges created by a supermajority of nodes (>2/3 of weight).</dd>
Neeharika-Sompalli marked this conversation as resolved.
Show resolved Hide resolved
<dt>Active Node</dt>
<dd>A node is an *Active Node* if it creates a "judge" in a significant fraction of rounds during a staking period,
for example 80%. Any node that is trustworthy and submits events will create judges. If the node doesn’t submit events
or when other nodes don’t build on this node’s events, the node will not create judges.</dd>
<dt>Staking Period</dt>
<dd>Staking rewards are currently recalculated once per day, though this duration is a setting called the
"staking period". Node rewards will be transferred on the same schedule. Everywhere that this document mentions
a "day", it should be interpreted as the staking period. And every time calculations about a fraction of rounds
are done, they are the average over a single staking period.</dd>
</dl>

## Motivation

The goal is to give a strong incentive for node operators to keep them active and properly configured throughout every
staking period.

## Rationale

Clearly defining the criteria for node activity, we can ensure that rewards are distributed fairly, incentivizing nodes
to remain active and contribute to the network’s stability and growth.

## User Stories

The implementation will be tested under various scenarios, including:

- As a node operator, I expect to receive at least a minimum node reward when the node is active for at least
`nodes.activeRoundsPercent` rounds
- As a node operator, I do not expect to receive any node rewards when the node is not active for at least
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I understand this is following the usual pattern for use cases, but I find it weird nevertheless. How about something like:

"As a user of the network, I expect that non-active nodes do not receive any node rewards..."

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

As a user of the network

Do you mean the node operator Michael?

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

No, I meant any regular user. It sounds weird for me to have a use case saying that somebody does not expect a certain behavior. However, as a regular user, I expect that the money I spend is only used for active nodes.

It's not important, though. It is clear what the expected behavior is.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It might be worthwhile to have both.
The current node operator use case, and another with text very similar to what @netopyr suggested.
It's good to be clear about expectations for both node operator and network user.

Copy link
Member Author

@Neeharika-Sompalli Neeharika-Sompalli Nov 13, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Ah makes sense. Thanks @netopyr . Added another use case

`nodes.activeRoundsPercent` rounds
- As a network operator, when `nodes.adjustNodeFees` is set to true, I expect each node reward to be reduced by the
average node fees already collected during that period.
- As a network operator, when `nodes.adjustNodeFees` is set to false, I do not expect each node reward to be reduced
by the average node fees already collected during that period.
- As a network operator, I expect each active node to receive at least a minimum reward defined by `nodes.minNodeReward`
- As a user of the network, I expect that non-active nodes do not receive any rewards

## Specification

There will be three configurable properties added.

- `nodes.minNodeReward` : A minimum daily node reward amount.
Each node SHALL receive at least this many tinybar each day that the node is "Active".
- `nodes.adjustNodeFees` : A flag to adjust node fees.
If this is set, the network SHALL reduce the rewards given for a staking period by the average node fees already
collected during that period.
- `nodes.activeRoundsPercent` : A percentage value relating to active nodes.
A node MUST create a judge in at least this percentage of rounds to be considered active.
This is set to 80% by default.

The reward SHALL be adjusted based on average fees paid for all nodes when `staking.fees.adjustNodeFees` is true.
Each node is paid the constant reward minus the average node fees received that day.
This reduction is averaged across all nodes; the network SHALL NOT subtract the node fees earned by one node in
particular.

The node rewards SHALL be distributed at the end of a staking period using synthetic `CryptoTransfer` transactions from
`ledger.fundingAccount` to each node account. Each synthetic `CryptoTransfer` only allows `ledger.transfers.maxLen`
transfers per transaction; it is possible, therefore, that multiple synthetic `CryptoTransfer` transactions will be
necessary to distribute rewards, based on the total number of node accounts in the network.

This HIP modifies the following state protobufs.

- Currently, PlatformState’s ConsensusSnapshot contains information about the `judge_hashes`. But, it doesn’t provide
the creator ids. We will deprecate the existing `judge_hashes` and add a list of `judges`. Each `Judge` will include
the creator node id and the judge hash for that node. Any node that didn’t create a judge in a round will not be
present in the list.

```
/**
* A consensus snapshot.<br/>
* This is a snapshot of the consensus state for a particular round.
*
* This message SHALL record consensus data necessary for restart
* and reconnect.
*/
message ConsensusSnapshot {
....
/**
* A list of SHA-384 hash values.<br/>
* The hashes of all judges for this round.
* <p>
* This list SHALL be ordered by creator ID.<br/>
* This list MUST be deterministically ordered.
*/
repeated bytes judge_hashes = 2 [deprecated=true];
....
/*
* A list of judges.
* This list SHALL be ordered by creator ID.<br/>
* This list MUST be deterministically ordered.
*/
repeated Judge judges = 6;
}

/**
* A judge information
*/
message Judge {
/**
* The creator node ID who created this judge.
*/
uint64 creator_id = 1;

/**
* SHA-384 hash value of this judge
*/
bytes judge_hash = 2;
}
```

- To store the number of rounds in a staking period and the number of rounds in which each node created judges, a new
singleton state `NodeRewards` will be added to `TokenService`. This singleton state will be updated at the end of
every round.

```
/**
* A record of node "active" status.<br/>
* This is used to record the number of "active" nodes in a staking
* period based on number of judges each node created in that period.
*
* A Node SHALL be considered "active" if it produced "judges" according
* to the consensus algorithm in a percentage of rounds, during the
* staking period, greater than the network configuration value for
* `nodes.activeRoundsPercent`.
*/
message NodeRewards {
/**
* A number of rounds so far, in this staking period.
*/
uint64 num_rounds_in_staking_period = 1;

/**
* A list of node activities.<br/>
* This records the number of rounds when each node created
* judges for the consensus algorithm.
* <p>
* This list SHALL contain one entry for each node participating
* in consensus during this staking period.
*/
repeated NodeActivity node_activities = 2;
}
/**
* A record of judge rounds for a single node.<br/>
* This records, for a single node, the number of rounds so far, during this staking
* period that contained at least one judge created by that node.
*
* This message SHALL NOT record the total number of rounds in a staking
* period.<br/>
* This message SHALL record a count of rounds for a single node.<br/>
* This message SHALL contain the number of rounds that contain at
* least one judge from the node identified.
*/
message NodeActivity {
/**
* A node identifier.
*/
uint64 node_id = 1;

/**
* A count of rounds.<br/>
* This is the count of rounds so far, in this staking period in which the node identified
* by `node_id` created judges.
*/
uint64 num_judge_rounds = 2;
}
```

## Backward Compatibility

Existing nodes will not require any changes to their software or configuration to be eligible for rewards, as long as
they meet the active criteria.

## Security Implications

No known security concerns.

## How to Teach This

This HIP incentivizes node operators for keeping the node active by guaranteeing a minimum reward in the periods that
they accomplish this.

## Rejected Ideas

- We considered using the number of transactions submitted by a node to assess node activity.
- This was rejected in favor of counting rounds with judges because the latter
approach incorporates consensus network assessment of node "honesty" and
is more "fair".

## Reference Implementation

## Copyright/license

This document is licensed under the Apache License, Version 2.0 --
see[LICENSE](https://github.com/Neeharika-Sompalli/hedera-improvement-proposal/blob/1cad867e75071dad048ec633e04e208ca242c0df/LICENSE)
or (https://www.apache.org/licenses/LICENSE-2.0)
Loading