@lodestar/config 1.35.0-dev.e9dd48f165 → 1.35.0-dev.f45a2be721

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (78) hide show
  1. package/README.md +4 -4
  2. package/lib/beaconConfig.d.ts.map +1 -0
  3. package/lib/chainConfig/configs/mainnet.d.ts.map +1 -0
  4. package/lib/chainConfig/configs/mainnet.js +24 -1
  5. package/lib/chainConfig/configs/mainnet.js.map +1 -1
  6. package/lib/chainConfig/configs/minimal.d.ts.map +1 -0
  7. package/lib/chainConfig/configs/minimal.js +24 -1
  8. package/lib/chainConfig/configs/minimal.js.map +1 -1
  9. package/lib/chainConfig/default.d.ts.map +1 -0
  10. package/lib/chainConfig/index.d.ts +2 -2
  11. package/lib/chainConfig/index.d.ts.map +1 -0
  12. package/lib/chainConfig/index.js +7 -2
  13. package/lib/chainConfig/index.js.map +1 -1
  14. package/lib/chainConfig/json.d.ts.map +1 -0
  15. package/lib/chainConfig/networks/chiado.d.ts.map +1 -0
  16. package/lib/chainConfig/networks/chiado.js +3 -0
  17. package/lib/chainConfig/networks/chiado.js.map +1 -1
  18. package/lib/chainConfig/networks/ephemery.d.ts.map +1 -0
  19. package/lib/chainConfig/networks/ephemery.js +3 -0
  20. package/lib/chainConfig/networks/ephemery.js.map +1 -1
  21. package/lib/chainConfig/networks/gnosis.d.ts.map +1 -0
  22. package/lib/chainConfig/networks/gnosis.js +3 -0
  23. package/lib/chainConfig/networks/gnosis.js.map +1 -1
  24. package/lib/chainConfig/networks/holesky.d.ts.map +1 -0
  25. package/lib/chainConfig/networks/holesky.js +14 -2
  26. package/lib/chainConfig/networks/holesky.js.map +1 -1
  27. package/lib/chainConfig/networks/hoodi.d.ts.map +1 -0
  28. package/lib/chainConfig/networks/hoodi.js +14 -2
  29. package/lib/chainConfig/networks/hoodi.js.map +1 -1
  30. package/lib/chainConfig/networks/mainnet.d.ts.map +1 -0
  31. package/lib/chainConfig/networks/sepolia.d.ts.map +1 -0
  32. package/lib/chainConfig/networks/sepolia.js +14 -2
  33. package/lib/chainConfig/networks/sepolia.js.map +1 -1
  34. package/lib/chainConfig/types.d.ts +12 -0
  35. package/lib/chainConfig/types.d.ts.map +1 -0
  36. package/lib/chainConfig/types.js +12 -0
  37. package/lib/chainConfig/types.js.map +1 -1
  38. package/lib/configs.d.ts.map +1 -0
  39. package/lib/default.d.ts +1 -1
  40. package/lib/default.d.ts.map +1 -0
  41. package/lib/forkConfig/index.d.ts.map +1 -0
  42. package/lib/forkConfig/index.js +30 -3
  43. package/lib/forkConfig/index.js.map +1 -1
  44. package/lib/forkConfig/types.d.ts +7 -2
  45. package/lib/forkConfig/types.d.ts.map +1 -0
  46. package/lib/genesisConfig/index.d.ts.map +1 -0
  47. package/lib/genesisConfig/index.js.map +1 -1
  48. package/lib/genesisConfig/types.d.ts.map +1 -0
  49. package/lib/index.d.ts +2 -2
  50. package/lib/index.d.ts.map +1 -0
  51. package/lib/index.js +2 -2
  52. package/lib/index.js.map +1 -1
  53. package/lib/networks.d.ts.map +1 -0
  54. package/lib/utils/validateBlobSchedule.d.ts.map +1 -0
  55. package/package.json +12 -10
  56. package/src/beaconConfig.ts +31 -0
  57. package/src/chainConfig/configs/mainnet.ts +172 -0
  58. package/src/chainConfig/configs/minimal.ts +167 -0
  59. package/src/chainConfig/default.ts +19 -0
  60. package/src/chainConfig/index.ts +33 -0
  61. package/src/chainConfig/json.ts +175 -0
  62. package/src/chainConfig/networks/chiado.ts +52 -0
  63. package/src/chainConfig/networks/ephemery.ts +74 -0
  64. package/src/chainConfig/networks/gnosis.ts +77 -0
  65. package/src/chainConfig/networks/holesky.ts +67 -0
  66. package/src/chainConfig/networks/hoodi.ts +67 -0
  67. package/src/chainConfig/networks/mainnet.ts +19 -0
  68. package/src/chainConfig/networks/sepolia.ts +64 -0
  69. package/src/chainConfig/types.ts +267 -0
  70. package/src/configs.ts +4 -0
  71. package/src/default.ts +6 -0
  72. package/src/forkConfig/index.ts +236 -0
  73. package/src/forkConfig/types.ts +68 -0
  74. package/src/genesisConfig/index.ts +180 -0
  75. package/src/genesisConfig/types.ts +29 -0
  76. package/src/index.ts +4 -0
  77. package/src/networks.ts +65 -0
  78. package/src/utils/validateBlobSchedule.ts +32 -0
@@ -0,0 +1,267 @@
1
+ import {PresetName} from "@lodestar/params";
2
+
3
+ /**
4
+ * Run-time chain configuration
5
+ */
6
+ export type ChainConfig = {
7
+ PRESET_BASE: PresetName;
8
+ /**
9
+ * Free-form short name of the network that this configuration applies to - known
10
+ * canonical network names include:
11
+ * * 'mainnet' - there can be only one
12
+ * * 'holesky' - testnet
13
+ * Must match the regex: [a-z0-9\-]
14
+ */
15
+ CONFIG_NAME: string;
16
+
17
+ // Transition
18
+ TERMINAL_TOTAL_DIFFICULTY: bigint;
19
+ TERMINAL_BLOCK_HASH: Uint8Array;
20
+ TERMINAL_BLOCK_HASH_ACTIVATION_EPOCH: number;
21
+
22
+ // Genesis
23
+ MIN_GENESIS_ACTIVE_VALIDATOR_COUNT: number;
24
+ MIN_GENESIS_TIME: number;
25
+ GENESIS_FORK_VERSION: Uint8Array;
26
+ GENESIS_DELAY: number;
27
+
28
+ // Forking
29
+ // Altair
30
+ ALTAIR_FORK_VERSION: Uint8Array;
31
+ ALTAIR_FORK_EPOCH: number;
32
+ // Bellatrix
33
+ BELLATRIX_FORK_VERSION: Uint8Array;
34
+ BELLATRIX_FORK_EPOCH: number;
35
+ // Capella
36
+ CAPELLA_FORK_VERSION: Uint8Array;
37
+ CAPELLA_FORK_EPOCH: number;
38
+ // DENEB
39
+ DENEB_FORK_VERSION: Uint8Array;
40
+ DENEB_FORK_EPOCH: number;
41
+ // ELECTRA
42
+ ELECTRA_FORK_VERSION: Uint8Array;
43
+ ELECTRA_FORK_EPOCH: number;
44
+ // FULU
45
+ FULU_FORK_VERSION: Uint8Array;
46
+ FULU_FORK_EPOCH: number;
47
+ // GLOAS
48
+ GLOAS_FORK_VERSION: Uint8Array;
49
+ GLOAS_FORK_EPOCH: number;
50
+
51
+ // Time parameters
52
+ /** @deprecated Use `SLOT_DURATION_MS` instead. */
53
+ SECONDS_PER_SLOT: number;
54
+ SLOT_DURATION_MS: number;
55
+ SECONDS_PER_ETH1_BLOCK: number;
56
+ MIN_VALIDATOR_WITHDRAWABILITY_DELAY: number;
57
+ SHARD_COMMITTEE_PERIOD: number;
58
+ ETH1_FOLLOW_DISTANCE: number;
59
+ PROPOSER_REORG_CUTOFF_BPS: number;
60
+ ATTESTATION_DUE_BPS: number;
61
+ AGGREGATE_DUE_BPS: number;
62
+ // Altair
63
+ SYNC_MESSAGE_DUE_BPS: number;
64
+ CONTRIBUTION_DUE_BPS: number;
65
+
66
+ ATTESTATION_DUE_BPS_GLOAS: number;
67
+ AGGREGATE_DUE_BPS_GLOAS: number;
68
+ SYNC_MESSAGE_DUE_BPS_GLOAS: number;
69
+ CONTRIBUTION_DUE_BPS_GLOAS: number;
70
+ PAYLOAD_ATTESTATION_DUE_BPS: number;
71
+
72
+ // Validator cycle
73
+ INACTIVITY_SCORE_BIAS: number;
74
+ INACTIVITY_SCORE_RECOVERY_RATE: number;
75
+ EJECTION_BALANCE: number;
76
+ MIN_PER_EPOCH_CHURN_LIMIT: number;
77
+ MAX_PER_EPOCH_ACTIVATION_CHURN_LIMIT: number;
78
+ CHURN_LIMIT_QUOTIENT: number;
79
+ MAX_PER_EPOCH_ACTIVATION_EXIT_CHURN_LIMIT: number;
80
+ MIN_PER_EPOCH_CHURN_LIMIT_ELECTRA: number;
81
+
82
+ // Fork choice
83
+ PROPOSER_SCORE_BOOST: number;
84
+ REORG_HEAD_WEIGHT_THRESHOLD: number;
85
+ REORG_PARENT_WEIGHT_THRESHOLD: number;
86
+ REORG_MAX_EPOCHS_SINCE_FINALIZATION: number;
87
+
88
+ // Deposit contract
89
+ DEPOSIT_CHAIN_ID: number;
90
+ DEPOSIT_NETWORK_ID: number;
91
+ DEPOSIT_CONTRACT_ADDRESS: Uint8Array;
92
+
93
+ // Networking
94
+ MAX_REQUEST_BLOCKS: number;
95
+ MAX_REQUEST_BLOCKS_DENEB: number;
96
+ MIN_EPOCHS_FOR_BLOCK_REQUESTS: number;
97
+ MIN_EPOCHS_FOR_BLOB_SIDECARS_REQUESTS: number;
98
+ MIN_EPOCHS_FOR_DATA_COLUMN_SIDECARS_REQUESTS: number;
99
+ BLOB_SIDECAR_SUBNET_COUNT: number;
100
+ MAX_BLOBS_PER_BLOCK: number;
101
+ MAX_REQUEST_BLOB_SIDECARS: number;
102
+ BLOB_SIDECAR_SUBNET_COUNT_ELECTRA: number;
103
+ DATA_COLUMN_SIDECAR_SUBNET_COUNT: number;
104
+ MAX_BLOBS_PER_BLOCK_ELECTRA: number;
105
+ MAX_REQUEST_BLOB_SIDECARS_ELECTRA: number;
106
+ MAX_REQUEST_DATA_COLUMN_SIDECARS: number;
107
+
108
+ // Fulu
109
+ NUMBER_OF_CUSTODY_GROUPS: number;
110
+ SAMPLES_PER_SLOT: number;
111
+ CUSTODY_REQUIREMENT: number;
112
+ VALIDATOR_CUSTODY_REQUIREMENT: number;
113
+ BALANCE_PER_ADDITIONAL_CUSTODY_GROUP: number;
114
+
115
+ // Blob Scheduling
116
+ BLOB_SCHEDULE: BlobSchedule;
117
+ };
118
+
119
+ export const chainConfigTypes: SpecTypes<ChainConfig> = {
120
+ PRESET_BASE: "string",
121
+ CONFIG_NAME: "string",
122
+
123
+ // Transition
124
+ TERMINAL_TOTAL_DIFFICULTY: "bigint",
125
+ TERMINAL_BLOCK_HASH: "bytes",
126
+ TERMINAL_BLOCK_HASH_ACTIVATION_EPOCH: "number",
127
+
128
+ // Genesis
129
+ MIN_GENESIS_ACTIVE_VALIDATOR_COUNT: "number",
130
+ MIN_GENESIS_TIME: "number",
131
+ GENESIS_FORK_VERSION: "bytes",
132
+ GENESIS_DELAY: "number",
133
+
134
+ // Forking
135
+ // Altair
136
+ ALTAIR_FORK_VERSION: "bytes",
137
+ ALTAIR_FORK_EPOCH: "number",
138
+ // Bellatrix
139
+ BELLATRIX_FORK_VERSION: "bytes",
140
+ BELLATRIX_FORK_EPOCH: "number",
141
+ // Capella
142
+ CAPELLA_FORK_VERSION: "bytes",
143
+ CAPELLA_FORK_EPOCH: "number",
144
+ // DENEB
145
+ DENEB_FORK_VERSION: "bytes",
146
+ DENEB_FORK_EPOCH: "number",
147
+ // ELECTRA
148
+ ELECTRA_FORK_VERSION: "bytes",
149
+ ELECTRA_FORK_EPOCH: "number",
150
+ // FULU
151
+ FULU_FORK_VERSION: "bytes",
152
+ FULU_FORK_EPOCH: "number",
153
+ // GLOAS
154
+ GLOAS_FORK_VERSION: "bytes",
155
+ GLOAS_FORK_EPOCH: "number",
156
+
157
+ // Time parameters
158
+ SECONDS_PER_SLOT: "number",
159
+ SLOT_DURATION_MS: "number",
160
+ SECONDS_PER_ETH1_BLOCK: "number",
161
+ MIN_VALIDATOR_WITHDRAWABILITY_DELAY: "number",
162
+ SHARD_COMMITTEE_PERIOD: "number",
163
+ ETH1_FOLLOW_DISTANCE: "number",
164
+ PROPOSER_REORG_CUTOFF_BPS: "number",
165
+ ATTESTATION_DUE_BPS: "number",
166
+ AGGREGATE_DUE_BPS: "number",
167
+ // Altair
168
+ SYNC_MESSAGE_DUE_BPS: "number",
169
+ CONTRIBUTION_DUE_BPS: "number",
170
+
171
+ ATTESTATION_DUE_BPS_GLOAS: "number",
172
+ AGGREGATE_DUE_BPS_GLOAS: "number",
173
+ SYNC_MESSAGE_DUE_BPS_GLOAS: "number",
174
+ CONTRIBUTION_DUE_BPS_GLOAS: "number",
175
+ PAYLOAD_ATTESTATION_DUE_BPS: "number",
176
+
177
+ // Validator cycle
178
+ INACTIVITY_SCORE_BIAS: "number",
179
+ INACTIVITY_SCORE_RECOVERY_RATE: "number",
180
+ EJECTION_BALANCE: "number",
181
+ MIN_PER_EPOCH_CHURN_LIMIT: "number",
182
+ MAX_PER_EPOCH_ACTIVATION_CHURN_LIMIT: "number",
183
+ CHURN_LIMIT_QUOTIENT: "number",
184
+ MAX_PER_EPOCH_ACTIVATION_EXIT_CHURN_LIMIT: "number",
185
+ MIN_PER_EPOCH_CHURN_LIMIT_ELECTRA: "number",
186
+
187
+ // Fork choice
188
+ PROPOSER_SCORE_BOOST: "number",
189
+ REORG_HEAD_WEIGHT_THRESHOLD: "number",
190
+ REORG_PARENT_WEIGHT_THRESHOLD: "number",
191
+ REORG_MAX_EPOCHS_SINCE_FINALIZATION: "number",
192
+
193
+ // Deposit contract
194
+ DEPOSIT_CHAIN_ID: "number",
195
+ DEPOSIT_NETWORK_ID: "number",
196
+ DEPOSIT_CONTRACT_ADDRESS: "bytes",
197
+
198
+ // Networking
199
+ MAX_REQUEST_BLOCKS: "number",
200
+ MAX_REQUEST_BLOCKS_DENEB: "number",
201
+ MIN_EPOCHS_FOR_BLOCK_REQUESTS: "number",
202
+ MIN_EPOCHS_FOR_BLOB_SIDECARS_REQUESTS: "number",
203
+ MIN_EPOCHS_FOR_DATA_COLUMN_SIDECARS_REQUESTS: "number",
204
+ BLOB_SIDECAR_SUBNET_COUNT: "number",
205
+ DATA_COLUMN_SIDECAR_SUBNET_COUNT: "number",
206
+ MAX_BLOBS_PER_BLOCK: "number",
207
+ MAX_REQUEST_BLOB_SIDECARS: "number",
208
+ BLOB_SIDECAR_SUBNET_COUNT_ELECTRA: "number",
209
+ MAX_BLOBS_PER_BLOCK_ELECTRA: "number",
210
+ MAX_REQUEST_BLOB_SIDECARS_ELECTRA: "number",
211
+ MAX_REQUEST_DATA_COLUMN_SIDECARS: "number",
212
+
213
+ // Fulu
214
+ NUMBER_OF_CUSTODY_GROUPS: "number",
215
+ SAMPLES_PER_SLOT: "number",
216
+ CUSTODY_REQUIREMENT: "number",
217
+ VALIDATOR_CUSTODY_REQUIREMENT: "number",
218
+ BALANCE_PER_ADDITIONAL_CUSTODY_GROUP: "number",
219
+
220
+ // Blob Scheduling
221
+ BLOB_SCHEDULE: "blob_schedule",
222
+ };
223
+
224
+ export type BlobScheduleEntry = {
225
+ EPOCH: number;
226
+ MAX_BLOBS_PER_BLOCK: number;
227
+ };
228
+
229
+ export type BlobSchedule = BlobScheduleEntry[];
230
+
231
+ export function isBlobSchedule(value: unknown): value is BlobSchedule {
232
+ return (
233
+ Array.isArray(value) &&
234
+ value.every(
235
+ (entry) =>
236
+ typeof entry === "object" &&
237
+ entry !== null &&
238
+ typeof entry.EPOCH === "number" &&
239
+ typeof entry.MAX_BLOBS_PER_BLOCK === "number"
240
+ )
241
+ );
242
+ }
243
+
244
+ /** Allows values in a Spec file */
245
+ export type SpecValue = number | bigint | Uint8Array | string | BlobSchedule;
246
+
247
+ /** Type value name of each spec field. Numbers are ignored since they are the most common */
248
+ export type SpecValueType<V extends SpecValue> = V extends number
249
+ ? "number"
250
+ : V extends bigint
251
+ ? "bigint"
252
+ : V extends Uint8Array
253
+ ? "bytes"
254
+ : V extends string
255
+ ? "string"
256
+ : V extends BlobSchedule
257
+ ? "blob_schedule"
258
+ : never;
259
+
260
+ /** All possible type names for a SpecValue */
261
+ export type SpecValueTypeName = SpecValueType<SpecValue>;
262
+
263
+ export type SpecTypes<Spec extends Record<string, SpecValue>> = {
264
+ [K in keyof Spec]: SpecValueType<Spec[K]>;
265
+ };
266
+
267
+ export type SpecJson = Record<string, string | Record<string, string>[]>;
package/src/configs.ts ADDED
@@ -0,0 +1,4 @@
1
+ import {chainConfig as mainnetChainConfig} from "./chainConfig/configs/mainnet.js";
2
+ import {chainConfig as minimalChainConfig} from "./chainConfig/configs/minimal.js";
3
+
4
+ export {mainnetChainConfig, minimalChainConfig};
package/src/default.ts ADDED
@@ -0,0 +1,6 @@
1
+ import {createChainForkConfig} from "./beaconConfig.js";
2
+ import {defaultChainConfig} from "./chainConfig/index.js";
3
+
4
+ export const chainConfig = defaultChainConfig;
5
+ // for testing purpose only
6
+ export const config = createChainForkConfig(defaultChainConfig);
@@ -0,0 +1,236 @@
1
+ import {
2
+ BASIS_POINTS,
3
+ ForkAll,
4
+ ForkName,
5
+ ForkPostAltair,
6
+ ForkPostBellatrix,
7
+ ForkPostDeneb,
8
+ ForkSeq,
9
+ GENESIS_EPOCH,
10
+ SLOTS_PER_EPOCH,
11
+ isForkPostAltair,
12
+ isForkPostBellatrix,
13
+ isForkPostDeneb,
14
+ isForkPostGloas,
15
+ } from "@lodestar/params";
16
+ import {Epoch, SSZTypesFor, Slot, Version, sszTypesFor} from "@lodestar/types";
17
+ import {ChainConfig} from "../chainConfig/index.js";
18
+ import {BlobParameters, ForkBoundary, ForkConfig, ForkInfo} from "./types.js";
19
+
20
+ export * from "./types.js";
21
+
22
+ export function createForkConfig(config: ChainConfig): ForkConfig {
23
+ const phase0: ForkInfo = {
24
+ name: ForkName.phase0,
25
+ seq: ForkSeq.phase0,
26
+ epoch: GENESIS_EPOCH,
27
+ version: config.GENESIS_FORK_VERSION,
28
+ // Will never be used
29
+ prevVersion: config.GENESIS_FORK_VERSION,
30
+ prevForkName: ForkName.phase0,
31
+ };
32
+ const altair: ForkInfo = {
33
+ name: ForkName.altair,
34
+ seq: ForkSeq.altair,
35
+ epoch: config.ALTAIR_FORK_EPOCH,
36
+ version: config.ALTAIR_FORK_VERSION,
37
+ prevVersion: config.GENESIS_FORK_VERSION,
38
+ prevForkName: ForkName.phase0,
39
+ };
40
+ const bellatrix: ForkInfo = {
41
+ name: ForkName.bellatrix,
42
+ seq: ForkSeq.bellatrix,
43
+ epoch: config.BELLATRIX_FORK_EPOCH,
44
+ version: config.BELLATRIX_FORK_VERSION,
45
+ prevVersion: config.ALTAIR_FORK_VERSION,
46
+ prevForkName: ForkName.altair,
47
+ };
48
+ const capella: ForkInfo = {
49
+ name: ForkName.capella,
50
+ seq: ForkSeq.capella,
51
+ epoch: config.CAPELLA_FORK_EPOCH,
52
+ version: config.CAPELLA_FORK_VERSION,
53
+ prevVersion: config.BELLATRIX_FORK_VERSION,
54
+ prevForkName: ForkName.bellatrix,
55
+ };
56
+ const deneb: ForkInfo = {
57
+ name: ForkName.deneb,
58
+ seq: ForkSeq.deneb,
59
+ epoch: config.DENEB_FORK_EPOCH,
60
+ version: config.DENEB_FORK_VERSION,
61
+ prevVersion: config.CAPELLA_FORK_VERSION,
62
+ prevForkName: ForkName.capella,
63
+ };
64
+ const electra: ForkInfo = {
65
+ name: ForkName.electra,
66
+ seq: ForkSeq.electra,
67
+ epoch: config.ELECTRA_FORK_EPOCH,
68
+ version: config.ELECTRA_FORK_VERSION,
69
+ prevVersion: config.DENEB_FORK_VERSION,
70
+ prevForkName: ForkName.deneb,
71
+ };
72
+ const fulu: ForkInfo = {
73
+ name: ForkName.fulu,
74
+ seq: ForkSeq.fulu,
75
+ epoch: config.FULU_FORK_EPOCH,
76
+ version: config.FULU_FORK_VERSION,
77
+ prevVersion: config.ELECTRA_FORK_VERSION,
78
+ prevForkName: ForkName.electra,
79
+ };
80
+ const gloas: ForkInfo = {
81
+ name: ForkName.gloas,
82
+ seq: ForkSeq.gloas,
83
+ epoch: config.GLOAS_FORK_EPOCH,
84
+ version: config.GLOAS_FORK_VERSION,
85
+ prevVersion: config.FULU_FORK_VERSION,
86
+ prevForkName: ForkName.fulu,
87
+ };
88
+
89
+ /** Forks in order order of occurence, `phase0` first */
90
+ // Note: Downstream code relies on proper ordering.
91
+ const forks = {phase0, altair, bellatrix, capella, deneb, electra, fulu, gloas};
92
+
93
+ // Prevents allocating an array on every getForkInfo() call
94
+ const forksAscendingEpochOrder = Object.values(forks);
95
+ const forksDescendingEpochOrder = Object.values(forks).reverse();
96
+
97
+ const blobScheduleDescendingEpochOrder = [...config.BLOB_SCHEDULE].sort((a, b) => b.EPOCH - a.EPOCH);
98
+
99
+ const forkBoundariesAscendingEpochOrder: ForkBoundary[] = [
100
+ // Normal hard-forks (phase0, altair, etc.)
101
+ ...forksAscendingEpochOrder.map((fork) => ({
102
+ fork: fork.name,
103
+ epoch: fork.epoch,
104
+ })),
105
+ // Blob Parameter Only (BPO) forks
106
+ // Note: Must be appended after normal hard-forks to have precedence if scheduled at the same epoch
107
+ ...config.BLOB_SCHEDULE.map((entry) => ({
108
+ fork: forksDescendingEpochOrder.find((f) => entry.EPOCH >= f.epoch)?.name ?? phase0.name,
109
+ epoch: entry.EPOCH,
110
+ })),
111
+ ]
112
+ // Remove unscheduled fork boundaries
113
+ .filter(({epoch}) => epoch !== Infinity)
114
+ // Sort by epoch in ascending order
115
+ .sort((a, b) => a.epoch - b.epoch);
116
+
117
+ const forkBoundariesDescendingEpochOrder = [...forkBoundariesAscendingEpochOrder].reverse();
118
+
119
+ return {
120
+ forks,
121
+ forksAscendingEpochOrder,
122
+ forksDescendingEpochOrder,
123
+ forkBoundariesAscendingEpochOrder,
124
+ forkBoundariesDescendingEpochOrder,
125
+
126
+ // Fork convenience methods
127
+ getForkInfo(slot: Slot): ForkInfo {
128
+ const epoch = Math.floor(Math.max(slot, 0) / SLOTS_PER_EPOCH);
129
+ return this.getForkInfoAtEpoch(epoch);
130
+ },
131
+ getForkInfoAtEpoch(epoch: Epoch): ForkInfo {
132
+ return forks[this.getForkBoundaryAtEpoch(epoch).fork];
133
+ },
134
+ getForkBoundaryAtEpoch(epoch: Epoch): ForkBoundary {
135
+ if (epoch < 0) epoch = 0;
136
+ // NOTE: fork boundaries must be sorted by descending epoch, latest first
137
+ for (const boundary of forkBoundariesDescendingEpochOrder) {
138
+ if (epoch >= boundary.epoch) return boundary;
139
+ }
140
+ throw Error("Unreachable as phase0 is scheduled at epoch 0");
141
+ },
142
+ getForkName(slot: Slot): ForkName {
143
+ return this.getForkInfo(slot).name;
144
+ },
145
+ getForkSeq(slot: Slot): ForkSeq {
146
+ return this.getForkInfo(slot).seq;
147
+ },
148
+ getForkSeqAtEpoch(epoch: Epoch): ForkSeq {
149
+ return this.getForkInfoAtEpoch(epoch).seq;
150
+ },
151
+ getForkVersion(slot: Slot): Version {
152
+ return this.getForkInfo(slot).version;
153
+ },
154
+ getForkTypes<F extends ForkName = ForkAll>(slot: Slot): SSZTypesFor<F> {
155
+ return sszTypesFor(this.getForkName(slot)) as SSZTypesFor<F>;
156
+ },
157
+ getPostBellatrixForkTypes(slot: Slot): SSZTypesFor<ForkPostBellatrix> {
158
+ const forkName = this.getForkName(slot);
159
+ if (!isForkPostBellatrix(forkName)) {
160
+ throw Error(`Invalid slot=${slot} fork=${forkName} for post-bellatrix fork types`);
161
+ }
162
+ return sszTypesFor(forkName);
163
+ },
164
+ getPostAltairForkTypes(slot: Slot): SSZTypesFor<ForkPostAltair> {
165
+ const forkName = this.getForkName(slot);
166
+ if (!isForkPostAltair(forkName)) {
167
+ throw Error(`Invalid slot=${slot} fork=${forkName} for post-altair fork types`);
168
+ }
169
+ return sszTypesFor(forkName);
170
+ },
171
+ getPostDenebForkTypes(slot: Slot): SSZTypesFor<ForkPostDeneb> {
172
+ const forkName = this.getForkName(slot);
173
+ if (!isForkPostDeneb(forkName)) {
174
+ throw Error(`Invalid slot=${slot} fork=${forkName} for post-deneb fork types`);
175
+ }
176
+ return sszTypesFor(forkName);
177
+ },
178
+ getMaxBlobsPerBlock(epoch: Epoch): number {
179
+ const fork = this.getForkInfoAtEpoch(epoch).name;
180
+
181
+ switch (fork) {
182
+ case ForkName.electra:
183
+ return config.MAX_BLOBS_PER_BLOCK_ELECTRA;
184
+ case ForkName.deneb:
185
+ return config.MAX_BLOBS_PER_BLOCK;
186
+ }
187
+
188
+ return this.getBlobParameters(epoch).maxBlobsPerBlock;
189
+ },
190
+ getBlobParameters(epoch: Epoch): BlobParameters {
191
+ if (epoch < config.FULU_FORK_EPOCH) {
192
+ throw Error(`getBlobParameters is not available pre-fulu epoch=${epoch}`);
193
+ }
194
+
195
+ // Find the latest applicable value from blob schedule
196
+ for (const entry of blobScheduleDescendingEpochOrder) {
197
+ if (epoch >= entry.EPOCH) {
198
+ return {epoch: entry.EPOCH, maxBlobsPerBlock: entry.MAX_BLOBS_PER_BLOCK};
199
+ }
200
+ }
201
+
202
+ return {epoch: config.ELECTRA_FORK_EPOCH, maxBlobsPerBlock: config.MAX_BLOBS_PER_BLOCK_ELECTRA};
203
+ },
204
+ getAttestationDueMs(fork: ForkName): number {
205
+ if (isForkPostGloas(fork)) {
206
+ return this.getSlotComponentDurationMs(config.ATTESTATION_DUE_BPS_GLOAS);
207
+ }
208
+ return this.getSlotComponentDurationMs(config.ATTESTATION_DUE_BPS);
209
+ },
210
+ getAggregateDueMs(fork: ForkName): number {
211
+ if (isForkPostGloas(fork)) {
212
+ return this.getSlotComponentDurationMs(config.AGGREGATE_DUE_BPS_GLOAS);
213
+ }
214
+ return this.getSlotComponentDurationMs(config.AGGREGATE_DUE_BPS);
215
+ },
216
+ getSyncMessageDueMs(fork: ForkName): number {
217
+ if (isForkPostGloas(fork)) {
218
+ return this.getSlotComponentDurationMs(config.SYNC_MESSAGE_DUE_BPS_GLOAS);
219
+ }
220
+ return this.getSlotComponentDurationMs(config.SYNC_MESSAGE_DUE_BPS);
221
+ },
222
+ getSyncContributionDueMs(fork: ForkName): number {
223
+ if (isForkPostGloas(fork)) {
224
+ return this.getSlotComponentDurationMs(config.CONTRIBUTION_DUE_BPS_GLOAS);
225
+ }
226
+ return this.getSlotComponentDurationMs(config.CONTRIBUTION_DUE_BPS);
227
+ },
228
+ getProposerReorgCutoffMs(_fork: ForkName): number {
229
+ return this.getSlotComponentDurationMs(config.PROPOSER_REORG_CUTOFF_BPS);
230
+ },
231
+
232
+ getSlotComponentDurationMs(basisPoints: number): number {
233
+ return Math.round((basisPoints * config.SLOT_DURATION_MS) / BASIS_POINTS);
234
+ },
235
+ };
236
+ }
@@ -0,0 +1,68 @@
1
+ import {ForkAll, ForkName, ForkPostAltair, ForkPostBellatrix, ForkPostDeneb, ForkSeq} from "@lodestar/params";
2
+ import {Epoch, SSZTypesFor, Slot, UintNum64, Version} from "@lodestar/types";
3
+
4
+ export type ForkInfo = {
5
+ name: ForkName;
6
+ seq: ForkSeq;
7
+ epoch: Epoch;
8
+ version: Version;
9
+ prevVersion: Version;
10
+ prevForkName: ForkName;
11
+ };
12
+
13
+ /**
14
+ * Fork boundaries include both normal hard-forks (phase0, altair, etc.)
15
+ * and Blob Parameter Only (BPO) forks and are used to un-/subscribe to gossip topics
16
+ * and compute the fork digest primarily for domain separation on the p2p layer.
17
+ */
18
+ export type ForkBoundary = {fork: ForkName; epoch: Epoch};
19
+
20
+ export type BlobParameters = {epoch: Epoch; maxBlobsPerBlock: UintNum64};
21
+
22
+ /**
23
+ * Fork schedule and helper methods
24
+ */
25
+ export type ForkConfig = {
26
+ /** Forks in order order of occurence, `phase0` first */
27
+ forks: {[K in ForkName]: ForkInfo};
28
+ forksAscendingEpochOrder: ForkInfo[];
29
+ forksDescendingEpochOrder: ForkInfo[];
30
+ forkBoundariesAscendingEpochOrder: ForkBoundary[];
31
+ forkBoundariesDescendingEpochOrder: ForkBoundary[];
32
+
33
+ /** Get the hard-fork info for the active fork at `slot` */
34
+ getForkInfo(slot: Slot): ForkInfo;
35
+ /** Get the hard-fork info for the active fork at `epoch` */
36
+ getForkInfoAtEpoch(epoch: Epoch): ForkInfo;
37
+ /** Get the active fork boundary at a given `epoch` */
38
+ getForkBoundaryAtEpoch(epoch: Epoch): ForkBoundary;
39
+ /** Get the hard-fork name at a given slot */
40
+ getForkName(slot: Slot): ForkName;
41
+ /** Get the hard-fork sequence number at a given slot */
42
+ getForkSeq(slot: Slot): ForkSeq;
43
+ /** Get the hard-fork sequence number at a given epoch */
44
+ getForkSeqAtEpoch(epoch: Epoch): ForkSeq;
45
+ /** Get the hard-fork version at a given slot */
46
+ getForkVersion(slot: Slot): Version;
47
+ /** Get SSZ types by hard-fork */
48
+ getForkTypes<F extends ForkName = ForkAll>(slot: Slot): SSZTypesFor<F>;
49
+ /** Get post-altair SSZ types by hard-fork*/
50
+ getPostAltairForkTypes(slot: Slot): SSZTypesFor<ForkPostAltair>;
51
+ /** Get post-bellatrix SSZ types by hard-fork*/
52
+ getPostBellatrixForkTypes(slot: Slot): SSZTypesFor<ForkPostBellatrix>;
53
+ /** Get post-deneb SSZ types by hard-fork*/
54
+ getPostDenebForkTypes(slot: Slot): SSZTypesFor<ForkPostDeneb>;
55
+ /** Get max blobs per block at a given epoch */
56
+ getMaxBlobsPerBlock(epoch: Epoch): number;
57
+ /** Get blob parameters at a given epoch */
58
+ getBlobParameters(epoch: Epoch): BlobParameters;
59
+
60
+ getAttestationDueMs(fork: ForkName): number;
61
+ getAggregateDueMs(fork: ForkName): number;
62
+ getSyncMessageDueMs(fork: ForkName): number;
63
+ getSyncContributionDueMs(fork: ForkName): number;
64
+ getProposerReorgCutoffMs(fork: ForkName): number;
65
+
66
+ /** Convert basis points to milliseconds into the slot */
67
+ getSlotComponentDurationMs(basisPoints: number): number;
68
+ };