@lodestar/beacon-node 1.43.0-dev.ca1fc40294 → 1.43.0-dev.dfb984e779
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.
- package/lib/api/impl/beacon/blocks/index.d.ts.map +1 -1
- package/lib/api/impl/beacon/blocks/index.js +3 -2
- package/lib/api/impl/beacon/blocks/index.js.map +1 -1
- package/lib/api/impl/lodestar/index.js +1 -1
- package/lib/api/impl/lodestar/index.js.map +1 -1
- package/lib/chain/blocks/importBlock.d.ts.map +1 -1
- package/lib/chain/blocks/importBlock.js +6 -3
- package/lib/chain/blocks/importBlock.js.map +1 -1
- package/lib/chain/blocks/importExecutionPayload.d.ts +26 -14
- package/lib/chain/blocks/importExecutionPayload.d.ts.map +1 -1
- package/lib/chain/blocks/importExecutionPayload.js +73 -77
- package/lib/chain/blocks/importExecutionPayload.js.map +1 -1
- package/lib/chain/blocks/index.d.ts +5 -3
- package/lib/chain/blocks/index.d.ts.map +1 -1
- package/lib/chain/blocks/index.js +28 -10
- package/lib/chain/blocks/index.js.map +1 -1
- package/lib/chain/blocks/payloadEnvelopeProcessor.js +2 -2
- package/lib/chain/blocks/payloadEnvelopeProcessor.js.map +1 -1
- package/lib/chain/blocks/types.d.ts +14 -20
- package/lib/chain/blocks/types.d.ts.map +1 -1
- package/lib/chain/blocks/utils/chainSegment.d.ts +23 -2
- package/lib/chain/blocks/utils/chainSegment.d.ts.map +1 -1
- package/lib/chain/blocks/utils/chainSegment.js +81 -12
- package/lib/chain/blocks/utils/chainSegment.js.map +1 -1
- package/lib/chain/blocks/verifyBlock.d.ts +3 -2
- package/lib/chain/blocks/verifyBlock.d.ts.map +1 -1
- package/lib/chain/blocks/verifyBlock.js +30 -5
- package/lib/chain/blocks/verifyBlock.js.map +1 -1
- package/lib/chain/blocks/verifyBlocksSanityChecks.d.ts.map +1 -1
- package/lib/chain/blocks/verifyBlocksSanityChecks.js +15 -4
- package/lib/chain/blocks/verifyBlocksSanityChecks.js.map +1 -1
- package/lib/chain/blocks/verifyExecutionPayloadEnvelope.d.ts +24 -0
- package/lib/chain/blocks/verifyExecutionPayloadEnvelope.d.ts.map +1 -0
- package/lib/chain/blocks/verifyExecutionPayloadEnvelope.js +76 -0
- package/lib/chain/blocks/verifyExecutionPayloadEnvelope.js.map +1 -0
- package/lib/chain/blocks/writePayloadEnvelopeInputToDb.d.ts +1 -1
- package/lib/chain/blocks/writePayloadEnvelopeInputToDb.d.ts.map +1 -1
- package/lib/chain/blocks/writePayloadEnvelopeInputToDb.js +2 -11
- package/lib/chain/blocks/writePayloadEnvelopeInputToDb.js.map +1 -1
- package/lib/chain/chain.d.ts +3 -2
- package/lib/chain/chain.d.ts.map +1 -1
- package/lib/chain/chain.js +14 -3
- package/lib/chain/chain.js.map +1 -1
- package/lib/chain/errors/blockError.d.ts +8 -1
- package/lib/chain/errors/blockError.d.ts.map +1 -1
- package/lib/chain/errors/blockError.js +2 -0
- package/lib/chain/errors/blockError.js.map +1 -1
- package/lib/chain/errors/executionPayloadBid.d.ts +5 -0
- package/lib/chain/errors/executionPayloadBid.d.ts.map +1 -1
- package/lib/chain/errors/executionPayloadBid.js +1 -0
- package/lib/chain/errors/executionPayloadBid.js.map +1 -1
- package/lib/chain/errors/executionPayloadEnvelope.d.ts +5 -0
- package/lib/chain/errors/executionPayloadEnvelope.d.ts.map +1 -1
- package/lib/chain/errors/executionPayloadEnvelope.js +1 -0
- package/lib/chain/errors/executionPayloadEnvelope.js.map +1 -1
- package/lib/chain/forkChoice/index.js +2 -2
- package/lib/chain/forkChoice/index.js.map +1 -1
- package/lib/chain/interface.d.ts +3 -2
- package/lib/chain/interface.d.ts.map +1 -1
- package/lib/chain/interface.js.map +1 -1
- package/lib/chain/prepareNextSlot.d.ts.map +1 -1
- package/lib/chain/prepareNextSlot.js +30 -10
- package/lib/chain/prepareNextSlot.js.map +1 -1
- package/lib/chain/produceBlock/produceBlockBody.d.ts +3 -2
- package/lib/chain/produceBlock/produceBlockBody.d.ts.map +1 -1
- package/lib/chain/produceBlock/produceBlockBody.js +34 -13
- package/lib/chain/produceBlock/produceBlockBody.js.map +1 -1
- package/lib/chain/seenCache/seenPayloadEnvelopeInput.d.ts +11 -4
- package/lib/chain/seenCache/seenPayloadEnvelopeInput.d.ts.map +1 -1
- package/lib/chain/seenCache/seenPayloadEnvelopeInput.js +20 -18
- package/lib/chain/seenCache/seenPayloadEnvelopeInput.js.map +1 -1
- package/lib/chain/validation/block.d.ts.map +1 -1
- package/lib/chain/validation/block.js +1 -0
- package/lib/chain/validation/block.js.map +1 -1
- package/lib/chain/validation/executionPayloadBid.d.ts.map +1 -1
- package/lib/chain/validation/executionPayloadBid.js +13 -1
- package/lib/chain/validation/executionPayloadBid.js.map +1 -1
- package/lib/chain/validation/executionPayloadEnvelope.d.ts.map +1 -1
- package/lib/chain/validation/executionPayloadEnvelope.js +11 -1
- package/lib/chain/validation/executionPayloadEnvelope.js.map +1 -1
- package/lib/metrics/metrics/lodestar.d.ts +1 -0
- package/lib/metrics/metrics/lodestar.d.ts.map +1 -1
- package/lib/metrics/metrics/lodestar.js +4 -0
- package/lib/metrics/metrics/lodestar.js.map +1 -1
- package/lib/network/processor/gossipHandlers.js +4 -6
- package/lib/network/processor/gossipHandlers.js.map +1 -1
- package/lib/network/reqresp/handlers/beaconBlocksByRange.d.ts.map +1 -1
- package/lib/network/reqresp/handlers/beaconBlocksByRange.js +14 -6
- package/lib/network/reqresp/handlers/beaconBlocksByRange.js.map +1 -1
- package/lib/network/reqresp/handlers/blobSidecarsByRange.d.ts.map +1 -1
- package/lib/network/reqresp/handlers/blobSidecarsByRange.js +11 -5
- package/lib/network/reqresp/handlers/blobSidecarsByRange.js.map +1 -1
- package/lib/network/reqresp/handlers/dataColumnSidecarsByRange.d.ts.map +1 -1
- package/lib/network/reqresp/handlers/dataColumnSidecarsByRange.js +17 -5
- package/lib/network/reqresp/handlers/dataColumnSidecarsByRange.js.map +1 -1
- package/lib/network/reqresp/handlers/executionPayloadEnvelopesByRange.d.ts.map +1 -1
- package/lib/network/reqresp/handlers/executionPayloadEnvelopesByRange.js +7 -4
- package/lib/network/reqresp/handlers/executionPayloadEnvelopesByRange.js.map +1 -1
- package/lib/node/notifier.js +7 -1
- package/lib/node/notifier.js.map +1 -1
- package/lib/sync/range/batch.d.ts +12 -2
- package/lib/sync/range/batch.d.ts.map +1 -1
- package/lib/sync/range/batch.js +56 -30
- package/lib/sync/range/batch.js.map +1 -1
- package/lib/sync/range/chain.d.ts +6 -2
- package/lib/sync/range/chain.d.ts.map +1 -1
- package/lib/sync/range/chain.js +4 -3
- package/lib/sync/range/chain.js.map +1 -1
- package/lib/sync/range/range.d.ts.map +1 -1
- package/lib/sync/range/range.js +17 -6
- package/lib/sync/range/range.js.map +1 -1
- package/lib/sync/types.d.ts +34 -0
- package/lib/sync/types.d.ts.map +1 -1
- package/lib/sync/types.js +34 -0
- package/lib/sync/types.js.map +1 -1
- package/lib/sync/unknownBlock.d.ts +24 -1
- package/lib/sync/unknownBlock.d.ts.map +1 -1
- package/lib/sync/unknownBlock.js +649 -53
- package/lib/sync/unknownBlock.js.map +1 -1
- package/lib/sync/utils/downloadByRange.d.ts +46 -10
- package/lib/sync/utils/downloadByRange.d.ts.map +1 -1
- package/lib/sync/utils/downloadByRange.js +147 -24
- package/lib/sync/utils/downloadByRange.js.map +1 -1
- package/lib/sync/utils/downloadByRoot.d.ts.map +1 -1
- package/lib/sync/utils/downloadByRoot.js +6 -2
- package/lib/sync/utils/downloadByRoot.js.map +1 -1
- package/lib/sync/utils/pendingBlocksTree.d.ts +0 -1
- package/lib/sync/utils/pendingBlocksTree.d.ts.map +1 -1
- package/lib/sync/utils/pendingBlocksTree.js +0 -9
- package/lib/sync/utils/pendingBlocksTree.js.map +1 -1
- package/package.json +16 -15
- package/src/api/impl/beacon/blocks/index.ts +5 -2
- package/src/api/impl/lodestar/index.ts +1 -1
- package/src/chain/blocks/importBlock.ts +4 -2
- package/src/chain/blocks/importExecutionPayload.ts +92 -97
- package/src/chain/blocks/index.ts +44 -13
- package/src/chain/blocks/payloadEnvelopeProcessor.ts +2 -2
- package/src/chain/blocks/types.ts +14 -25
- package/src/chain/blocks/utils/chainSegment.ts +106 -17
- package/src/chain/blocks/verifyBlock.ts +35 -6
- package/src/chain/blocks/verifyBlocksSanityChecks.ts +16 -7
- package/src/chain/blocks/verifyExecutionPayloadEnvelope.ts +129 -0
- package/src/chain/blocks/writePayloadEnvelopeInputToDb.ts +9 -18
- package/src/chain/chain.ts +23 -3
- package/src/chain/errors/blockError.ts +4 -1
- package/src/chain/errors/executionPayloadBid.ts +6 -0
- package/src/chain/errors/executionPayloadEnvelope.ts +6 -0
- package/src/chain/forkChoice/index.ts +2 -2
- package/src/chain/interface.ts +7 -1
- package/src/chain/prepareNextSlot.ts +42 -12
- package/src/chain/produceBlock/produceBlockBody.ts +37 -11
- package/src/chain/seenCache/seenPayloadEnvelopeInput.ts +22 -20
- package/src/chain/validation/block.ts +1 -0
- package/src/chain/validation/executionPayloadBid.ts +14 -0
- package/src/chain/validation/executionPayloadEnvelope.ts +12 -2
- package/src/metrics/metrics/lodestar.ts +4 -0
- package/src/network/processor/gossipHandlers.ts +6 -6
- package/src/network/reqresp/handlers/beaconBlocksByRange.ts +14 -6
- package/src/network/reqresp/handlers/blobSidecarsByRange.ts +11 -5
- package/src/network/reqresp/handlers/dataColumnSidecarsByRange.ts +17 -5
- package/src/network/reqresp/handlers/executionPayloadEnvelopesByRange.ts +7 -4
- package/src/node/notifier.ts +8 -1
- package/src/sync/range/batch.ts +90 -35
- package/src/sync/range/chain.ts +13 -5
- package/src/sync/range/range.ts +18 -6
- package/src/sync/types.ts +72 -0
- package/src/sync/unknownBlock.ts +810 -57
- package/src/sync/utils/downloadByRange.ts +256 -39
- package/src/sync/utils/downloadByRoot.ts +12 -2
- package/src/sync/utils/pendingBlocksTree.ts +0 -15
package/lib/sync/unknownBlock.js
CHANGED
|
@@ -1,19 +1,24 @@
|
|
|
1
|
+
import { routes } from "@lodestar/api";
|
|
1
2
|
import { ForkSeq } from "@lodestar/params";
|
|
2
3
|
import { RequestError, RequestErrorCode } from "@lodestar/reqresp";
|
|
3
4
|
import { computeTimeAtSlot } from "@lodestar/state-transition";
|
|
4
|
-
import { prettyPrintIndices, pruneSetToMax, sleep } from "@lodestar/utils";
|
|
5
|
+
import { fromHex, prettyPrintIndices, pruneSetToMax, sleep, toRootHex } from "@lodestar/utils";
|
|
5
6
|
import { isBlockInputBlobs, isBlockInputColumns } from "../chain/blocks/blockInput/blockInput.js";
|
|
6
7
|
import { BlockInputSource } from "../chain/blocks/blockInput/types.js";
|
|
8
|
+
import { PayloadError, PayloadErrorCode } from "../chain/blocks/importExecutionPayload.js";
|
|
9
|
+
import { PayloadEnvelopeInputSource } from "../chain/blocks/payloadEnvelopeInput/index.js";
|
|
7
10
|
import { BlockError, BlockErrorCode } from "../chain/errors/index.js";
|
|
8
11
|
import { ChainEvent } from "../chain/index.js";
|
|
12
|
+
import { validateGloasBlockDataColumnSidecars } from "../chain/validation/dataColumnSidecar.js";
|
|
13
|
+
import { validateGossipExecutionPayloadEnvelope } from "../chain/validation/executionPayloadEnvelope.js";
|
|
9
14
|
import { NetworkEvent, prettyPrintPeerIdStr } from "../network/index.js";
|
|
10
15
|
import { shuffle } from "../util/shuffle.js";
|
|
11
16
|
import { sortBy } from "../util/sortBy.js";
|
|
12
17
|
import { wrapError } from "../util/wrapError.js";
|
|
13
18
|
import { MAX_CONCURRENT_REQUESTS } from "./constants.js";
|
|
14
|
-
import { PendingBlockInputStatus, PendingBlockType, getBlockInputSyncCacheItemRootHex, getBlockInputSyncCacheItemSlot, isPendingBlockInput, } from "./types.js";
|
|
19
|
+
import { PendingBlockInputStatus, PendingBlockType, PendingPayloadInputStatus, getBlockInputSyncCacheItemRootHex, getBlockInputSyncCacheItemSlot, getPayloadSyncCacheItemRootHex, getPayloadSyncCacheItemSlot, isPendingBlockInput, isPendingPayloadEnvelope, isPendingPayloadInput, } from "./types.js";
|
|
15
20
|
import { DownloadByRootError, downloadByRoot } from "./utils/downloadByRoot.js";
|
|
16
|
-
import { getAllDescendantBlocks,
|
|
21
|
+
import { getAllDescendantBlocks, getUnknownAndAncestorBlocks } from "./utils/pendingBlocksTree.js";
|
|
17
22
|
const MAX_ATTEMPTS_PER_BLOCK = 5;
|
|
18
23
|
const MAX_KNOWN_BAD_BLOCKS = 500;
|
|
19
24
|
const MAX_PENDING_BLOCKS = 100;
|
|
@@ -68,6 +73,8 @@ export class BlockInputSync {
|
|
|
68
73
|
* block RootHex -> PendingBlock. To avoid finding same root at the same time
|
|
69
74
|
*/
|
|
70
75
|
pendingBlocks = new Map();
|
|
76
|
+
// Payload sync is keyed by beacon block root as well, so block and payload queues can unblock each other.
|
|
77
|
+
pendingPayloads = new Map();
|
|
71
78
|
knownBadBlocks = new Set();
|
|
72
79
|
maxPendingBlocks;
|
|
73
80
|
subscribedToNetworkEvents = false;
|
|
@@ -83,6 +90,7 @@ export class BlockInputSync {
|
|
|
83
90
|
this.peerBalancer = new UnknownBlockPeerBalancer();
|
|
84
91
|
if (metrics) {
|
|
85
92
|
metrics.blockInputSync.pendingBlocks.addCollect(() => metrics.blockInputSync.pendingBlocks.set(this.pendingBlocks.size));
|
|
93
|
+
metrics.blockInputSync.pendingPayloads.addCollect(() => metrics.blockInputSync.pendingPayloads.set(this.pendingPayloads.size));
|
|
86
94
|
metrics.blockInputSync.knownBadBlocks.addCollect(() => metrics.blockInputSync.knownBadBlocks.set(this.knownBadBlocks.size));
|
|
87
95
|
}
|
|
88
96
|
}
|
|
@@ -95,8 +103,13 @@ export class BlockInputSync {
|
|
|
95
103
|
if (!this.subscribedToNetworkEvents) {
|
|
96
104
|
this.logger.verbose("BlockInputSync enabled.");
|
|
97
105
|
this.chain.emitter.on(ChainEvent.unknownBlockRoot, this.onUnknownBlockRoot);
|
|
106
|
+
this.chain.emitter.on(ChainEvent.unknownEnvelopeBlockRoot, this.onUnknownEnvelopeBlockRoot);
|
|
98
107
|
this.chain.emitter.on(ChainEvent.incompleteBlockInput, this.onIncompleteBlockInput);
|
|
108
|
+
this.chain.emitter.on(ChainEvent.incompletePayloadEnvelope, this.onIncompletePayloadEnvelope);
|
|
99
109
|
this.chain.emitter.on(ChainEvent.blockUnknownParent, this.onUnknownParent);
|
|
110
|
+
this.chain.emitter.on(ChainEvent.envelopeUnknownBlock, this.onEnvelopeUnknownBlock);
|
|
111
|
+
this.chain.emitter.on(routes.events.EventType.block, this.onBlockImported);
|
|
112
|
+
this.chain.emitter.on(routes.events.EventType.executionPayload, this.onPayloadImported);
|
|
100
113
|
this.network.events.on(NetworkEvent.peerConnected, this.onPeerConnected);
|
|
101
114
|
this.network.events.on(NetworkEvent.peerDisconnected, this.onPeerDisconnected);
|
|
102
115
|
this.subscribedToNetworkEvents = true;
|
|
@@ -105,8 +118,13 @@ export class BlockInputSync {
|
|
|
105
118
|
unsubscribeFromNetwork() {
|
|
106
119
|
this.logger.verbose("BlockInputSync disabled.");
|
|
107
120
|
this.chain.emitter.off(ChainEvent.unknownBlockRoot, this.onUnknownBlockRoot);
|
|
121
|
+
this.chain.emitter.off(ChainEvent.unknownEnvelopeBlockRoot, this.onUnknownEnvelopeBlockRoot);
|
|
108
122
|
this.chain.emitter.off(ChainEvent.incompleteBlockInput, this.onIncompleteBlockInput);
|
|
123
|
+
this.chain.emitter.off(ChainEvent.incompletePayloadEnvelope, this.onIncompletePayloadEnvelope);
|
|
109
124
|
this.chain.emitter.off(ChainEvent.blockUnknownParent, this.onUnknownParent);
|
|
125
|
+
this.chain.emitter.off(ChainEvent.envelopeUnknownBlock, this.onEnvelopeUnknownBlock);
|
|
126
|
+
this.chain.emitter.off(routes.events.EventType.block, this.onBlockImported);
|
|
127
|
+
this.chain.emitter.off(routes.events.EventType.executionPayload, this.onPayloadImported);
|
|
110
128
|
this.network.events.off(NetworkEvent.peerConnected, this.onPeerConnected);
|
|
111
129
|
this.network.events.off(NetworkEvent.peerDisconnected, this.onPeerDisconnected);
|
|
112
130
|
this.subscribedToNetworkEvents = false;
|
|
@@ -145,12 +163,54 @@ export class BlockInputSync {
|
|
|
145
163
|
this.logger.debug("Error handling incompleteBlockInput event", {}, e);
|
|
146
164
|
}
|
|
147
165
|
};
|
|
166
|
+
onUnknownEnvelopeBlockRoot = (data) => {
|
|
167
|
+
try {
|
|
168
|
+
this.addByPayloadRootHex(data.rootHex, data.peer);
|
|
169
|
+
this.triggerUnknownBlockSearch();
|
|
170
|
+
this.metrics?.blockInputSync.requests.inc({ type: PendingBlockType.UNKNOWN_DATA });
|
|
171
|
+
this.metrics?.blockInputSync.source.inc({ source: data.source });
|
|
172
|
+
}
|
|
173
|
+
catch (e) {
|
|
174
|
+
this.logger.debug("Error handling unknownEnvelopeBlockRoot event", {}, e);
|
|
175
|
+
}
|
|
176
|
+
};
|
|
177
|
+
onIncompletePayloadEnvelope = (data) => {
|
|
178
|
+
try {
|
|
179
|
+
this.addByPayloadInput(data.payloadInput, data.peer);
|
|
180
|
+
this.triggerUnknownBlockSearch();
|
|
181
|
+
this.metrics?.blockInputSync.requests.inc({ type: PendingBlockType.UNKNOWN_DATA });
|
|
182
|
+
this.metrics?.blockInputSync.source.inc({ source: data.source });
|
|
183
|
+
}
|
|
184
|
+
catch (e) {
|
|
185
|
+
this.logger.debug("Error handling incompletePayloadEnvelope event", {}, e);
|
|
186
|
+
}
|
|
187
|
+
};
|
|
148
188
|
/**
|
|
149
189
|
* Process an unknownBlockParent event and register the block in `pendingBlocks` Map.
|
|
150
190
|
*/
|
|
151
191
|
onUnknownParent = (data) => {
|
|
152
192
|
try {
|
|
153
|
-
this.
|
|
193
|
+
const missingDependency = this.getMissingBlockDependency(data.blockInput);
|
|
194
|
+
if (missingDependency.kind === "invalidParentPayload") {
|
|
195
|
+
this.addByBlockInput(data.blockInput, data.peer);
|
|
196
|
+
const pendingBlock = this.pendingBlocks.get(data.blockInput.blockRootHex);
|
|
197
|
+
if (pendingBlock && isPendingBlockInput(pendingBlock)) {
|
|
198
|
+
this.logger.debug("Ignoring block with conflicting parent payload hash", {
|
|
199
|
+
slot: pendingBlock.blockInput.slot,
|
|
200
|
+
root: pendingBlock.blockInput.blockRootHex,
|
|
201
|
+
parentRoot: missingDependency.parentRootHex,
|
|
202
|
+
parentBlockHash: missingDependency.parentBlockHashHex,
|
|
203
|
+
});
|
|
204
|
+
this.removeAndDownScoreAllDescendants(pendingBlock);
|
|
205
|
+
}
|
|
206
|
+
return;
|
|
207
|
+
}
|
|
208
|
+
if (missingDependency.kind === "parentPayload") {
|
|
209
|
+
this.addByPayloadRootHex(missingDependency.rootHex, data.peer);
|
|
210
|
+
}
|
|
211
|
+
else if (missingDependency.kind === "parentBlock") {
|
|
212
|
+
this.addByRootHex(missingDependency.rootHex, data.peer);
|
|
213
|
+
}
|
|
154
214
|
this.addByBlockInput(data.blockInput, data.peer);
|
|
155
215
|
this.triggerUnknownBlockSearch();
|
|
156
216
|
this.metrics?.blockInputSync.requests.inc({ type: PendingBlockType.UNKNOWN_PARENT });
|
|
@@ -160,8 +220,31 @@ export class BlockInputSync {
|
|
|
160
220
|
this.logger.debug("Error handling unknownParent event", {}, e);
|
|
161
221
|
}
|
|
162
222
|
};
|
|
223
|
+
onEnvelopeUnknownBlock = (data) => {
|
|
224
|
+
try {
|
|
225
|
+
const blockRootHex = toRootHex(data.envelope.message.beaconBlockRoot);
|
|
226
|
+
this.addByRootHex(blockRootHex, data.peer);
|
|
227
|
+
this.addByPayloadEnvelope(data.envelope, data.peer);
|
|
228
|
+
this.triggerUnknownBlockSearch();
|
|
229
|
+
this.metrics?.blockInputSync.requests.inc({ type: PendingBlockType.UNKNOWN_DATA });
|
|
230
|
+
this.metrics?.blockInputSync.source.inc({ source: data.source });
|
|
231
|
+
}
|
|
232
|
+
catch (e) {
|
|
233
|
+
this.logger.debug("Error handling envelopeUnknownBlock event", {}, e);
|
|
234
|
+
}
|
|
235
|
+
};
|
|
236
|
+
onBlockImported = () => {
|
|
237
|
+
if (this.pendingPayloads.size > 0) {
|
|
238
|
+
this.triggerUnknownBlockSearch();
|
|
239
|
+
}
|
|
240
|
+
};
|
|
241
|
+
onPayloadImported = ({ blockRoot, }) => {
|
|
242
|
+
this.pendingPayloads.delete(blockRoot);
|
|
243
|
+
this.triggerUnknownBlockSearch();
|
|
244
|
+
};
|
|
163
245
|
addByRootHex = (rootHex, peerIdStr) => {
|
|
164
246
|
let pendingBlock = this.pendingBlocks.get(rootHex);
|
|
247
|
+
let added = false;
|
|
165
248
|
if (!pendingBlock) {
|
|
166
249
|
pendingBlock = {
|
|
167
250
|
status: PendingBlockInputStatus.pending,
|
|
@@ -170,6 +253,7 @@ export class BlockInputSync {
|
|
|
170
253
|
timeAddedSec: Date.now() / 1000,
|
|
171
254
|
};
|
|
172
255
|
this.pendingBlocks.set(rootHex, pendingBlock);
|
|
256
|
+
added = true;
|
|
173
257
|
this.logger.verbose("Added new rootHex to BlockInputSync.pendingBlocks", {
|
|
174
258
|
root: pendingBlock.rootHex,
|
|
175
259
|
peerIdStr: peerIdStr ?? "unknown peer",
|
|
@@ -184,6 +268,7 @@ export class BlockInputSync {
|
|
|
184
268
|
if (prunedItemCount > 0) {
|
|
185
269
|
this.logger.verbose(`Pruned ${prunedItemCount} items from BlockInputSync.pendingBlocks`);
|
|
186
270
|
}
|
|
271
|
+
return added;
|
|
187
272
|
};
|
|
188
273
|
addByBlockInput = (blockInput, peerIdStr) => {
|
|
189
274
|
let pendingBlock = this.pendingBlocks.get(blockInput.blockRootHex);
|
|
@@ -211,6 +296,75 @@ export class BlockInputSync {
|
|
|
211
296
|
this.logger.verbose(`Pruned ${prunedItemCount} items from BlockInputSync.pendingBlocks`);
|
|
212
297
|
}
|
|
213
298
|
};
|
|
299
|
+
addByPayloadRootHex = (rootHex, peerIdStr) => {
|
|
300
|
+
let pendingPayload = this.pendingPayloads.get(rootHex);
|
|
301
|
+
let added = false;
|
|
302
|
+
if (!pendingPayload) {
|
|
303
|
+
pendingPayload = {
|
|
304
|
+
status: PendingPayloadInputStatus.pending,
|
|
305
|
+
rootHex,
|
|
306
|
+
peerIdStrings: new Set(),
|
|
307
|
+
timeAddedSec: Date.now() / 1000,
|
|
308
|
+
};
|
|
309
|
+
this.pendingPayloads.set(rootHex, pendingPayload);
|
|
310
|
+
added = true;
|
|
311
|
+
this.logger.verbose("Added new payload rootHex to BlockInputSync.pendingPayloads", {
|
|
312
|
+
root: rootHex,
|
|
313
|
+
peerIdStr: peerIdStr ?? "unknown peer",
|
|
314
|
+
});
|
|
315
|
+
}
|
|
316
|
+
if (peerIdStr) {
|
|
317
|
+
pendingPayload.peerIdStrings.add(peerIdStr);
|
|
318
|
+
}
|
|
319
|
+
const prunedItemCount = pruneSetToMax(this.pendingPayloads, this.maxPendingBlocks);
|
|
320
|
+
if (prunedItemCount > 0) {
|
|
321
|
+
this.logger.verbose(`Pruned ${prunedItemCount} items from BlockInputSync.pendingPayloads`);
|
|
322
|
+
}
|
|
323
|
+
return added;
|
|
324
|
+
};
|
|
325
|
+
addByPayloadEnvelope = (envelope, peerIdStr) => {
|
|
326
|
+
const rootHex = toRootHex(envelope.message.beaconBlockRoot);
|
|
327
|
+
const existingPendingPayload = this.pendingPayloads.get(rootHex);
|
|
328
|
+
let pendingPayload = this.pendingPayloads.get(rootHex);
|
|
329
|
+
if (!pendingPayload || !isPendingPayloadEnvelope(pendingPayload)) {
|
|
330
|
+
pendingPayload = {
|
|
331
|
+
status: PendingPayloadInputStatus.waitingForBlock,
|
|
332
|
+
envelope,
|
|
333
|
+
peerIdStrings: new Set(existingPendingPayload?.peerIdStrings ?? []),
|
|
334
|
+
timeAddedSec: existingPendingPayload?.timeAddedSec ?? Date.now() / 1000,
|
|
335
|
+
};
|
|
336
|
+
this.pendingPayloads.set(rootHex, pendingPayload);
|
|
337
|
+
this.logger.verbose("Added payload envelope to BlockInputSync.pendingPayloads", {
|
|
338
|
+
slot: envelope.message.payload.slotNumber,
|
|
339
|
+
root: rootHex,
|
|
340
|
+
});
|
|
341
|
+
}
|
|
342
|
+
else {
|
|
343
|
+
this.logger.debug("Overwriting pending payload envelope for root already waiting for block", {
|
|
344
|
+
slot: envelope.message.payload.slotNumber,
|
|
345
|
+
root: rootHex,
|
|
346
|
+
});
|
|
347
|
+
pendingPayload.envelope = envelope;
|
|
348
|
+
}
|
|
349
|
+
if (peerIdStr) {
|
|
350
|
+
pendingPayload.peerIdStrings.add(peerIdStr);
|
|
351
|
+
}
|
|
352
|
+
const prunedItemCount = pruneSetToMax(this.pendingPayloads, this.maxPendingBlocks);
|
|
353
|
+
if (prunedItemCount > 0) {
|
|
354
|
+
this.logger.verbose(`Pruned ${prunedItemCount} items from BlockInputSync.pendingPayloads`);
|
|
355
|
+
}
|
|
356
|
+
};
|
|
357
|
+
addByPayloadInput = (payloadInput, peerIdStr, envelope) => {
|
|
358
|
+
const pendingPayload = this.toPendingPayloadInput(payloadInput, this.pendingPayloads.get(payloadInput.blockRootHex), envelope);
|
|
359
|
+
if (peerIdStr) {
|
|
360
|
+
pendingPayload.peerIdStrings.add(peerIdStr);
|
|
361
|
+
}
|
|
362
|
+
this.pendingPayloads.set(payloadInput.blockRootHex, pendingPayload);
|
|
363
|
+
const prunedItemCount = pruneSetToMax(this.pendingPayloads, this.maxPendingBlocks);
|
|
364
|
+
if (prunedItemCount > 0) {
|
|
365
|
+
this.logger.verbose(`Pruned ${prunedItemCount} items from BlockInputSync.pendingPayloads`);
|
|
366
|
+
}
|
|
367
|
+
};
|
|
214
368
|
onPeerConnected = (data) => {
|
|
215
369
|
try {
|
|
216
370
|
const peerId = data.peer;
|
|
@@ -226,47 +380,172 @@ export class BlockInputSync {
|
|
|
226
380
|
const peerId = data.peer;
|
|
227
381
|
this.peerBalancer.onPeerDisconnected(peerId);
|
|
228
382
|
};
|
|
383
|
+
/**
|
|
384
|
+
* Post-gloas, a locally complete block can still be blocked on its parent's execution payload lineage.
|
|
385
|
+
* Distinguish which dependency is missing so the scheduler can enqueue the right follow-up work.
|
|
386
|
+
*/
|
|
387
|
+
getMissingBlockDependency(blockInput) {
|
|
388
|
+
const parentRootHex = blockInput.parentRootHex;
|
|
389
|
+
if (!this.chain.forkChoice.hasBlockHex(parentRootHex)) {
|
|
390
|
+
return { kind: "parentBlock", rootHex: parentRootHex };
|
|
391
|
+
}
|
|
392
|
+
if (!blockInput.hasBlock()) {
|
|
393
|
+
return { kind: "block", rootHex: blockInput.blockRootHex };
|
|
394
|
+
}
|
|
395
|
+
if (this.config.getForkSeq(blockInput.slot) < ForkSeq.gloas) {
|
|
396
|
+
return { kind: "ready" };
|
|
397
|
+
}
|
|
398
|
+
const block = blockInput.getBlock();
|
|
399
|
+
const parentBlockHashHex = toRootHex(block.message.body.signedExecutionPayloadBid.message.parentBlockHash);
|
|
400
|
+
if (this.chain.forkChoice.getBlockHexAndBlockHash(parentRootHex, parentBlockHashHex) !== null) {
|
|
401
|
+
return { kind: "ready" };
|
|
402
|
+
}
|
|
403
|
+
if (this.chain.forkChoice.hasPayloadHexUnsafe(parentRootHex)) {
|
|
404
|
+
return { kind: "invalidParentPayload", parentRootHex, parentBlockHashHex };
|
|
405
|
+
}
|
|
406
|
+
const parentPayloadInput = this.chain.seenPayloadEnvelopeInputCache.get(parentRootHex);
|
|
407
|
+
if (parentPayloadInput) {
|
|
408
|
+
if (parentPayloadInput.getBlockHashHex() === parentBlockHashHex) {
|
|
409
|
+
return { kind: "parentPayload", rootHex: parentRootHex };
|
|
410
|
+
}
|
|
411
|
+
return { kind: "invalidParentPayload", parentRootHex, parentBlockHashHex };
|
|
412
|
+
}
|
|
413
|
+
return { kind: "parentPayload", rootHex: parentRootHex };
|
|
414
|
+
}
|
|
415
|
+
advancePendingBlock(pendingBlock) {
|
|
416
|
+
const missingDependency = this.getMissingBlockDependency(pendingBlock.blockInput);
|
|
417
|
+
switch (missingDependency.kind) {
|
|
418
|
+
case "ready":
|
|
419
|
+
return "ready";
|
|
420
|
+
case "block":
|
|
421
|
+
pendingBlock.status = PendingBlockInputStatus.pending;
|
|
422
|
+
return "queued_block";
|
|
423
|
+
case "parentBlock": {
|
|
424
|
+
let added = this.addByRootHex(missingDependency.rootHex);
|
|
425
|
+
for (const peerIdStr of pendingBlock.peerIdStrings) {
|
|
426
|
+
added = this.addByRootHex(missingDependency.rootHex, peerIdStr) || added;
|
|
427
|
+
}
|
|
428
|
+
return added ? "queued_parent_block" : "blocked";
|
|
429
|
+
}
|
|
430
|
+
case "parentPayload": {
|
|
431
|
+
let added = this.addByPayloadRootHex(missingDependency.rootHex);
|
|
432
|
+
for (const peerIdStr of pendingBlock.peerIdStrings) {
|
|
433
|
+
added = this.addByPayloadRootHex(missingDependency.rootHex, peerIdStr) || added;
|
|
434
|
+
}
|
|
435
|
+
return added ? "queued_parent_payload" : "blocked";
|
|
436
|
+
}
|
|
437
|
+
case "invalidParentPayload":
|
|
438
|
+
this.logger.debug("Removing block with conflicting parent payload hash", {
|
|
439
|
+
slot: pendingBlock.blockInput.slot,
|
|
440
|
+
root: pendingBlock.blockInput.blockRootHex,
|
|
441
|
+
parentRoot: missingDependency.parentRootHex,
|
|
442
|
+
parentBlockHash: missingDependency.parentBlockHashHex,
|
|
443
|
+
});
|
|
444
|
+
this.removeAndDownScoreAllDescendants(pendingBlock);
|
|
445
|
+
return "removed";
|
|
446
|
+
}
|
|
447
|
+
}
|
|
448
|
+
toPendingPayloadInput(payloadInput, previous, envelope) {
|
|
449
|
+
// Normalize every payload queueing path into the same cache shape while preserving first-seen
|
|
450
|
+
// timing and peer provenance from any earlier by-root or envelope-only entry.
|
|
451
|
+
const queuedEnvelope = envelope ?? (previous && isPendingPayloadEnvelope(previous) ? previous.envelope : undefined);
|
|
452
|
+
if (queuedEnvelope && !payloadInput.hasPayloadEnvelope()) {
|
|
453
|
+
payloadInput.addPayloadEnvelope({
|
|
454
|
+
envelope: queuedEnvelope,
|
|
455
|
+
source: PayloadEnvelopeInputSource.byRoot,
|
|
456
|
+
seenTimestampSec: Date.now() / 1000,
|
|
457
|
+
});
|
|
458
|
+
}
|
|
459
|
+
return {
|
|
460
|
+
status: payloadInput.isComplete() ? PendingPayloadInputStatus.downloaded : PendingPayloadInputStatus.pending,
|
|
461
|
+
payloadInput,
|
|
462
|
+
timeAddedSec: previous?.timeAddedSec ?? Date.now() / 1000,
|
|
463
|
+
timeSyncedSec: payloadInput.isComplete() ? Date.now() / 1000 : undefined,
|
|
464
|
+
peerIdStrings: new Set(previous?.peerIdStrings ?? []),
|
|
465
|
+
};
|
|
466
|
+
}
|
|
229
467
|
/**
|
|
230
468
|
* Gather tip parent blocks with unknown parent and do a search for all of them
|
|
231
469
|
*/
|
|
232
470
|
triggerUnknownBlockSearch = () => {
|
|
233
471
|
// Cheap early stop to prevent calling the network.getConnectedPeers()
|
|
234
|
-
if (this.pendingBlocks.size === 0) {
|
|
472
|
+
if (this.pendingBlocks.size === 0 && this.pendingPayloads.size === 0) {
|
|
235
473
|
return;
|
|
236
474
|
}
|
|
237
|
-
// If the node loses all peers with pending unknown blocks, the sync will stall
|
|
475
|
+
// If the node loses all peers with pending unknown blocks or payloads, the sync will stall
|
|
238
476
|
const connectedPeers = this.network.getConnectedPeers();
|
|
239
|
-
|
|
240
|
-
this.logger.debug("No connected peers, skipping unknown block search.");
|
|
241
|
-
return;
|
|
242
|
-
}
|
|
477
|
+
const hasConnectedPeers = connectedPeers.length > 0;
|
|
243
478
|
const { unknowns, ancestors } = getUnknownAndAncestorBlocks(this.pendingBlocks);
|
|
244
|
-
|
|
245
|
-
|
|
246
|
-
|
|
247
|
-
|
|
248
|
-
|
|
249
|
-
|
|
250
|
-
if (this.chain.forkChoice.hasBlockHex(block.blockInput.parentRootHex)) {
|
|
479
|
+
let processedBlocks = 0;
|
|
480
|
+
let shouldRerunBlockSearch = false;
|
|
481
|
+
for (const block of ancestors) {
|
|
482
|
+
const advanceResult = this.advancePendingBlock(block);
|
|
483
|
+
switch (advanceResult) {
|
|
484
|
+
case "ready":
|
|
251
485
|
processedBlocks++;
|
|
252
|
-
this.
|
|
486
|
+
this.processReadyBlock(block).catch((e) => {
|
|
253
487
|
this.logger.debug("Unexpected error - process old downloaded block", {}, e);
|
|
254
488
|
});
|
|
489
|
+
break;
|
|
490
|
+
case "queued_block":
|
|
491
|
+
case "queued_parent_block":
|
|
492
|
+
shouldRerunBlockSearch = true;
|
|
493
|
+
break;
|
|
494
|
+
case "queued_parent_payload":
|
|
495
|
+
case "blocked":
|
|
496
|
+
case "removed":
|
|
497
|
+
break;
|
|
498
|
+
}
|
|
499
|
+
}
|
|
500
|
+
if (unknowns.length > 0) {
|
|
501
|
+
if (!hasConnectedPeers) {
|
|
502
|
+
this.logger.debug("No connected peers, skipping unknown block download.");
|
|
503
|
+
}
|
|
504
|
+
else {
|
|
505
|
+
// Most of the time there is exactly 1 unknown block
|
|
506
|
+
for (const block of unknowns) {
|
|
507
|
+
this.downloadBlock(block).catch((e) => {
|
|
508
|
+
this.logger.debug("Unexpected error - downloadBlock", { root: getBlockInputSyncCacheItemRootHex(block) }, e);
|
|
509
|
+
});
|
|
255
510
|
}
|
|
256
511
|
}
|
|
512
|
+
}
|
|
513
|
+
else if (ancestors.length > 0) {
|
|
514
|
+
// It's rare when there is no unknown block
|
|
515
|
+
// see https://github.com/ChainSafe/lodestar/issues/5649#issuecomment-1594213550
|
|
257
516
|
this.logger.verbose("No unknown block, process ancestor downloaded blocks", {
|
|
258
517
|
pendingBlocks: this.pendingBlocks.size,
|
|
259
518
|
ancestorBlocks: ancestors.length,
|
|
260
519
|
processedBlocks,
|
|
261
520
|
});
|
|
262
|
-
return;
|
|
263
521
|
}
|
|
264
|
-
//
|
|
265
|
-
for (const
|
|
266
|
-
|
|
267
|
-
this.
|
|
522
|
+
// Blocks can unblock payloads and payloads can unblock blocks, so every scheduler pass services both queues.
|
|
523
|
+
for (const payload of Array.from(this.pendingPayloads.values())) {
|
|
524
|
+
if (isPendingPayloadInput(payload) && payload.status === PendingPayloadInputStatus.downloaded) {
|
|
525
|
+
this.processPayload(payload).catch((e) => {
|
|
526
|
+
this.logger.debug("Unexpected error - process downloaded payload", {}, e);
|
|
527
|
+
});
|
|
528
|
+
continue;
|
|
529
|
+
}
|
|
530
|
+
if (isPendingPayloadEnvelope(payload)) {
|
|
531
|
+
this.reconcilePayloadEnvelope(payload).catch((e) => {
|
|
532
|
+
this.logger.debug("Unexpected error - reconcile pending payload envelope", {}, e);
|
|
533
|
+
});
|
|
534
|
+
continue;
|
|
535
|
+
}
|
|
536
|
+
if (!hasConnectedPeers) {
|
|
537
|
+
this.logger.debug("No connected peers, skipping unknown payload download.", {
|
|
538
|
+
root: getPayloadSyncCacheItemRootHex(payload),
|
|
539
|
+
});
|
|
540
|
+
continue;
|
|
541
|
+
}
|
|
542
|
+
this.downloadPayload(payload).catch((e) => {
|
|
543
|
+
this.logger.debug("Unexpected error - downloadPayload", { root: getPayloadSyncCacheItemRootHex(payload) }, e);
|
|
268
544
|
});
|
|
269
545
|
}
|
|
546
|
+
if (shouldRerunBlockSearch) {
|
|
547
|
+
this.triggerUnknownBlockSearch();
|
|
548
|
+
}
|
|
270
549
|
};
|
|
271
550
|
async downloadBlock(block) {
|
|
272
551
|
if (block.status !== PendingBlockInputStatus.pending) {
|
|
@@ -297,10 +576,24 @@ export class BlockInputSync {
|
|
|
297
576
|
};
|
|
298
577
|
this.logger.verbose("Downloaded unknown block", logCtx2);
|
|
299
578
|
if (parentInForkChoice) {
|
|
300
|
-
//
|
|
301
|
-
|
|
302
|
-
|
|
303
|
-
|
|
579
|
+
// If the direct parent is already in fork choice, let the block state machine decide if
|
|
580
|
+
// the next step is block import, parent payload download, or branch removal.
|
|
581
|
+
const advanceResult = this.advancePendingBlock(pending);
|
|
582
|
+
switch (advanceResult) {
|
|
583
|
+
case "ready":
|
|
584
|
+
this.processReadyBlock(pending).catch((e) => {
|
|
585
|
+
this.logger.debug("Unexpected error - process newly downloaded block", logCtx2, e);
|
|
586
|
+
});
|
|
587
|
+
break;
|
|
588
|
+
case "queued_block":
|
|
589
|
+
case "queued_parent_block":
|
|
590
|
+
case "queued_parent_payload":
|
|
591
|
+
this.triggerUnknownBlockSearch();
|
|
592
|
+
break;
|
|
593
|
+
case "blocked":
|
|
594
|
+
case "removed":
|
|
595
|
+
break;
|
|
596
|
+
}
|
|
304
597
|
}
|
|
305
598
|
else if (blockSlot <= finalizedSlot) {
|
|
306
599
|
// the common ancestor of the downloading chain and canonical chain should be at least the finalized slot and
|
|
@@ -325,26 +618,11 @@ export class BlockInputSync {
|
|
|
325
618
|
}
|
|
326
619
|
}
|
|
327
620
|
/**
|
|
328
|
-
*
|
|
329
|
-
* On error, remove and downscore
|
|
330
|
-
* This function could run recursively for all descendant blocks
|
|
621
|
+
* Import a block that has already passed the local dependency checks in BlockInputSync.
|
|
622
|
+
* On error, remove and downscore descendants as appropriate for the failure type.
|
|
331
623
|
*/
|
|
332
|
-
async
|
|
333
|
-
// pending block status is `downloaded` right after `downloadBlock`
|
|
334
|
-
// but could be `pending` if added by `onUnknownBlockParent` event and this function is called recursively
|
|
624
|
+
async processReadyBlock(pendingBlock) {
|
|
335
625
|
if (pendingBlock.status !== PendingBlockInputStatus.downloaded) {
|
|
336
|
-
if (pendingBlock.status === PendingBlockInputStatus.pending) {
|
|
337
|
-
const connectedPeers = this.network.getConnectedPeers();
|
|
338
|
-
if (connectedPeers.length === 0) {
|
|
339
|
-
this.logger.debug("No connected peers, skipping download block", {
|
|
340
|
-
slot: pendingBlock.blockInput.slot,
|
|
341
|
-
blockRoot: pendingBlock.blockInput.blockRootHex,
|
|
342
|
-
});
|
|
343
|
-
return;
|
|
344
|
-
}
|
|
345
|
-
// if the download is a success we'll call `processBlock()` for this block
|
|
346
|
-
await this.downloadBlock(pendingBlock);
|
|
347
|
-
}
|
|
348
626
|
return;
|
|
349
627
|
}
|
|
350
628
|
pendingBlock.status = PendingBlockInputStatus.processing;
|
|
@@ -384,14 +662,9 @@ export class BlockInputSync {
|
|
|
384
662
|
if (!res.err) {
|
|
385
663
|
// no need to update status to "processed", delete anyway
|
|
386
664
|
this.pendingBlocks.delete(pendingBlock.blockInput.blockRootHex);
|
|
387
|
-
//
|
|
388
|
-
|
|
389
|
-
|
|
390
|
-
this.processBlock(descendantBlock).catch((e) => {
|
|
391
|
-
this.logger.debug("Unexpected error - process descendant block", {}, e);
|
|
392
|
-
});
|
|
393
|
-
}
|
|
394
|
-
}
|
|
665
|
+
// Re-enter the scheduler so descendants blocked on either parent blocks or parent payloads
|
|
666
|
+
// are advanced through the same dependency checks as every other pending item.
|
|
667
|
+
this.triggerUnknownBlockSearch();
|
|
395
668
|
}
|
|
396
669
|
else {
|
|
397
670
|
const errorData = { slot: pendingBlock.blockInput.slot, root: pendingBlock.blockInput.blockRootHex };
|
|
@@ -406,6 +679,14 @@ export class BlockInputSync {
|
|
|
406
679
|
this.logger.debug("Attempted to process block but its parent was still unknown", errorData, res.err);
|
|
407
680
|
pendingBlock.status = PendingBlockInputStatus.downloaded;
|
|
408
681
|
break;
|
|
682
|
+
case BlockErrorCode.PARENT_PAYLOAD_UNKNOWN:
|
|
683
|
+
this.logger.error("processReadyBlock() hit unexpected parent payload dependency after readiness checks", {
|
|
684
|
+
...errorData,
|
|
685
|
+
parentRoot: pendingBlock.blockInput.parentRootHex,
|
|
686
|
+
parentBlockHash: res.err.type.parentBlockHash,
|
|
687
|
+
}, res.err);
|
|
688
|
+
pendingBlock.status = PendingBlockInputStatus.downloaded;
|
|
689
|
+
break;
|
|
409
690
|
case BlockErrorCode.EXECUTION_ENGINE_ERROR:
|
|
410
691
|
// Removing the block(s) without penalizing the peers, hoping for EL to
|
|
411
692
|
// recover on a latter download + verify attempt
|
|
@@ -424,6 +705,299 @@ export class BlockInputSync {
|
|
|
424
705
|
}
|
|
425
706
|
}
|
|
426
707
|
}
|
|
708
|
+
/**
|
|
709
|
+
* Reconcile an envelope-first payload entry once the block import path has seeded its
|
|
710
|
+
* PayloadEnvelopeInput. This may queue block download, validate the speculative envelope, or
|
|
711
|
+
* downgrade back to by-root fetching when the cached envelope does not match the imported block.
|
|
712
|
+
*/
|
|
713
|
+
async reconcilePayloadEnvelope(pendingPayload) {
|
|
714
|
+
const rootHex = getPayloadSyncCacheItemRootHex(pendingPayload);
|
|
715
|
+
if (this.chain.forkChoice.hasPayloadHexUnsafe(rootHex)) {
|
|
716
|
+
this.pendingPayloads.delete(rootHex);
|
|
717
|
+
return;
|
|
718
|
+
}
|
|
719
|
+
const payloadInput = this.chain.seenPayloadEnvelopeInputCache.get(rootHex);
|
|
720
|
+
if (!payloadInput) {
|
|
721
|
+
if (!this.chain.forkChoice.hasBlockHex(rootHex)) {
|
|
722
|
+
// Column commitments live on the block body, so an envelope-only entry has to pull the block first.
|
|
723
|
+
if (!this.pendingBlocks.has(rootHex)) {
|
|
724
|
+
this.addByRootHex(rootHex);
|
|
725
|
+
}
|
|
726
|
+
const pendingBlock = this.pendingBlocks.get(rootHex);
|
|
727
|
+
if (pendingBlock && this.network.getConnectedPeers().length > 0) {
|
|
728
|
+
await this.downloadBlock(pendingBlock);
|
|
729
|
+
}
|
|
730
|
+
}
|
|
731
|
+
else {
|
|
732
|
+
this.logger.debug("Missing PayloadEnvelopeInput for known block while reconciling payload envelope", {
|
|
733
|
+
root: rootHex,
|
|
734
|
+
});
|
|
735
|
+
}
|
|
736
|
+
return;
|
|
737
|
+
}
|
|
738
|
+
if (!payloadInput.hasPayloadEnvelope()) {
|
|
739
|
+
const validationResult = await wrapError(validateGossipExecutionPayloadEnvelope(this.chain, pendingPayload.envelope));
|
|
740
|
+
if (validationResult.err) {
|
|
741
|
+
this.logger.debug("Pending payload envelope failed validation after block import, refetching by root", { slot: pendingPayload.envelope.message.payload.slotNumber, root: rootHex }, validationResult.err);
|
|
742
|
+
const pendingPayloadByRoot = {
|
|
743
|
+
status: PendingPayloadInputStatus.pending,
|
|
744
|
+
rootHex,
|
|
745
|
+
timeAddedSec: pendingPayload.timeAddedSec,
|
|
746
|
+
peerIdStrings: new Set(pendingPayload.peerIdStrings),
|
|
747
|
+
};
|
|
748
|
+
this.pendingPayloads.set(rootHex, pendingPayloadByRoot);
|
|
749
|
+
if (this.network.getConnectedPeers().length > 0) {
|
|
750
|
+
await this.downloadPayload(pendingPayloadByRoot);
|
|
751
|
+
}
|
|
752
|
+
return;
|
|
753
|
+
}
|
|
754
|
+
}
|
|
755
|
+
const upgradedPayload = this.toPendingPayloadInput(payloadInput, pendingPayload, pendingPayload.envelope);
|
|
756
|
+
this.pendingPayloads.set(rootHex, upgradedPayload);
|
|
757
|
+
if (upgradedPayload.status === PendingPayloadInputStatus.downloaded) {
|
|
758
|
+
await this.processPayload(upgradedPayload);
|
|
759
|
+
return;
|
|
760
|
+
}
|
|
761
|
+
await this.downloadPayload(upgradedPayload);
|
|
762
|
+
}
|
|
763
|
+
async downloadPayload(payload) {
|
|
764
|
+
if (isPendingPayloadEnvelope(payload)) {
|
|
765
|
+
await this.reconcilePayloadEnvelope(payload);
|
|
766
|
+
return;
|
|
767
|
+
}
|
|
768
|
+
const rootHex = getPayloadSyncCacheItemRootHex(payload);
|
|
769
|
+
if (this.chain.forkChoice.hasPayloadHexUnsafe(rootHex)) {
|
|
770
|
+
this.pendingPayloads.delete(rootHex);
|
|
771
|
+
return;
|
|
772
|
+
}
|
|
773
|
+
if (payload.status !== PendingPayloadInputStatus.pending) {
|
|
774
|
+
return;
|
|
775
|
+
}
|
|
776
|
+
const logCtx = {
|
|
777
|
+
slot: getPayloadSyncCacheItemSlot(payload),
|
|
778
|
+
root: rootHex,
|
|
779
|
+
pendingPayloads: this.pendingPayloads.size,
|
|
780
|
+
};
|
|
781
|
+
this.logger.verbose("BlockInputSync.downloadPayload()", logCtx);
|
|
782
|
+
payload.status = PendingPayloadInputStatus.fetching;
|
|
783
|
+
const res = await wrapError(this.fetchPayloadInput(payload));
|
|
784
|
+
if (!res.err) {
|
|
785
|
+
const pendingPayload = res.result;
|
|
786
|
+
this.pendingPayloads.set(getPayloadSyncCacheItemRootHex(pendingPayload), pendingPayload);
|
|
787
|
+
if (isPendingPayloadEnvelope(pendingPayload)) {
|
|
788
|
+
await this.reconcilePayloadEnvelope(pendingPayload);
|
|
789
|
+
}
|
|
790
|
+
else if (pendingPayload.status === PendingPayloadInputStatus.downloaded) {
|
|
791
|
+
await this.processPayload(pendingPayload);
|
|
792
|
+
}
|
|
793
|
+
return;
|
|
794
|
+
}
|
|
795
|
+
this.logger.debug("Ignoring unknown payload root after failed download", logCtx, res.err);
|
|
796
|
+
if (!isPendingPayloadEnvelope(payload)) {
|
|
797
|
+
payload.status = PendingPayloadInputStatus.pending;
|
|
798
|
+
}
|
|
799
|
+
}
|
|
800
|
+
async processPayload(pendingPayload) {
|
|
801
|
+
const rootHex = pendingPayload.payloadInput.blockRootHex;
|
|
802
|
+
const logCtx = { slot: pendingPayload.payloadInput.slot, root: rootHex };
|
|
803
|
+
if (pendingPayload.status !== PendingPayloadInputStatus.downloaded) {
|
|
804
|
+
this.logger.debug("Skipping payload processing before payload input is downloaded", {
|
|
805
|
+
...logCtx,
|
|
806
|
+
status: pendingPayload.status,
|
|
807
|
+
});
|
|
808
|
+
return;
|
|
809
|
+
}
|
|
810
|
+
if (this.chain.forkChoice.hasPayloadHexUnsafe(rootHex)) {
|
|
811
|
+
this.logger.debug("Payload already imported while processing unknown payload", logCtx);
|
|
812
|
+
this.pendingPayloads.delete(rootHex);
|
|
813
|
+
return;
|
|
814
|
+
}
|
|
815
|
+
if (!this.chain.forkChoice.hasBlockHex(rootHex)) {
|
|
816
|
+
this.logger.debug("Payload input is ready before its block is in fork choice", logCtx);
|
|
817
|
+
const added = this.addByRootHex(rootHex);
|
|
818
|
+
pendingPayload.status = PendingPayloadInputStatus.downloaded;
|
|
819
|
+
if (added) {
|
|
820
|
+
this.triggerUnknownBlockSearch();
|
|
821
|
+
}
|
|
822
|
+
return;
|
|
823
|
+
}
|
|
824
|
+
pendingPayload.status = PendingPayloadInputStatus.processing;
|
|
825
|
+
const res = await wrapError(this.chain.processExecutionPayload(pendingPayload.payloadInput));
|
|
826
|
+
if (!res.err) {
|
|
827
|
+
this.logger.debug("Processed payload from unknown sync", logCtx);
|
|
828
|
+
this.pendingPayloads.delete(rootHex);
|
|
829
|
+
this.triggerUnknownBlockSearch();
|
|
830
|
+
return;
|
|
831
|
+
}
|
|
832
|
+
if (res.err instanceof PayloadError) {
|
|
833
|
+
switch (res.err.type.code) {
|
|
834
|
+
case PayloadErrorCode.BLOCK_NOT_IN_FORK_CHOICE:
|
|
835
|
+
// Payload sync discovered the block dependency before the block queue did. Re-enqueue the
|
|
836
|
+
// block and keep the payload ready so the scheduler can retry once the block reaches fork choice.
|
|
837
|
+
if (this.addByRootHex(rootHex)) {
|
|
838
|
+
this.triggerUnknownBlockSearch();
|
|
839
|
+
}
|
|
840
|
+
// Keep the payload out of any synchronous requeue pass; a later scheduler pass will retry it.
|
|
841
|
+
pendingPayload.status = PendingPayloadInputStatus.downloaded;
|
|
842
|
+
break;
|
|
843
|
+
case PayloadErrorCode.EXECUTION_ENGINE_ERROR:
|
|
844
|
+
this.logger.debug("Execution engine error while processing payload from unknown sync", logCtx, res.err);
|
|
845
|
+
pendingPayload.status = PendingPayloadInputStatus.downloaded;
|
|
846
|
+
break;
|
|
847
|
+
case PayloadErrorCode.EXECUTION_ENGINE_INVALID:
|
|
848
|
+
case PayloadErrorCode.ENVELOPE_VERIFICATION_ERROR:
|
|
849
|
+
case PayloadErrorCode.INVALID_SIGNATURE:
|
|
850
|
+
// TODO GLOAS: Decide how invalid payload inputs should eventually leave memory without
|
|
851
|
+
// reintroducing envelope replacement / recreation flows.
|
|
852
|
+
this.logger.debug("Error processing payload from unknown sync", logCtx, res.err);
|
|
853
|
+
this.removePendingPayloadAndDescendants(rootHex);
|
|
854
|
+
break;
|
|
855
|
+
default:
|
|
856
|
+
this.logger.debug("Error processing payload from unknown sync", logCtx, res.err);
|
|
857
|
+
this.pendingPayloads.delete(rootHex);
|
|
858
|
+
}
|
|
859
|
+
return;
|
|
860
|
+
}
|
|
861
|
+
this.logger.debug("Unknown error processing payload from unknown sync", logCtx, res.err);
|
|
862
|
+
pendingPayload.status = PendingPayloadInputStatus.downloaded;
|
|
863
|
+
}
|
|
864
|
+
/**
|
|
865
|
+
* Download payload material keyed by beacon block root. Unlike block download, payload sync may
|
|
866
|
+
* already have a locally cached envelope or partial columns, so each attempt starts from local state
|
|
867
|
+
* and only asks peers for the remaining pieces.
|
|
868
|
+
*/
|
|
869
|
+
async fetchPayloadInput(cacheItem) {
|
|
870
|
+
const rootHex = getPayloadSyncCacheItemRootHex(cacheItem);
|
|
871
|
+
const blockRoot = fromHex(rootHex);
|
|
872
|
+
const excludedPeers = new Set();
|
|
873
|
+
let slot = getPayloadSyncCacheItemSlot(cacheItem);
|
|
874
|
+
let payloadInput = isPendingPayloadInput(cacheItem)
|
|
875
|
+
? cacheItem.payloadInput
|
|
876
|
+
: this.chain.seenPayloadEnvelopeInputCache.get(rootHex);
|
|
877
|
+
let envelope = payloadInput?.hasPayloadEnvelope() ? payloadInput.getPayloadEnvelope() : undefined;
|
|
878
|
+
let i = 0;
|
|
879
|
+
while (i++ < this.getMaxDownloadAttempts()) {
|
|
880
|
+
const pendingColumns = payloadInput?.hasAllData()
|
|
881
|
+
? new Set()
|
|
882
|
+
: new Set(payloadInput?.getMissingSampledColumnMeta().missing ?? []);
|
|
883
|
+
const peerMeta = this.peerBalancer.bestPeerForPendingColumns(pendingColumns, excludedPeers);
|
|
884
|
+
if (peerMeta === null) {
|
|
885
|
+
throw Error(`Error fetching payload by root slot=${slot} root=${rootHex} after ${i}: cannot find peer with needed columns=${prettyPrintIndices(Array.from(pendingColumns))}`);
|
|
886
|
+
}
|
|
887
|
+
const { peerId, client: peerClient } = peerMeta;
|
|
888
|
+
cacheItem.peerIdStrings.add(peerId);
|
|
889
|
+
try {
|
|
890
|
+
if (!envelope) {
|
|
891
|
+
envelope = await this.fetchExecutionPayloadEnvelope(peerId, blockRoot, rootHex);
|
|
892
|
+
slot = envelope.message.payload.slotNumber;
|
|
893
|
+
}
|
|
894
|
+
payloadInput ??= this.chain.seenPayloadEnvelopeInputCache.get(rootHex);
|
|
895
|
+
if (!payloadInput) {
|
|
896
|
+
if (this.chain.forkChoice.hasBlockHex(rootHex)) {
|
|
897
|
+
throw new Error(`Missing PayloadEnvelopeInput for known block ${rootHex}`);
|
|
898
|
+
}
|
|
899
|
+
// Keep the validated envelope around, but wait for the block body before turning it into a full payload input.
|
|
900
|
+
return {
|
|
901
|
+
status: PendingPayloadInputStatus.waitingForBlock,
|
|
902
|
+
envelope,
|
|
903
|
+
timeAddedSec: cacheItem.timeAddedSec,
|
|
904
|
+
peerIdStrings: cacheItem.peerIdStrings,
|
|
905
|
+
};
|
|
906
|
+
}
|
|
907
|
+
if (!payloadInput.hasPayloadEnvelope()) {
|
|
908
|
+
await validateGossipExecutionPayloadEnvelope(this.chain, envelope);
|
|
909
|
+
}
|
|
910
|
+
let pendingPayload = this.toPendingPayloadInput(payloadInput, cacheItem, envelope);
|
|
911
|
+
if (!pendingPayload.payloadInput.hasAllData()) {
|
|
912
|
+
const missing = pendingPayload.payloadInput.getMissingSampledColumnMeta().missing;
|
|
913
|
+
if (missing.length > 0) {
|
|
914
|
+
const columnSidecars = await this.fetchPayloadColumns(peerMeta, pendingPayload.payloadInput, missing);
|
|
915
|
+
const seenTimestampSec = Date.now() / 1000;
|
|
916
|
+
for (const columnSidecar of columnSidecars) {
|
|
917
|
+
if (pendingPayload.payloadInput.hasColumn(columnSidecar.index)) {
|
|
918
|
+
continue;
|
|
919
|
+
}
|
|
920
|
+
pendingPayload.payloadInput.addColumn({
|
|
921
|
+
columnSidecar,
|
|
922
|
+
source: PayloadEnvelopeInputSource.byRoot,
|
|
923
|
+
seenTimestampSec,
|
|
924
|
+
peerIdStr: peerId,
|
|
925
|
+
});
|
|
926
|
+
}
|
|
927
|
+
pendingPayload = this.toPendingPayloadInput(pendingPayload.payloadInput, pendingPayload);
|
|
928
|
+
}
|
|
929
|
+
}
|
|
930
|
+
this.logger.verbose("BlockInputSync.fetchPayloadInput: successful download", {
|
|
931
|
+
slot,
|
|
932
|
+
rootHex,
|
|
933
|
+
peerId,
|
|
934
|
+
peerClient,
|
|
935
|
+
hasPayload: pendingPayload.payloadInput.hasPayloadEnvelope(),
|
|
936
|
+
hasAllData: pendingPayload.payloadInput.hasAllData(),
|
|
937
|
+
});
|
|
938
|
+
if (pendingPayload.status === PendingPayloadInputStatus.downloaded) {
|
|
939
|
+
return pendingPayload;
|
|
940
|
+
}
|
|
941
|
+
cacheItem = pendingPayload;
|
|
942
|
+
payloadInput = pendingPayload.payloadInput;
|
|
943
|
+
}
|
|
944
|
+
catch (e) {
|
|
945
|
+
this.logger.debug("Error downloading payload in BlockInputSync.fetchPayloadInput", { slot, rootHex, attempt: i, peer: peerId, peerClient }, e);
|
|
946
|
+
if (e instanceof RequestError) {
|
|
947
|
+
switch (e.type.code) {
|
|
948
|
+
case RequestErrorCode.REQUEST_RATE_LIMITED:
|
|
949
|
+
case RequestErrorCode.REQUEST_TIMEOUT:
|
|
950
|
+
break;
|
|
951
|
+
default:
|
|
952
|
+
excludedPeers.add(peerId);
|
|
953
|
+
break;
|
|
954
|
+
}
|
|
955
|
+
}
|
|
956
|
+
else {
|
|
957
|
+
excludedPeers.add(peerId);
|
|
958
|
+
}
|
|
959
|
+
}
|
|
960
|
+
finally {
|
|
961
|
+
this.peerBalancer.onRequestCompleted(peerId);
|
|
962
|
+
}
|
|
963
|
+
}
|
|
964
|
+
throw Error(`Error fetching payload with slot=${slot} root=${rootHex} after ${i - 1} attempts.`);
|
|
965
|
+
}
|
|
966
|
+
async fetchExecutionPayloadEnvelope(peerIdStr, blockRoot, rootHex) {
|
|
967
|
+
const response = await this.network.sendExecutionPayloadEnvelopesByRoot(peerIdStr, [blockRoot]);
|
|
968
|
+
const envelope = response.at(0);
|
|
969
|
+
if (!envelope) {
|
|
970
|
+
throw new Error(`Missing execution payload envelope for root=${rootHex}`);
|
|
971
|
+
}
|
|
972
|
+
const receivedRootHex = toRootHex(envelope.message.beaconBlockRoot);
|
|
973
|
+
if (receivedRootHex !== rootHex) {
|
|
974
|
+
throw new Error(`Execution payload envelope root mismatch requested=${rootHex} received=${receivedRootHex}`);
|
|
975
|
+
}
|
|
976
|
+
return envelope;
|
|
977
|
+
}
|
|
978
|
+
async fetchPayloadColumns(peerMeta, payloadInput, missing) {
|
|
979
|
+
const { peerId: peerIdStr } = peerMeta;
|
|
980
|
+
const peerColumns = new Set(peerMeta.custodyColumns ?? []);
|
|
981
|
+
const requestedColumns = missing.filter((columnIndex) => peerColumns.has(columnIndex));
|
|
982
|
+
if (requestedColumns.length === 0) {
|
|
983
|
+
return [];
|
|
984
|
+
}
|
|
985
|
+
const columnSidecars = (await this.network.sendDataColumnSidecarsByRoot(peerIdStr, [
|
|
986
|
+
{ blockRoot: fromHex(payloadInput.blockRootHex), columns: requestedColumns },
|
|
987
|
+
]));
|
|
988
|
+
if (columnSidecars.length === 0) {
|
|
989
|
+
throw new Error(`No data column sidecars returned for payload root=${payloadInput.blockRootHex}`);
|
|
990
|
+
}
|
|
991
|
+
const requestedColumnsSet = new Set(requestedColumns);
|
|
992
|
+
const extraColumns = columnSidecars.filter((columnSidecar) => !requestedColumnsSet.has(columnSidecar.index));
|
|
993
|
+
if (extraColumns.length > 0) {
|
|
994
|
+
throw new Error(`Received unexpected payload data columns indices=${prettyPrintIndices(extraColumns.map((column) => column.index))}`);
|
|
995
|
+
}
|
|
996
|
+
// PayloadEnvelopeInput already carries the block slot, root, and commitments, so reuse the
|
|
997
|
+
// block-based Gloas validator rather than maintaining a second payload-specific variant.
|
|
998
|
+
await validateGloasBlockDataColumnSidecars(payloadInput.slot, fromHex(payloadInput.blockRootHex), payloadInput.getBlobKzgCommitments(), columnSidecars, this.chain.metrics?.peerDas);
|
|
999
|
+
return columnSidecars;
|
|
1000
|
+
}
|
|
427
1001
|
/**
|
|
428
1002
|
* From a set of shuffled peers:
|
|
429
1003
|
* - fetch the block
|
|
@@ -582,6 +1156,25 @@ export class BlockInputSync {
|
|
|
582
1156
|
// Prune knownBadBlocks
|
|
583
1157
|
pruneSetToMax(this.knownBadBlocks, MAX_KNOWN_BAD_BLOCKS);
|
|
584
1158
|
}
|
|
1159
|
+
// Once a parent payload is invalid, every descendant waiting on that payload lineage becomes unrecoverable too.
|
|
1160
|
+
removePendingPayloadAndDescendants(rootHex) {
|
|
1161
|
+
// Keep PayloadEnvelopeInput resident in the seen cache. importBlock() owns that object and
|
|
1162
|
+
// later validation/finalization logic decides when it can leave memory.
|
|
1163
|
+
this.pendingPayloads.delete(rootHex);
|
|
1164
|
+
const badPendingBlocks = getAllDescendantBlocks(rootHex, this.pendingBlocks);
|
|
1165
|
+
this.metrics?.blockInputSync.removedBlocks.inc(badPendingBlocks.length);
|
|
1166
|
+
for (const block of badPendingBlocks) {
|
|
1167
|
+
const descendantRootHex = getBlockInputSyncCacheItemRootHex(block);
|
|
1168
|
+
this.pendingBlocks.delete(descendantRootHex);
|
|
1169
|
+
this.pendingPayloads.delete(descendantRootHex);
|
|
1170
|
+
this.chain.seenBlockInputCache.prune(descendantRootHex);
|
|
1171
|
+
this.logger.debug("Removing pending descendant after invalid parent payload", {
|
|
1172
|
+
slot: getBlockInputSyncCacheItemSlot(block),
|
|
1173
|
+
blockRoot: descendantRootHex,
|
|
1174
|
+
parentPayloadRoot: rootHex,
|
|
1175
|
+
});
|
|
1176
|
+
}
|
|
1177
|
+
}
|
|
585
1178
|
removeAllDescendants(block) {
|
|
586
1179
|
const rootHex = getBlockInputSyncCacheItemRootHex(block);
|
|
587
1180
|
const slot = getBlockInputSyncCacheItemSlot(block);
|
|
@@ -591,7 +1184,10 @@ export class BlockInputSync {
|
|
|
591
1184
|
for (const block of badPendingBlocks) {
|
|
592
1185
|
const rootHex = getBlockInputSyncCacheItemRootHex(block);
|
|
593
1186
|
this.pendingBlocks.delete(rootHex);
|
|
1187
|
+
this.pendingPayloads.delete(rootHex);
|
|
594
1188
|
this.chain.seenBlockInputCache.prune(rootHex);
|
|
1189
|
+
// Keep PayloadEnvelopeInput resident in the seen cache for consistency with the
|
|
1190
|
+
// importBlock()-owned lifecycle.
|
|
595
1191
|
this.logger.debug("Removing bad/unknown/incomplete BlockInputSyncCacheItem", {
|
|
596
1192
|
slot,
|
|
597
1193
|
blockRoot: rootHex,
|