@lodestar/beacon-node 1.44.0-dev.552cdce8d0 → 1.44.0-dev.6c5c48ad59

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 (164) hide show
  1. package/lib/api/impl/beacon/blocks/index.d.ts.map +1 -1
  2. package/lib/api/impl/beacon/blocks/index.js +30 -0
  3. package/lib/api/impl/beacon/blocks/index.js.map +1 -1
  4. package/lib/api/impl/beacon/pool/index.d.ts.map +1 -1
  5. package/lib/api/impl/beacon/pool/index.js +1 -1
  6. package/lib/api/impl/beacon/pool/index.js.map +1 -1
  7. package/lib/api/impl/config/constants.d.ts +1 -0
  8. package/lib/api/impl/config/constants.d.ts.map +1 -1
  9. package/lib/api/impl/config/constants.js +2 -1
  10. package/lib/api/impl/config/constants.js.map +1 -1
  11. package/lib/api/impl/debug/index.d.ts.map +1 -1
  12. package/lib/api/impl/debug/index.js +69 -12
  13. package/lib/api/impl/debug/index.js.map +1 -1
  14. package/lib/api/impl/validator/index.d.ts.map +1 -1
  15. package/lib/api/impl/validator/index.js +101 -42
  16. package/lib/api/impl/validator/index.js.map +1 -1
  17. package/lib/chain/archiveStore/archiveStore.d.ts +0 -1
  18. package/lib/chain/archiveStore/archiveStore.d.ts.map +1 -1
  19. package/lib/chain/archiveStore/archiveStore.js +0 -4
  20. package/lib/chain/archiveStore/archiveStore.js.map +1 -1
  21. package/lib/chain/blocks/importBlock.d.ts.map +1 -1
  22. package/lib/chain/blocks/importBlock.js +5 -2
  23. package/lib/chain/blocks/importBlock.js.map +1 -1
  24. package/lib/chain/blocks/importExecutionPayload.d.ts.map +1 -1
  25. package/lib/chain/blocks/importExecutionPayload.js +5 -3
  26. package/lib/chain/blocks/importExecutionPayload.js.map +1 -1
  27. package/lib/chain/chain.d.ts +1 -1
  28. package/lib/chain/chain.d.ts.map +1 -1
  29. package/lib/chain/chain.js +2 -1
  30. package/lib/chain/chain.js.map +1 -1
  31. package/lib/chain/errors/executionPayloadBid.d.ts +11 -1
  32. package/lib/chain/errors/executionPayloadBid.d.ts.map +1 -1
  33. package/lib/chain/errors/executionPayloadBid.js +2 -0
  34. package/lib/chain/errors/executionPayloadBid.js.map +1 -1
  35. package/lib/chain/errors/payloadAttestation.d.ts +6 -0
  36. package/lib/chain/errors/payloadAttestation.d.ts.map +1 -1
  37. package/lib/chain/errors/payloadAttestation.js +1 -0
  38. package/lib/chain/errors/payloadAttestation.js.map +1 -1
  39. package/lib/chain/forkChoice/index.d.ts.map +1 -1
  40. package/lib/chain/forkChoice/index.js +14 -4
  41. package/lib/chain/forkChoice/index.js.map +1 -1
  42. package/lib/chain/opPools/executionPayloadBidPool.d.ts +4 -4
  43. package/lib/chain/opPools/executionPayloadBidPool.d.ts.map +1 -1
  44. package/lib/chain/opPools/executionPayloadBidPool.js +6 -4
  45. package/lib/chain/opPools/executionPayloadBidPool.js.map +1 -1
  46. package/lib/chain/prepareNextSlot.d.ts.map +1 -1
  47. package/lib/chain/prepareNextSlot.js +2 -1
  48. package/lib/chain/prepareNextSlot.js.map +1 -1
  49. package/lib/chain/produceBlock/produceBlockBody.d.ts +7 -1
  50. package/lib/chain/produceBlock/produceBlockBody.d.ts.map +1 -1
  51. package/lib/chain/produceBlock/produceBlockBody.js +107 -18
  52. package/lib/chain/produceBlock/produceBlockBody.js.map +1 -1
  53. package/lib/chain/regen/interface.d.ts +2 -1
  54. package/lib/chain/regen/interface.d.ts.map +1 -1
  55. package/lib/chain/regen/interface.js +2 -0
  56. package/lib/chain/regen/interface.js.map +1 -1
  57. package/lib/chain/regen/queued.d.ts +0 -1
  58. package/lib/chain/regen/queued.d.ts.map +1 -1
  59. package/lib/chain/regen/queued.js +0 -4
  60. package/lib/chain/regen/queued.js.map +1 -1
  61. package/lib/chain/stateCache/fifoBlockStateCache.d.ts +0 -5
  62. package/lib/chain/stateCache/fifoBlockStateCache.d.ts.map +1 -1
  63. package/lib/chain/stateCache/fifoBlockStateCache.js +0 -5
  64. package/lib/chain/stateCache/fifoBlockStateCache.js.map +1 -1
  65. package/lib/chain/stateCache/persistentCheckpointsCache.d.ts +1 -4
  66. package/lib/chain/stateCache/persistentCheckpointsCache.d.ts.map +1 -1
  67. package/lib/chain/stateCache/persistentCheckpointsCache.js +5 -2
  68. package/lib/chain/stateCache/persistentCheckpointsCache.js.map +1 -1
  69. package/lib/chain/stateCache/types.d.ts +0 -2
  70. package/lib/chain/stateCache/types.d.ts.map +1 -1
  71. package/lib/chain/stateCache/types.js.map +1 -1
  72. package/lib/chain/validation/executionPayloadBid.d.ts +7 -3
  73. package/lib/chain/validation/executionPayloadBid.d.ts.map +1 -1
  74. package/lib/chain/validation/executionPayloadBid.js +58 -15
  75. package/lib/chain/validation/executionPayloadBid.js.map +1 -1
  76. package/lib/chain/validation/payloadAttestationMessage.d.ts.map +1 -1
  77. package/lib/chain/validation/payloadAttestationMessage.js +24 -4
  78. package/lib/chain/validation/payloadAttestationMessage.js.map +1 -1
  79. package/lib/chain/validatorMonitor.d.ts +1 -0
  80. package/lib/chain/validatorMonitor.d.ts.map +1 -1
  81. package/lib/chain/validatorMonitor.js +16 -0
  82. package/lib/chain/validatorMonitor.js.map +1 -1
  83. package/lib/execution/builder/index.d.ts +1 -2
  84. package/lib/execution/builder/index.d.ts.map +1 -1
  85. package/lib/execution/builder/index.js +0 -1
  86. package/lib/execution/builder/index.js.map +1 -1
  87. package/lib/execution/engine/interface.d.ts +1 -0
  88. package/lib/execution/engine/interface.d.ts.map +1 -1
  89. package/lib/execution/engine/types.d.ts +2 -0
  90. package/lib/execution/engine/types.d.ts.map +1 -1
  91. package/lib/execution/engine/types.js +2 -0
  92. package/lib/execution/engine/types.js.map +1 -1
  93. package/lib/metrics/metrics/lodestar.d.ts +1 -1
  94. package/lib/metrics/metrics/lodestar.d.ts.map +1 -1
  95. package/lib/metrics/metrics/lodestar.js +4 -3
  96. package/lib/metrics/metrics/lodestar.js.map +1 -1
  97. package/lib/network/gossip/topic.d.ts +1 -1
  98. package/lib/network/interface.d.ts +1 -0
  99. package/lib/network/interface.d.ts.map +1 -1
  100. package/lib/network/network.d.ts +1 -0
  101. package/lib/network/network.d.ts.map +1 -1
  102. package/lib/network/network.js +5 -0
  103. package/lib/network/network.js.map +1 -1
  104. package/lib/network/processor/gossipHandlers.d.ts.map +1 -1
  105. package/lib/network/processor/gossipHandlers.js +12 -4
  106. package/lib/network/processor/gossipHandlers.js.map +1 -1
  107. package/lib/network/reqresp/handlers/dataColumnSidecarsByRange.d.ts.map +1 -1
  108. package/lib/network/reqresp/handlers/dataColumnSidecarsByRange.js +5 -0
  109. package/lib/network/reqresp/handlers/dataColumnSidecarsByRange.js.map +1 -1
  110. package/lib/network/reqresp/handlers/dataColumnSidecarsByRoot.js +1 -1
  111. package/lib/network/reqresp/handlers/dataColumnSidecarsByRoot.js.map +1 -1
  112. package/lib/network/reqresp/handlers/executionPayloadEnvelopesByRange.js +3 -3
  113. package/lib/network/reqresp/handlers/executionPayloadEnvelopesByRange.js.map +1 -1
  114. package/lib/network/reqresp/handlers/executionPayloadEnvelopesByRoot.d.ts +2 -1
  115. package/lib/network/reqresp/handlers/executionPayloadEnvelopesByRoot.d.ts.map +1 -1
  116. package/lib/network/reqresp/handlers/executionPayloadEnvelopesByRoot.js +15 -1
  117. package/lib/network/reqresp/handlers/executionPayloadEnvelopesByRoot.js.map +1 -1
  118. package/lib/network/reqresp/handlers/index.js +2 -2
  119. package/lib/network/reqresp/handlers/index.js.map +1 -1
  120. package/lib/network/reqresp/utils/dataColumnResponseValidation.d.ts.map +1 -1
  121. package/lib/network/reqresp/utils/dataColumnResponseValidation.js +22 -3
  122. package/lib/network/reqresp/utils/dataColumnResponseValidation.js.map +1 -1
  123. package/package.json +14 -16
  124. package/src/api/impl/beacon/blocks/index.ts +36 -0
  125. package/src/api/impl/beacon/pool/index.ts +3 -1
  126. package/src/api/impl/config/constants.ts +2 -0
  127. package/src/api/impl/debug/index.ts +73 -12
  128. package/src/api/impl/validator/index.ts +112 -43
  129. package/src/chain/archiveStore/archiveStore.ts +0 -5
  130. package/src/chain/blocks/importBlock.ts +10 -2
  131. package/src/chain/blocks/importExecutionPayload.ts +8 -2
  132. package/src/chain/chain.ts +2 -0
  133. package/src/chain/errors/executionPayloadBid.ts +6 -1
  134. package/src/chain/errors/payloadAttestation.ts +2 -0
  135. package/src/chain/forkChoice/index.ts +14 -4
  136. package/src/chain/opPools/executionPayloadBidPool.ts +10 -9
  137. package/src/chain/prepareNextSlot.ts +2 -1
  138. package/src/chain/produceBlock/produceBlockBody.ts +158 -25
  139. package/src/chain/regen/interface.ts +2 -1
  140. package/src/chain/regen/queued.ts +0 -5
  141. package/src/chain/stateCache/fifoBlockStateCache.ts +0 -6
  142. package/src/chain/stateCache/persistentCheckpointsCache.ts +6 -2
  143. package/src/chain/stateCache/types.ts +0 -2
  144. package/src/chain/validation/executionPayloadBid.ts +66 -19
  145. package/src/chain/validation/payloadAttestationMessage.ts +26 -4
  146. package/src/chain/validatorMonitor.ts +18 -0
  147. package/src/execution/builder/index.ts +1 -4
  148. package/src/execution/engine/interface.ts +1 -0
  149. package/src/execution/engine/types.ts +4 -0
  150. package/src/metrics/metrics/lodestar.ts +4 -3
  151. package/src/network/interface.ts +1 -0
  152. package/src/network/network.ts +11 -0
  153. package/src/network/processor/gossipHandlers.ts +17 -4
  154. package/src/network/reqresp/handlers/dataColumnSidecarsByRange.ts +6 -0
  155. package/src/network/reqresp/handlers/dataColumnSidecarsByRoot.ts +1 -1
  156. package/src/network/reqresp/handlers/executionPayloadEnvelopesByRange.ts +3 -3
  157. package/src/network/reqresp/handlers/executionPayloadEnvelopesByRoot.ts +20 -1
  158. package/src/network/reqresp/handlers/index.ts +2 -2
  159. package/src/network/reqresp/utils/dataColumnResponseValidation.ts +21 -3
  160. package/lib/execution/builder/utils.d.ts +0 -5
  161. package/lib/execution/builder/utils.d.ts.map +0 -1
  162. package/lib/execution/builder/utils.js +0 -17
  163. package/lib/execution/builder/utils.js.map +0 -1
  164. package/src/execution/builder/utils.ts +0 -19
@@ -18,7 +18,9 @@ import {
18
18
  G2_POINT_AT_INFINITY,
19
19
  IBeaconStateView,
20
20
  type IBeaconStateViewBellatrix,
21
+ computeEpochAtSlot,
21
22
  computeTimeAtSlot,
23
+ getExpectedGasLimit,
22
24
  isStatePostBellatrix,
23
25
  isStatePostCapella,
24
26
  isStatePostGloas,
@@ -48,20 +50,16 @@ import {
48
50
  gloas,
49
51
  ssz,
50
52
  } from "@lodestar/types";
51
- import {Logger, byteArrayEquals, fromHex, sleep, toHex, toPubkeyHex, toRootHex} from "@lodestar/utils";
53
+ import {GWEI_TO_WEI, Logger, byteArrayEquals, fromHex, sleep, toHex, toPubkeyHex, toRootHex} from "@lodestar/utils";
52
54
  import {ZERO_HASH_HEX} from "../../constants/index.js";
53
55
  import {numToQuantity} from "../../execution/engine/utils.js";
54
- import {
55
- IExecutionBuilder,
56
- IExecutionEngine,
57
- PayloadAttributes,
58
- PayloadId,
59
- getExpectedGasLimit,
60
- } from "../../execution/index.js";
56
+ import {IExecutionBuilder, IExecutionEngine, PayloadAttributes, PayloadId} from "../../execution/index.js";
57
+ import {getShufflingDependentRoot} from "../../util/dependentRoot.js";
61
58
  import {fromGraffitiBytes} from "../../util/graffiti.js";
62
59
  import {kzg} from "../../util/kzg.js";
63
60
  import type {BeaconChain} from "../chain.js";
64
61
  import {CommonBlockBody} from "../interface.js";
62
+ import {ProposerPreferencesPool} from "../opPools/index.js";
65
63
  import {validateBlobsAndKzgCommitments, validateCellsAndKzgCommitments} from "./validateBlobsAndKzgCommitments.js";
66
64
 
67
65
  // Time to provide the EL to generate a payload from new payload id
@@ -93,6 +91,8 @@ export type BlockAttributes = {
93
91
  slot: Slot;
94
92
  parentBlock: ProtoBlock;
95
93
  feeRecipient?: string;
94
+ /** When provided, build block with this builder bid instead of a self-build bid */
95
+ builderBid?: gloas.SignedExecutionPayloadBid;
96
96
  };
97
97
 
98
98
  export enum BlockType {
@@ -152,6 +152,28 @@ export type ProduceResult =
152
152
  | ProduceFullPhase0
153
153
  | ProduceBlinded;
154
154
 
155
+ /**
156
+ * Drop voluntary exits that `parent_execution_requests` have invalidated (e.g. a withdrawal
157
+ * request initiating an exit on the same validator). Op pool selected against the unapplied
158
+ * state, so re-validate against the post-apply state to avoid producing an invalid block.
159
+ *
160
+ * `getStateAfterParentPayload` is a thunk so the post-apply state is only materialized when
161
+ * actually needed (i.e. when extending the parent payload and there are exits to filter).
162
+ */
163
+ function maybeFilterInvalidatedVoluntaryExits(
164
+ commonBlockBody: CommonBlockBody,
165
+ isExtendingPayload: boolean,
166
+ getStateAfterParentPayload: () => IBeaconStateViewBellatrix
167
+ ): CommonBlockBody["voluntaryExits"] {
168
+ if (!isExtendingPayload || commonBlockBody.voluntaryExits.length === 0) {
169
+ return commonBlockBody.voluntaryExits;
170
+ }
171
+ const state = getStateAfterParentPayload();
172
+ return commonBlockBody.voluntaryExits.filter((signedVoluntaryExit) =>
173
+ state.isValidVoluntaryExit(signedVoluntaryExit, false)
174
+ );
175
+ }
176
+
155
177
  export async function produceBlockBody<T extends BlockType>(
156
178
  this: BeaconChain,
157
179
  blockType: T,
@@ -174,6 +196,7 @@ export async function produceBlockBody<T extends BlockType>(
174
196
  proposerIndex,
175
197
  proposerPubKey,
176
198
  commonBlockBodyPromise,
199
+ builderBid,
177
200
  } = blockAttr;
178
201
  let executionPayloadValue: Wei;
179
202
  let blockBody: AssembledBodyType<T>;
@@ -194,7 +217,43 @@ export async function produceBlockBody<T extends BlockType>(
194
217
  };
195
218
  this.logger.verbose("Producing beacon block body", logMeta);
196
219
 
197
- if (isForkPostGloas(fork)) {
220
+ if (builderBid !== undefined) {
221
+ if (!isStatePostGloas(currentState)) {
222
+ throw new Error("Expected Gloas state for builder bid block production");
223
+ }
224
+
225
+ const isExtendingPayload = byteArrayEquals(
226
+ builderBid.message.parentBlockHash,
227
+ currentState.latestExecutionPayloadBid.blockHash
228
+ );
229
+ const parentExecutionRequests = isExtendingPayload
230
+ ? await this.getParentExecutionRequests(parentBlock.slot, parentBlock.blockRoot)
231
+ : ssz.electra.ExecutionRequests.defaultValue();
232
+ executionPayloadValue = BigInt(builderBid.message.value) * GWEI_TO_WEI;
233
+
234
+ const commonBlockBody = await commonBlockBodyPromise;
235
+ const gloasBody = Object.assign({}, commonBlockBody) as gloas.BeaconBlockBody;
236
+ gloasBody.signedExecutionPayloadBid = builderBid;
237
+ gloasBody.payloadAttestations = this.payloadAttestationPool.getPayloadAttestationsForBlock(
238
+ parentBlock.blockRoot,
239
+ blockSlot - 1
240
+ );
241
+ gloasBody.parentExecutionRequests = parentExecutionRequests;
242
+ gloasBody.voluntaryExits = maybeFilterInvalidatedVoluntaryExits(commonBlockBody, isExtendingPayload, () =>
243
+ currentState.withParentPayloadApplied(parentExecutionRequests)
244
+ );
245
+ blockBody = gloasBody as AssembledBodyType<T>;
246
+
247
+ this.logger.verbose("Produced block with builder bid", {
248
+ slot: blockSlot,
249
+ builderIndex: builderBid.message.builderIndex,
250
+ bidValue: builderBid.message.value,
251
+ parentBlockHash: toRootHex(builderBid.message.parentBlockHash),
252
+ parentBlockRoot: toRootHex(builderBid.message.parentBlockRoot),
253
+ blockHash: toRootHex(builderBid.message.blockHash),
254
+ isExtendingPayload,
255
+ });
256
+ } else if (isForkPostGloas(fork)) {
198
257
  if (!isStatePostGloas(currentState)) {
199
258
  throw new Error("Expected Gloas state for Gloas block production");
200
259
  }
@@ -204,23 +263,23 @@ export async function produceBlockBody<T extends BlockType>(
204
263
  // this into a completely separate function and have pre/post gloas more separated
205
264
  const safeBlockHash = getSafeExecutionBlockHash(this.forkChoice);
206
265
  const finalizedBlockHash = this.forkChoice.getFinalizedBlock().executionPayloadBlockHash ?? ZERO_HASH_HEX;
266
+ // TODO GLOAS: post-Gloas, proposer feeRecipient is also carried (signed) in
267
+ // ProposerPreferencesPool. Consider using this unified cache instead
268
+ // see https://github.com/ChainSafe/lodestar/issues/9379
207
269
  const feeRecipient = requestedFeeRecipient ?? this.beaconProposerCache.getOrDefault(proposerIndex);
208
270
 
209
271
  const endExecutionPayload = this.metrics?.executionBlockProductionTimeSteps.startTimer();
210
272
 
211
- this.logger.verbose("Preparing execution payload from engine", {
212
- slot: blockSlot,
213
- parentBlockRoot: toRootHex(parentBlockRoot),
214
- feeRecipient,
215
- });
216
-
217
273
  // Get execution payload from EL
218
274
  let parentBlockHash: Bytes32;
219
275
  let parentExecutionRequests: electra.ExecutionRequests;
220
276
  // Apply parent payload once here as it's reused by EL prep and voluntary exit filtering below
221
277
  let stateAfterParentPayload: IBeaconStateViewBellatrix = currentState;
222
- const isExtendingPayload = this.forkChoice.shouldExtendPayload(toRootHex(parentBlockRoot));
223
- if (isExtendingPayload) {
278
+ // Spec: should_build_on_full(store, head). `parentBlock` is the proposer's head
279
+ // (set by chain.getProposerHead(slot)). Returns false when the PTC majority signalled
280
+ // the blob data is not available or the payload was not timely, forcing a build on EMPTY (reorg).
281
+ const isBuildingOnFull = this.forkChoice.shouldBuildOnFull(parentBlock, blockSlot);
282
+ if (isBuildingOnFull) {
224
283
  parentBlockHash = currentState.latestExecutionPayloadBid.blockHash;
225
284
  parentExecutionRequests = await this.getParentExecutionRequests(parentBlock.slot, parentBlock.blockRoot);
226
285
  stateAfterParentPayload = currentState.withParentPayloadApplied(parentExecutionRequests);
@@ -243,6 +302,16 @@ export async function produceBlockBody<T extends BlockType>(
243
302
  const {prepType, payloadId} = prepareRes;
244
303
  Object.assign(logMeta, {executionPayloadPrepType: prepType});
245
304
 
305
+ this.logger.verbose("Prepared execution payload from engine", {
306
+ slot: blockSlot,
307
+ parentBlockRoot: toRootHex(parentBlockRoot),
308
+ parentBlockHash: toRootHex(parentBlockHash),
309
+ feeRecipient,
310
+ prepType,
311
+ payloadId,
312
+ isBuildingOnFull,
313
+ });
314
+
246
315
  if (prepType !== PayloadPreparationType.Cached) {
247
316
  await sleep(PAYLOAD_GENERATION_TIME_MS);
248
317
  }
@@ -296,14 +365,11 @@ export async function produceBlockBody<T extends BlockType>(
296
365
  blockSlot - 1
297
366
  );
298
367
  gloasBody.parentExecutionRequests = parentExecutionRequests;
299
- // Drop voluntary exits that parent_execution_requests have invalidated (e.g. a withdrawal
300
- // request initiating an exit on the same validator). Op pool selected against the unapplied
301
- // state, so re-validate against the post-apply state to avoid producing an invalid block.
302
- if (isExtendingPayload && commonBlockBody.voluntaryExits.length > 0) {
303
- gloasBody.voluntaryExits = commonBlockBody.voluntaryExits.filter((signedVoluntaryExit) =>
304
- stateAfterParentPayload.isValidVoluntaryExit(signedVoluntaryExit, false)
305
- );
306
- }
368
+ gloasBody.voluntaryExits = maybeFilterInvalidatedVoluntaryExits(
369
+ commonBlockBody,
370
+ isBuildingOnFull,
371
+ () => stateAfterParentPayload
372
+ );
307
373
  blockBody = gloasBody as AssembledBodyType<T>;
308
374
 
309
375
  // Store execution payload data required to construct execution payload envelope later
@@ -324,6 +390,7 @@ export async function produceBlockBody<T extends BlockType>(
324
390
  fetchedTime,
325
391
  executionBlockHash: toRootHex(executionPayload.blockHash),
326
392
  blobs: blobsBundle.commitments.length,
393
+ gasLimit: executionPayload.gasLimit,
327
394
  });
328
395
 
329
396
  Object.assign(logMeta, {
@@ -633,6 +700,8 @@ export async function prepareExecutionPayload(
633
700
  chain: {
634
701
  executionEngine: IExecutionEngine;
635
702
  config: ChainForkConfig;
703
+ forkChoice: IForkChoice;
704
+ proposerPreferencesPool: ProposerPreferencesPool;
636
705
  },
637
706
  logger: Logger,
638
707
  fork: ForkPostBellatrix,
@@ -733,6 +802,7 @@ export function getPayloadAttributesForSSE(
733
802
  chain: {
734
803
  config: ChainForkConfig;
735
804
  forkChoice: IForkChoice;
805
+ proposerPreferencesPool: ProposerPreferencesPool;
736
806
  },
737
807
  {
738
808
  prepareState,
@@ -789,6 +859,8 @@ function preparePayloadAttributes(
789
859
  fork: ForkPostBellatrix,
790
860
  chain: {
791
861
  config: ChainForkConfig;
862
+ forkChoice: IForkChoice;
863
+ proposerPreferencesPool: ProposerPreferencesPool;
792
864
  },
793
865
  {
794
866
  prepareState,
@@ -851,12 +923,73 @@ function preparePayloadAttributes(
851
923
  }
852
924
 
853
925
  if (ForkSeq[fork] >= ForkSeq.gloas) {
926
+ if (!isStatePostGloas(prepareState)) {
927
+ throw new Error("Expected Gloas state for Gloas payload attributes");
928
+ }
854
929
  (payloadAttributes as gloas.SSEPayloadAttributes["payloadAttributes"]).slotNumber = prepareSlot;
930
+ (payloadAttributes as gloas.SSEPayloadAttributes["payloadAttributes"]).targetGasLimit = getProposerTargetGasLimit(
931
+ chain,
932
+ prepareSlot,
933
+ parentBlockRoot,
934
+ parentBlockHash
935
+ );
855
936
  }
856
937
 
857
938
  return payloadAttributes;
858
939
  }
859
940
 
941
+ /**
942
+ * Resolve the proposer's preferred (target) gas limit for the Gloas `PayloadAttributesV4`
943
+ * `targetGasLimit` field (consensus-specs#5235, execution-apis#796).
944
+ *
945
+ * Sourced from the `SignedProposerPreferences` the proposer's VC submitted to the pool
946
+ * (same `(slot, dependent_root)` lookup as gossip bid validation). When no matching
947
+ * preferences are pooled, target the parent payload's gas limit so the gas limit stays
948
+ * unchanged (`is_gas_limit_target_compatible` then requires `gas_limit == parent_gas_limit`).
949
+ *
950
+ * The parent payload's gas_limit is read from fork choice — the variant matching
951
+ * `(parentBlockRoot, parentBlockHash)` carries the correct value for both FULL parents
952
+ * (FULL.executionPayloadGasLimit = delivered payload's gas_limit) and EMPTY parents
953
+ * (EMPTY.executionPayloadGasLimit = inherited grandparent's gas_limit).
954
+ */
955
+ function getProposerTargetGasLimit(
956
+ chain: {forkChoice: IForkChoice; proposerPreferencesPool: ProposerPreferencesPool},
957
+ prepareSlot: Slot,
958
+ parentBlockRoot: Root,
959
+ parentBlockHash: Bytes32
960
+ ): number {
961
+ const parentBlockRootHex = toRootHex(parentBlockRoot);
962
+ const parentBlock = chain.forkChoice.getBlockHexDefaultStatus(parentBlockRootHex);
963
+ const dependentRootHex = (() => {
964
+ if (parentBlock === null) {
965
+ return null;
966
+ }
967
+ try {
968
+ return getShufflingDependentRoot(
969
+ chain.forkChoice,
970
+ computeEpochAtSlot(prepareSlot),
971
+ computeEpochAtSlot(parentBlock.slot),
972
+ parentBlock
973
+ );
974
+ } catch {
975
+ return null;
976
+ }
977
+ })();
978
+
979
+ const pref = dependentRootHex !== null ? chain.proposerPreferencesPool.get(prepareSlot, dependentRootHex) : null;
980
+ if (pref !== null) {
981
+ return pref.message.targetGasLimit;
982
+ }
983
+
984
+ const parentPayloadVariant = chain.forkChoice.getBlockHexAndBlockHash(parentBlockRootHex, toRootHex(parentBlockHash));
985
+ if (parentPayloadVariant === null || parentPayloadVariant.executionPayloadBlockHash === null) {
986
+ throw new Error(
987
+ `Cannot resolve parent payload gas_limit for proposer targetGasLimit fallback parentBlockRoot=${parentBlockRootHex} parentBlockHash=${toRootHex(parentBlockHash)}`
988
+ );
989
+ }
990
+ return parentPayloadVariant.executionPayloadGasLimit;
991
+ }
992
+
860
993
  export async function produceCommonBlockBody<T extends BlockType>(
861
994
  this: BeaconChain,
862
995
  blockType: T,
@@ -17,10 +17,12 @@ export enum RegenCaller {
17
17
  predictProposerHead = "predictProposerHead",
18
18
  produceAttestationData = "produceAttestationData",
19
19
  processBlocksInEpoch = "processBlocksInEpoch",
20
+ importExecutionPayload = "importExecutionPayload",
20
21
  validateGossipAggregateAndProof = "validateGossipAggregateAndProof",
21
22
  validateGossipAttestation = "validateGossipAttestation",
22
23
  validateGossipVoluntaryExit = "validateGossipVoluntaryExit",
23
24
  validateGossipExecutionPayloadBid = "validateGossipExecutionPayloadBid",
25
+ validateGossipPayloadAttestationMessage = "validateGossipPayloadAttestationMessage",
24
26
  validateGossipProposerPreferences = "validateGossipProposerPreferences",
25
27
  onForkChoiceFinalized = "onForkChoiceFinalized",
26
28
  restApi = "restApi",
@@ -45,7 +47,6 @@ export interface IStateRegenerator extends IStateRegeneratorInternal {
45
47
  getCheckpointStateSync(cp: CheckpointHex): IBeaconStateView | null;
46
48
  getClosestHeadState(head: ProtoBlock): IBeaconStateView | null;
47
49
  pruneOnCheckpoint(finalizedEpoch: Epoch, justifiedEpoch: Epoch, headStateRoot: RootHex): void;
48
- pruneOnFinalized(finalizedEpoch: Epoch): void;
49
50
  processState(blockRootHex: RootHex, postState: IBeaconStateView): void;
50
51
  addCheckpointState(cp: phase0.Checkpoint, item: IBeaconStateView): void;
51
52
  updateHeadState(newHead: ProtoBlock, maybeHeadState: IBeaconStateView): void;
@@ -143,11 +143,6 @@ export class QueuedStateRegenerator implements IStateRegenerator {
143
143
  this.blockStateCache.prune(headStateRoot);
144
144
  }
145
145
 
146
- pruneOnFinalized(finalizedEpoch: number): void {
147
- this.checkpointStateCache.pruneFinalized(finalizedEpoch);
148
- this.blockStateCache.deleteAllBeforeEpoch(finalizedEpoch);
149
- }
150
-
151
146
  processState(blockRootHex: RootHex, postState: IBeaconStateView): void {
152
147
  this.blockStateCache.add(postState);
153
148
  this.checkpointStateCache.processState(blockRootHex, postState).catch((e) => {
@@ -167,12 +167,6 @@ export class FIFOBlockStateCache implements BlockStateCache {
167
167
  }
168
168
  }
169
169
 
170
- /**
171
- * No need for this implementation
172
- * This is only to conform to the old api
173
- */
174
- deleteAllBeforeEpoch(): void {}
175
-
176
170
  /**
177
171
  * ONLY FOR DEBUGGING PURPOSES. For lodestar debug API.
178
172
  */
@@ -414,11 +414,12 @@ export class PersistentCheckpointStateCache implements CheckpointStateCache {
414
414
 
415
415
  /**
416
416
  * Prune all checkpoint states before the provided finalized epoch.
417
+ * Driven sequentially from processState() so it never interleaves with persist.
417
418
  */
418
- pruneFinalized(finalizedEpoch: Epoch): void {
419
+ private async pruneFinalized(finalizedEpoch: Epoch): Promise<void> {
419
420
  for (const epoch of this.epochIndex.keys()) {
420
421
  if (epoch < finalizedEpoch) {
421
- this.deleteAllEpochItems(epoch).catch((e) =>
422
+ await this.deleteAllEpochItems(epoch).catch((e) =>
422
423
  this.logger.debug("Error delete all epoch items", {epoch, finalizedEpoch}, e as Error)
423
424
  );
424
425
  }
@@ -476,6 +477,9 @@ export class PersistentCheckpointStateCache implements CheckpointStateCache {
476
477
  * As of Mar 2024, it takes <=350ms to persist a holesky state on fast server
477
478
  */
478
479
  async processState(blockRootHex: RootHex, state: IBeaconStateView): Promise<number> {
480
+ // prune finalized in the same flow so a finalized cp state is pruned, never persisted
481
+ await this.pruneFinalized(state.finalizedCheckpoint.epoch);
482
+
479
483
  let persistCount = 0;
480
484
  // it's important to sort the epochs in ascending order, in case of big reorg we always want to keep the most recent checkpoint states
481
485
  const sortedEpochs = Array.from(this.epochIndex.keys()).sort((a, b) => a - b);
@@ -30,7 +30,6 @@ export interface BlockStateCache {
30
30
  clear(): void;
31
31
  size: number;
32
32
  prune(headStateRootHex: RootHex): void;
33
- deleteAllBeforeEpoch(finalizedEpoch: Epoch): void;
34
33
  dumpSummary(): routes.lodestar.StateCacheItem[];
35
34
  /** Expose beacon states stored in cache. Use with caution */
36
35
  getStates(): IterableIterator<IBeaconStateView>;
@@ -67,7 +66,6 @@ export interface CheckpointStateCache {
67
66
  getOrReloadLatest(rootHex: RootHex, maxEpoch: Epoch): Promise<IBeaconStateView | null>;
68
67
  updatePreComputedCheckpoint(rootHex: RootHex, epoch: Epoch): number | null;
69
68
  prune(finalizedEpoch: Epoch, justifiedEpoch: Epoch): void;
70
- pruneFinalized(finalizedEpoch: Epoch): void;
71
69
  processState(blockRootHex: RootHex, state: IBeaconStateView): Promise<number>;
72
70
  clear(): void;
73
71
  dumpSummary(): routes.lodestar.StateCacheItem[];
@@ -4,9 +4,10 @@ import {
4
4
  createSingleSignatureSetFromComponents,
5
5
  getExecutionPayloadBidSigningRoot,
6
6
  isActiveBuilder,
7
+ isGasLimitTargetCompatible,
7
8
  isStatePostGloas,
8
9
  } from "@lodestar/state-transition";
9
- import {gloas} from "@lodestar/types";
10
+ import {ValidatorIndex, gloas} from "@lodestar/types";
10
11
  import {byteArrayEquals, toHex, toRootHex} from "@lodestar/utils";
11
12
  import {getShufflingDependentRoot} from "../../util/dependentRoot.js";
12
13
  import {ExecutionPayloadBidError, ExecutionPayloadBidErrorCode, GossipAction} from "../errors/index.js";
@@ -16,28 +17,24 @@ import {RegenCaller} from "../regen/index.js";
16
17
  export async function validateApiExecutionPayloadBid(
17
18
  chain: IBeaconChain,
18
19
  signedExecutionPayloadBid: gloas.SignedExecutionPayloadBid
19
- ): Promise<void> {
20
+ ): Promise<{proposerIndex: ValidatorIndex}> {
20
21
  return validateExecutionPayloadBid(chain, signedExecutionPayloadBid);
21
22
  }
22
23
 
23
24
  export async function validateGossipExecutionPayloadBid(
24
25
  chain: IBeaconChain,
25
26
  signedExecutionPayloadBid: gloas.SignedExecutionPayloadBid
26
- ): Promise<void> {
27
+ ): Promise<{proposerIndex: ValidatorIndex}> {
27
28
  return validateExecutionPayloadBid(chain, signedExecutionPayloadBid);
28
29
  }
29
30
 
30
31
  async function validateExecutionPayloadBid(
31
32
  chain: IBeaconChain,
32
33
  signedExecutionPayloadBid: gloas.SignedExecutionPayloadBid
33
- ): Promise<void> {
34
+ ): Promise<{proposerIndex: ValidatorIndex}> {
34
35
  const bid = signedExecutionPayloadBid.message;
35
36
  const parentBlockRootHex = toRootHex(bid.parentBlockRoot);
36
37
  const parentBlockHashHex = toRootHex(bid.parentBlockHash);
37
- const state = await chain.getHeadStateAtCurrentEpoch(RegenCaller.validateGossipExecutionPayloadBid);
38
- if (!isStatePostGloas(state)) {
39
- throw new Error(`Expected gloas+ state for execution payload bid validation, got fork=${state.forkName}`);
40
- }
41
38
 
42
39
  // [IGNORE] `bid.slot` is the current slot or the next slot.
43
40
  const currentSlot = chain.clock.currentSlot;
@@ -60,6 +57,17 @@ async function validateExecutionPayloadBid(
60
57
  });
61
58
  }
62
59
 
60
+ // [REJECT] The bid is for a higher slot than its parent block -- i.e.
61
+ // validate that `bid.slot` is greater than the slot of the block with root
62
+ // `bid.parent_block_root`.
63
+ if (bid.slot <= parentBlock.slot) {
64
+ throw new ExecutionPayloadBidError(GossipAction.REJECT, {
65
+ code: ExecutionPayloadBidErrorCode.NOT_LATER_THAN_PARENT,
66
+ parentSlot: parentBlock.slot,
67
+ slot: bid.slot,
68
+ });
69
+ }
70
+
63
71
  // [IGNORE] A `SignedProposerPreferences` matching `bid.slot` and the bid's branch has been
64
72
  // seen — i.e. `proposal_slot == bid.slot` AND `dependent_root ==
65
73
  // get_proposer_dependent_root(parent_state, compute_epoch_at_slot(bid.slot))`.
@@ -99,9 +107,31 @@ async function validateExecutionPayloadBid(
99
107
  });
100
108
  }
101
109
 
110
+ // Use the bid's parent branch state for builder checks
111
+ const state = await chain.regen
112
+ .getBlockSlotState(parentBlock, bid.slot, {dontTransferCache: true}, RegenCaller.validateGossipExecutionPayloadBid)
113
+ .catch(() => {
114
+ throw new ExecutionPayloadBidError(GossipAction.IGNORE, {
115
+ code: ExecutionPayloadBidErrorCode.UNKNOWN_BLOCK_ROOT,
116
+ parentBlockRoot: parentBlockRootHex,
117
+ });
118
+ });
119
+
120
+ if (!isStatePostGloas(state)) {
121
+ throw new Error(`Expected gloas+ state for execution payload bid validation, got fork=${state.forkName}`);
122
+ }
123
+
102
124
  // [REJECT] `bid.builder_index` is a valid/active builder index -- i.e.
103
125
  // `is_active_builder(state, bid.builder_index)` returns `True`.
104
- const builder = state.getBuilder(bid.builderIndex);
126
+ let builder: gloas.Builder;
127
+ try {
128
+ builder = state.getBuilder(bid.builderIndex);
129
+ } catch {
130
+ throw new ExecutionPayloadBidError(GossipAction.REJECT, {
131
+ code: ExecutionPayloadBidErrorCode.BUILDER_NOT_ELIGIBLE,
132
+ builderIndex: bid.builderIndex,
133
+ });
134
+ }
105
135
  if (!isActiveBuilder(builder, state.finalizedCheckpoint.epoch)) {
106
136
  throw new ExecutionPayloadBidError(GossipAction.REJECT, {
107
137
  code: ExecutionPayloadBidErrorCode.BUILDER_NOT_ELIGIBLE,
@@ -128,14 +158,33 @@ async function validateExecutionPayloadBid(
128
158
  });
129
159
  }
130
160
 
131
- // [REJECT] `bid.gas_limit == proposer_preferences.gas_limit`.
161
+ // [IGNORE] `bid.parent_block_hash` is the block hash of a known execution payload in fork
162
+ // choice. Looks up the variant of `bid.parent_block_root` whose payload hash matches
163
+ // `bid.parent_block_hash` — works for both FULL parents (FULL variant carries the delivered
164
+ // payload's hash) and EMPTY parents (EMPTY/PENDING variants carry the inherited parent
165
+ // payload's hash, since the new block doesn't have its own payload). Variant carries the
166
+ // executed payload's gas_limit, which we use as `parent_gas_limit` below.
167
+ const parentPayloadVariant = chain.forkChoice.getBlockHexAndBlockHash(parentBlockRootHex, parentBlockHashHex);
168
+ if (parentPayloadVariant === null || parentPayloadVariant.executionPayloadBlockHash === null) {
169
+ throw new ExecutionPayloadBidError(GossipAction.IGNORE, {
170
+ code: ExecutionPayloadBidErrorCode.UNKNOWN_PARENT_BLOCK_HASH,
171
+ parentBlockHash: parentBlockHashHex,
172
+ });
173
+ }
174
+
175
+ // [IGNORE] `is_gas_limit_target_compatible(parent_gas_limit, bid.gas_limit, target_gas_limit)`,
176
+ // where `parent_gas_limit` is the `gas_limit` of the parent execution payload and
177
+ // `target_gas_limit` is `proposer_preferences.target_gas_limit`.
132
178
  const bidGasLimit = Number(bid.gasLimit);
133
- if (bidGasLimit !== proposerPreferences.message.gasLimit) {
134
- throw new ExecutionPayloadBidError(GossipAction.REJECT, {
179
+ const parentGasLimit = parentPayloadVariant.executionPayloadGasLimit;
180
+ const targetGasLimit = proposerPreferences.message.targetGasLimit;
181
+ if (!isGasLimitTargetCompatible(parentGasLimit, bidGasLimit, targetGasLimit)) {
182
+ throw new ExecutionPayloadBidError(GossipAction.IGNORE, {
135
183
  code: ExecutionPayloadBidErrorCode.PROPOSER_PREFERENCES_GAS_LIMIT_MISMATCH,
136
184
  builderIndex: bid.builderIndex,
137
185
  bidGasLimit,
138
- expectedGasLimit: proposerPreferences.message.gasLimit,
186
+ parentGasLimit,
187
+ targetGasLimit,
139
188
  });
140
189
  }
141
190
 
@@ -166,11 +215,11 @@ async function validateExecutionPayloadBid(
166
215
  // [IGNORE] this bid is the highest value bid seen for the tuple
167
216
  // `(bid.slot, bid.parent_block_hash, bid.parent_block_root)`.
168
217
  const bestBid = chain.executionPayloadBidPool.getBestBid(bid.slot, parentBlockHashHex, parentBlockRootHex);
169
- if (bestBid !== null && bestBid.value >= bid.value) {
218
+ if (bestBid !== null && bestBid.message.value >= bid.value) {
170
219
  throw new ExecutionPayloadBidError(GossipAction.IGNORE, {
171
220
  code: ExecutionPayloadBidErrorCode.BID_TOO_LOW,
172
221
  bidValue: bid.value,
173
- currentHighestBid: bestBid.value,
222
+ currentHighestBid: bestBid.message.value,
174
223
  });
175
224
  }
176
225
  // [IGNORE] `bid.value` is less or equal than the builder's excess balance --
@@ -183,10 +232,6 @@ async function validateExecutionPayloadBid(
183
232
  });
184
233
  }
185
234
 
186
- // [IGNORE] `bid.parent_block_hash` is the block hash of a known execution
187
- // payload in fork choice.
188
- // TODO GLOAS: implement this
189
-
190
235
  // [REJECT] `signed_execution_payload_bid.signature` is valid with respect to the `bid.builder_index`.
191
236
  const signatureSet = createSingleSignatureSetFromComponents(
192
237
  PublicKey.fromBytes(builder.pubkey),
@@ -204,4 +249,6 @@ async function validateExecutionPayloadBid(
204
249
 
205
250
  // Valid
206
251
  chain.seenExecutionPayloadBids.add(bid.slot, bid.builderIndex);
252
+
253
+ return {proposerIndex: proposerPreferences.message.validatorIndex};
207
254
  }
@@ -8,6 +8,7 @@ import {RootHex, gloas, ssz} from "@lodestar/types";
8
8
  import {toRootHex} from "@lodestar/utils";
9
9
  import {GossipAction, PayloadAttestationError, PayloadAttestationErrorCode} from "../errors/index.js";
10
10
  import {IBeaconChain} from "../index.js";
11
+ import {RegenCaller} from "../regen/index.js";
11
12
 
12
13
  export type PayloadAttestationValidationResult = {
13
14
  attDataRootHex: RootHex;
@@ -61,22 +62,43 @@ async function validatePayloadAttestationMessage(
61
62
  // [IGNORE] The message's block `data.beacon_block_root` has been seen (via
62
63
  // gossip or non-gossip sources) (a client MAY queue attestation for processing
63
64
  // once the block is retrieved. Note a client might want to request payload after).
64
- if (!chain.forkChoice.hasBlock(data.beaconBlockRoot)) {
65
+ const block = chain.forkChoice.getBlockDefaultStatus(data.beaconBlockRoot);
66
+ if (!block) {
65
67
  throw new PayloadAttestationError(GossipAction.IGNORE, {
66
68
  code: PayloadAttestationErrorCode.UNKNOWN_BLOCK_ROOT,
67
69
  blockRoot: toRootHex(data.beaconBlockRoot),
68
70
  });
69
71
  }
70
72
 
71
- const state = chain.getHeadState();
72
- if (!isStatePostGloas(state)) {
73
- throw new Error(`Expected gloas+ state for payload attestation validation, got fork=${state.forkName}`);
73
+ // [IGNORE] The block referenced by `data.beacon_block_root` is at slot `data.slot`,
74
+ // i.e. the block has `block.slot == data.slot`.
75
+ if (block.slot !== data.slot) {
76
+ throw new PayloadAttestationError(GossipAction.IGNORE, {
77
+ code: PayloadAttestationErrorCode.INVALID_BLOCK_SLOT,
78
+ blockRoot: toRootHex(data.beaconBlockRoot),
79
+ blockSlot: block.slot,
80
+ slot: data.slot,
81
+ });
74
82
  }
75
83
 
76
84
  // [REJECT] The message's block `data.beacon_block_root` passes validation.
77
85
  // TODO GLOAS: implement this. Technically if we cannot get proto block from fork choice,
78
86
  // it is possible that the block didn't pass the validation
79
87
 
88
+ // Use the referenced block's branch state for the PTC committee check
89
+ const state = await chain.regen
90
+ .getBlockSlotState(block, data.slot, {dontTransferCache: true}, RegenCaller.validateGossipPayloadAttestationMessage)
91
+ .catch(() => {
92
+ throw new PayloadAttestationError(GossipAction.IGNORE, {
93
+ code: PayloadAttestationErrorCode.UNKNOWN_BLOCK_ROOT,
94
+ blockRoot: toRootHex(data.beaconBlockRoot),
95
+ });
96
+ });
97
+
98
+ if (!isStatePostGloas(state)) {
99
+ throw new Error(`Expected gloas+ state for payload attestation validation, got fork=${state.forkName}`);
100
+ }
101
+
80
102
  // [REJECT] The message's validator index is within the payload committee in
81
103
  // `get_ptc(state, data.slot)`. The `state` is the head state corresponding to
82
104
  // processing the block up to the current slot as determined by the fork choice.
@@ -66,6 +66,7 @@ export type ValidatorMonitor = {
66
66
  delaySec: Seconds,
67
67
  envelope: gloas.SignedExecutionPayloadEnvelope
68
68
  ): void;
69
+ registerExecutionPayloadBid(src: OpSource, proposerIndex: ValidatorIndex, bid: gloas.ExecutionPayloadBid): void;
69
70
  registerImportedBlock(block: BeaconBlock, data: {proposerBalanceDelta: number}): void;
70
71
  onPoolSubmitUnaggregatedAttestation(
71
72
  seenTimestampSec: number,
@@ -459,6 +460,23 @@ export function createValidatorMonitor(
459
460
  // TODO GLOAS: implement execution payload envelope monitoring
460
461
  },
461
462
 
463
+ registerExecutionPayloadBid(src, proposerIndex, bid) {
464
+ if (!validators.has(proposerIndex)) {
465
+ return;
466
+ }
467
+ log("Received an execution payload bid for monitored proposer", {
468
+ slot: bid.slot,
469
+ proposerIndex,
470
+ src,
471
+ builderIndex: bid.builderIndex,
472
+ gasLimit: bid.gasLimit,
473
+ value: bid.value.toString(),
474
+ parentBlockRoot: toRootHex(bid.parentBlockRoot),
475
+ parentBlockHash: toRootHex(bid.parentBlockHash),
476
+ blockHash: toRootHex(bid.blockHash),
477
+ });
478
+ },
479
+
462
480
  registerImportedBlock(block, {proposerBalanceDelta}) {
463
481
  const validator = validators.get(block.proposerIndex);
464
482
  if (validator) {
@@ -1,11 +1,8 @@
1
1
  import {ChainForkConfig} from "@lodestar/config";
2
2
  import {Logger} from "@lodestar/logger";
3
3
  import {Metrics} from "../../metrics/metrics.js";
4
- import {IExecutionBuilder} from "./interface.js";
5
-
6
- export {getExpectedGasLimit} from "./utils.js";
7
-
8
4
  import {ExecutionBuilderHttp, ExecutionBuilderHttpOpts, defaultExecutionBuilderHttpOpts} from "./http.js";
5
+ import {IExecutionBuilder} from "./interface.js";
9
6
 
10
7
  export {ExecutionBuilderHttp, defaultExecutionBuilderHttpOpts};
11
8
 
@@ -88,6 +88,7 @@ export type PayloadAttributes = {
88
88
  withdrawals?: capella.Withdrawal[];
89
89
  parentBeaconBlockRoot?: Uint8Array;
90
90
  slotNumber?: number; // EIP-7843
91
+ targetGasLimit?: number; // GLOAS (PayloadAttributesV4, execution-apis#796)
91
92
  };
92
93
 
93
94
  export type VersionedHashes = Uint8Array[];