-
Notifications
You must be signed in to change notification settings - Fork 35
/
validators.py
126 lines (102 loc) · 4.64 KB
/
validators.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
"""
# Validator Mechanisms
Validator processes such as validator activation, staking, and uptime.
"""
import typing
import model.constants as constants
import model.parts.utils.ethereum_spec as spec
from model.parts.utils import get_number_of_awake_validators
from model.types import ETH, Gwei
def policy_staking(
params, substep, state_history, previous_state
) -> typing.Dict[str, ETH]:
"""
## Staking Policy
A policy used when driving the model with the `eth_staked_process`,
for generating phase-space analyses, e.g. simulating a set of discrete `eth_staked` values.
When the `eth_staked_process` is disabled, the model is driven using the `validator_process`,
for generating state-space analyses.
"""
# Parameters
dt = params["dt"]
eth_staked_process = params["eth_staked_process"]
# State Variables
run = previous_state["run"]
timestep = previous_state["timestep"]
eth_supply = previous_state["eth_supply"]
number_of_validators = previous_state["number_of_active_validators"]
average_effective_balance = previous_state["average_effective_balance"]
# If the eth_staked_process is defined
if eth_staked_process(0, 0) is not None:
# Get the ETH staked sample for the current run and timestep
eth_staked = eth_staked_process(run, timestep * dt)
# Else, calculate from the number of validators
else:
eth_staked = number_of_validators * average_effective_balance / constants.gwei
# Assert expected conditions
assert eth_staked <= eth_supply, f"ETH staked can't be more than ETH supply"
return {"eth_staked": eth_staked}
def policy_validators(params, substep, state_history, previous_state):
"""
## Validator Policy Function
Calculate the number of validators driven by the ETH staked or validator processes.
"""
# Parameters
dt = params["dt"]
eth_staked_process = params["eth_staked_process"]
validator_process = params["validator_process"]
validator_uptime_process = params["validator_uptime_process"]
# State Variables
run = previous_state["run"]
timestep = previous_state["timestep"]
number_of_active_validators = previous_state["number_of_active_validators"]
number_of_validators_in_activation_queue = previous_state[
"number_of_validators_in_activation_queue"
]
average_effective_balance = previous_state["average_effective_balance"]
# Calculate the number of validators using ETH staked
if eth_staked_process(0, 0) is not None:
eth_staked = eth_staked_process(run, timestep * dt)
number_of_active_validators = int(
round(eth_staked / (average_effective_balance / constants.gwei))
)
else:
new_validators_per_epoch = validator_process(run, timestep * dt)
number_of_validators_in_activation_queue += new_validators_per_epoch * dt
validator_churn_limit = (
spec.get_validator_churn_limit(params, previous_state) * dt
)
activated_validators = min(
number_of_validators_in_activation_queue, validator_churn_limit
)
number_of_active_validators += activated_validators
number_of_validators_in_activation_queue -= activated_validators
# Calculate the number of "awake" validators
# See proposal: https://ethresear.ch/t/simplified-active-validator-cap-and-rotation-proposal
number_of_awake_validators = spec.get_awake_validator_indices(
params, previous_state
)
# Calculate the validator uptime
validator_uptime = validator_uptime_process(run, timestep * dt)
# Assume a participation of more than 2/3 due to lack of inactivity leak mechanism
assert validator_uptime >= 2 / 3, "Validator uptime must be greater than 2/3"
return {
"number_of_validators_in_activation_queue": number_of_validators_in_activation_queue,
"number_of_active_validators": number_of_active_validators,
"number_of_awake_validators": number_of_awake_validators,
"validator_uptime": validator_uptime,
}
def policy_average_effective_balance(
params, substep, state_history, previous_state
) -> typing.Dict[str, Gwei]:
"""
## Average Effective Balance Policy Function
Calculate the validator average effective balance.
"""
# State Variables
number_of_validators = get_number_of_awake_validators(params, previous_state)
# Get total active balance
total_active_balance = spec.get_total_active_balance(params, previous_state)
# Aggregate by averaging over all validators
average_effective_balance = total_active_balance / number_of_validators
return {"average_effective_balance": average_effective_balance}