@aztec/validator-client 0.0.1-commit.e2b2873ed → 0.0.1-commit.e304674f1
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/README.md +41 -2
- package/dest/checkpoint_builder.d.ts +21 -8
- package/dest/checkpoint_builder.d.ts.map +1 -1
- package/dest/checkpoint_builder.js +124 -46
- package/dest/config.d.ts +1 -1
- package/dest/config.d.ts.map +1 -1
- package/dest/config.js +22 -6
- package/dest/duties/validation_service.d.ts +3 -4
- package/dest/duties/validation_service.d.ts.map +1 -1
- package/dest/duties/validation_service.js +11 -22
- package/dest/factory.d.ts +7 -4
- package/dest/factory.d.ts.map +1 -1
- package/dest/factory.js +6 -5
- package/dest/index.d.ts +2 -3
- package/dest/index.d.ts.map +1 -1
- package/dest/index.js +1 -2
- package/dest/key_store/ha_key_store.js +1 -1
- package/dest/metrics.d.ts +10 -2
- package/dest/metrics.d.ts.map +1 -1
- package/dest/metrics.js +12 -0
- package/dest/proposal_handler.d.ts +107 -0
- package/dest/proposal_handler.d.ts.map +1 -0
- package/dest/proposal_handler.js +966 -0
- package/dest/validator.d.ts +18 -15
- package/dest/validator.d.ts.map +1 -1
- package/dest/validator.js +80 -191
- package/package.json +19 -19
- package/src/checkpoint_builder.ts +142 -39
- package/src/config.ts +22 -6
- package/src/duties/validation_service.ts +18 -24
- package/src/factory.ts +9 -3
- package/src/index.ts +1 -2
- package/src/key_store/ha_key_store.ts +1 -1
- package/src/metrics.ts +19 -1
- package/src/proposal_handler.ts +1033 -0
- package/src/validator.ts +103 -210
- package/dest/block_proposal_handler.d.ts +0 -63
- package/dest/block_proposal_handler.d.ts.map +0 -1
- package/dest/block_proposal_handler.js +0 -546
- package/dest/tx_validator/index.d.ts +0 -3
- package/dest/tx_validator/index.d.ts.map +0 -1
- package/dest/tx_validator/index.js +0 -2
- package/dest/tx_validator/nullifier_cache.d.ts +0 -14
- package/dest/tx_validator/nullifier_cache.d.ts.map +0 -1
- package/dest/tx_validator/nullifier_cache.js +0 -24
- package/dest/tx_validator/tx_validator_factory.d.ts +0 -19
- package/dest/tx_validator/tx_validator_factory.d.ts.map +0 -1
- package/dest/tx_validator/tx_validator_factory.js +0 -54
- package/src/block_proposal_handler.ts +0 -555
- package/src/tx_validator/index.ts +0 -2
- package/src/tx_validator/nullifier_cache.ts +0 -30
- package/src/tx_validator/tx_validator_factory.ts +0 -154
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@aztec/validator-client",
|
|
3
|
-
"version": "0.0.1-commit.
|
|
3
|
+
"version": "0.0.1-commit.e304674f1",
|
|
4
4
|
"main": "dest/index.js",
|
|
5
5
|
"type": "module",
|
|
6
6
|
"exports": {
|
|
@@ -64,30 +64,30 @@
|
|
|
64
64
|
]
|
|
65
65
|
},
|
|
66
66
|
"dependencies": {
|
|
67
|
-
"@aztec/blob-client": "0.0.1-commit.
|
|
68
|
-
"@aztec/blob-lib": "0.0.1-commit.
|
|
69
|
-
"@aztec/constants": "0.0.1-commit.
|
|
70
|
-
"@aztec/epoch-cache": "0.0.1-commit.
|
|
71
|
-
"@aztec/ethereum": "0.0.1-commit.
|
|
72
|
-
"@aztec/foundation": "0.0.1-commit.
|
|
73
|
-
"@aztec/node-keystore": "0.0.1-commit.
|
|
74
|
-
"@aztec/noir-protocol-circuits-types": "0.0.1-commit.
|
|
75
|
-
"@aztec/p2p": "0.0.1-commit.
|
|
76
|
-
"@aztec/protocol-contracts": "0.0.1-commit.
|
|
77
|
-
"@aztec/prover-client": "0.0.1-commit.
|
|
78
|
-
"@aztec/simulator": "0.0.1-commit.
|
|
79
|
-
"@aztec/slasher": "0.0.1-commit.
|
|
80
|
-
"@aztec/stdlib": "0.0.1-commit.
|
|
81
|
-
"@aztec/telemetry-client": "0.0.1-commit.
|
|
82
|
-
"@aztec/validator-ha-signer": "0.0.1-commit.
|
|
67
|
+
"@aztec/blob-client": "0.0.1-commit.e304674f1",
|
|
68
|
+
"@aztec/blob-lib": "0.0.1-commit.e304674f1",
|
|
69
|
+
"@aztec/constants": "0.0.1-commit.e304674f1",
|
|
70
|
+
"@aztec/epoch-cache": "0.0.1-commit.e304674f1",
|
|
71
|
+
"@aztec/ethereum": "0.0.1-commit.e304674f1",
|
|
72
|
+
"@aztec/foundation": "0.0.1-commit.e304674f1",
|
|
73
|
+
"@aztec/node-keystore": "0.0.1-commit.e304674f1",
|
|
74
|
+
"@aztec/noir-protocol-circuits-types": "0.0.1-commit.e304674f1",
|
|
75
|
+
"@aztec/p2p": "0.0.1-commit.e304674f1",
|
|
76
|
+
"@aztec/protocol-contracts": "0.0.1-commit.e304674f1",
|
|
77
|
+
"@aztec/prover-client": "0.0.1-commit.e304674f1",
|
|
78
|
+
"@aztec/simulator": "0.0.1-commit.e304674f1",
|
|
79
|
+
"@aztec/slasher": "0.0.1-commit.e304674f1",
|
|
80
|
+
"@aztec/stdlib": "0.0.1-commit.e304674f1",
|
|
81
|
+
"@aztec/telemetry-client": "0.0.1-commit.e304674f1",
|
|
82
|
+
"@aztec/validator-ha-signer": "0.0.1-commit.e304674f1",
|
|
83
83
|
"koa": "^2.16.1",
|
|
84
84
|
"koa-router": "^13.1.1",
|
|
85
85
|
"tslib": "^2.4.0",
|
|
86
86
|
"viem": "npm:@aztec/viem@2.38.2"
|
|
87
87
|
},
|
|
88
88
|
"devDependencies": {
|
|
89
|
-
"@aztec/archiver": "0.0.1-commit.
|
|
90
|
-
"@aztec/world-state": "0.0.1-commit.
|
|
89
|
+
"@aztec/archiver": "0.0.1-commit.e304674f1",
|
|
90
|
+
"@aztec/world-state": "0.0.1-commit.e304674f1",
|
|
91
91
|
"@electric-sql/pglite": "^0.3.14",
|
|
92
92
|
"@jest/globals": "^30.0.0",
|
|
93
93
|
"@types/jest": "^30.0.0",
|
|
@@ -1,10 +1,12 @@
|
|
|
1
|
+
import { NUM_CHECKPOINT_END_MARKER_FIELDS, getNumBlockEndBlobFields } from '@aztec/blob-lib/encoding';
|
|
2
|
+
import { BLOBS_PER_CHECKPOINT, FIELDS_PER_BLOB, MAX_PROCESSABLE_DA_GAS_PER_CHECKPOINT } from '@aztec/constants';
|
|
1
3
|
import { BlockNumber, CheckpointNumber } from '@aztec/foundation/branded-types';
|
|
2
|
-
import { merge, pick } from '@aztec/foundation/collection';
|
|
4
|
+
import { merge, pick, sum } from '@aztec/foundation/collection';
|
|
3
5
|
import { Fr } from '@aztec/foundation/curves/bn254';
|
|
4
6
|
import { type Logger, type LoggerBindings, createLogger } from '@aztec/foundation/log';
|
|
5
7
|
import { bufferToHex } from '@aztec/foundation/string';
|
|
6
8
|
import { DateProvider, elapsed } from '@aztec/foundation/timer';
|
|
7
|
-
import { getDefaultAllowedSetupFunctions } from '@aztec/p2p/msg_validators';
|
|
9
|
+
import { createTxValidatorForBlockBuilding, getDefaultAllowedSetupFunctions } from '@aztec/p2p/msg_validators';
|
|
8
10
|
import { LightweightCheckpointBuilder } from '@aztec/prover-client/light';
|
|
9
11
|
import {
|
|
10
12
|
GuardedMerkleTreeOperations,
|
|
@@ -18,21 +20,22 @@ import type { ContractDataSource } from '@aztec/stdlib/contract';
|
|
|
18
20
|
import type { L1RollupConstants } from '@aztec/stdlib/epoch-helpers';
|
|
19
21
|
import { Gas } from '@aztec/stdlib/gas';
|
|
20
22
|
import {
|
|
23
|
+
type BlockBuilderOptions,
|
|
21
24
|
type BuildBlockInCheckpointResult,
|
|
22
25
|
type FullNodeBlockBuilderConfig,
|
|
23
26
|
FullNodeBlockBuilderConfigKeys,
|
|
24
27
|
type ICheckpointBlockBuilder,
|
|
25
28
|
type ICheckpointsBuilder,
|
|
29
|
+
InsufficientValidTxsError,
|
|
26
30
|
type MerkleTreeWriteOperations,
|
|
27
|
-
NoValidTxsError,
|
|
28
31
|
type PublicProcessorLimits,
|
|
29
32
|
type WorldStateSynchronizer,
|
|
30
33
|
} from '@aztec/stdlib/interfaces/server';
|
|
34
|
+
import { type DebugLogStore, NullDebugLogStore } from '@aztec/stdlib/logs';
|
|
31
35
|
import { MerkleTreeId } from '@aztec/stdlib/trees';
|
|
32
36
|
import { type CheckpointGlobalVariables, GlobalVariables, StateReference, Tx } from '@aztec/stdlib/tx';
|
|
33
37
|
import { type TelemetryClient, getTelemetryClient } from '@aztec/telemetry-client';
|
|
34
|
-
|
|
35
|
-
import { createValidatorForBlockBuilding } from './tx_validator/tx_validator_factory.js';
|
|
38
|
+
import { ForkCheckpoint } from '@aztec/world-state';
|
|
36
39
|
|
|
37
40
|
// Re-export for backward compatibility
|
|
38
41
|
export type { BuildBlockInCheckpointResult } from '@aztec/stdlib/interfaces/server';
|
|
@@ -44,6 +47,9 @@ export type { BuildBlockInCheckpointResult } from '@aztec/stdlib/interfaces/serv
|
|
|
44
47
|
export class CheckpointBuilder implements ICheckpointBlockBuilder {
|
|
45
48
|
private log: Logger;
|
|
46
49
|
|
|
50
|
+
/** Persistent contracts DB shared across all blocks in this checkpoint. */
|
|
51
|
+
protected contractsDB: PublicContractsDB;
|
|
52
|
+
|
|
47
53
|
constructor(
|
|
48
54
|
private checkpointBuilder: LightweightCheckpointBuilder,
|
|
49
55
|
private fork: MerkleTreeWriteOperations,
|
|
@@ -52,11 +58,13 @@ export class CheckpointBuilder implements ICheckpointBlockBuilder {
|
|
|
52
58
|
private dateProvider: DateProvider,
|
|
53
59
|
private telemetryClient: TelemetryClient,
|
|
54
60
|
bindings?: LoggerBindings,
|
|
61
|
+
private debugLogStore: DebugLogStore = new NullDebugLogStore(),
|
|
55
62
|
) {
|
|
56
63
|
this.log = createLogger('checkpoint-builder', {
|
|
57
64
|
...bindings,
|
|
58
65
|
instanceId: `checkpoint-${checkpointBuilder.checkpointNumber}`,
|
|
59
66
|
});
|
|
67
|
+
this.contractsDB = new PublicContractsDB(this.contractDataSource, this.log.getBindings());
|
|
60
68
|
}
|
|
61
69
|
|
|
62
70
|
getConstantData(): CheckpointGlobalVariables {
|
|
@@ -65,12 +73,13 @@ export class CheckpointBuilder implements ICheckpointBlockBuilder {
|
|
|
65
73
|
|
|
66
74
|
/**
|
|
67
75
|
* Builds a single block within this checkpoint.
|
|
76
|
+
* Automatically caps gas and blob field limits based on checkpoint-level budgets and prior blocks.
|
|
68
77
|
*/
|
|
69
78
|
async buildBlock(
|
|
70
79
|
pendingTxs: Iterable<Tx> | AsyncIterable<Tx>,
|
|
71
80
|
blockNumber: BlockNumber,
|
|
72
81
|
timestamp: bigint,
|
|
73
|
-
opts:
|
|
82
|
+
opts: BlockBuilderOptions & { expectedEndState?: StateReference },
|
|
74
83
|
): Promise<BuildBlockInCheckpointResult> {
|
|
75
84
|
const slot = this.checkpointBuilder.constants.slotNumber;
|
|
76
85
|
|
|
@@ -94,39 +103,60 @@ export class CheckpointBuilder implements ICheckpointBlockBuilder {
|
|
|
94
103
|
});
|
|
95
104
|
const { processor, validator } = await this.makeBlockBuilderDeps(globalVariables, this.fork);
|
|
96
105
|
|
|
97
|
-
|
|
98
|
-
|
|
99
|
-
|
|
100
|
-
|
|
101
|
-
|
|
102
|
-
// (only the first block in a checkpoint can be empty)
|
|
103
|
-
if (processedTxs.length === 0 && this.checkpointBuilder.getBlockCount() > 0) {
|
|
104
|
-
throw new NoValidTxsError(failedTxs);
|
|
105
|
-
}
|
|
106
|
-
|
|
107
|
-
// Add block to checkpoint
|
|
108
|
-
const block = await this.checkpointBuilder.addBlock(globalVariables, processedTxs, {
|
|
109
|
-
expectedEndState: opts.expectedEndState,
|
|
110
|
-
});
|
|
106
|
+
// Cap gas limits amd available blob fields by remaining checkpoint-level budgets
|
|
107
|
+
const cappedOpts: PublicProcessorLimits & { expectedEndState?: StateReference } = {
|
|
108
|
+
...opts,
|
|
109
|
+
...this.capLimitsByCheckpointBudgets(opts),
|
|
110
|
+
};
|
|
111
111
|
|
|
112
|
-
//
|
|
113
|
-
|
|
112
|
+
// Create a block-level checkpoint on the contracts DB so we can roll back on failure
|
|
113
|
+
this.contractsDB.createCheckpoint();
|
|
114
|
+
// We execute all merkle tree operations on a world state fork checkpoint
|
|
115
|
+
// This enables us to discard all modifications in the event that we fail to successfully process sufficient transactions
|
|
116
|
+
const forkCheckpoint = await ForkCheckpoint.new(this.fork);
|
|
114
117
|
|
|
115
|
-
|
|
116
|
-
|
|
117
|
-
|
|
118
|
-
|
|
119
|
-
});
|
|
118
|
+
try {
|
|
119
|
+
const [publicProcessorDuration, [processedTxs, failedTxs, usedTxs]] = await elapsed(() =>
|
|
120
|
+
processor.process(pendingTxs, cappedOpts, validator),
|
|
121
|
+
);
|
|
120
122
|
|
|
121
|
-
|
|
122
|
-
|
|
123
|
-
|
|
124
|
-
|
|
125
|
-
|
|
126
|
-
|
|
127
|
-
|
|
128
|
-
|
|
129
|
-
|
|
123
|
+
// Throw before updating state if we don't have enough valid txs
|
|
124
|
+
const minValidTxs = opts.minValidTxs ?? 0;
|
|
125
|
+
if (processedTxs.length < minValidTxs) {
|
|
126
|
+
throw new InsufficientValidTxsError(processedTxs.length, minValidTxs, failedTxs);
|
|
127
|
+
}
|
|
128
|
+
|
|
129
|
+
// Commit the fork checkpoint
|
|
130
|
+
await forkCheckpoint.commit();
|
|
131
|
+
|
|
132
|
+
// Add block to checkpoint
|
|
133
|
+
const { block } = await this.checkpointBuilder.addBlock(globalVariables, processedTxs, {
|
|
134
|
+
expectedEndState: opts.expectedEndState,
|
|
135
|
+
});
|
|
136
|
+
|
|
137
|
+
this.contractsDB.commitCheckpoint();
|
|
138
|
+
|
|
139
|
+
this.log.debug('Built block within checkpoint', {
|
|
140
|
+
header: block.header.toInspect(),
|
|
141
|
+
processedTxs: processedTxs.map(tx => tx.hash.toString()),
|
|
142
|
+
failedTxs: failedTxs.map(tx => tx.tx.txHash.toString()),
|
|
143
|
+
});
|
|
144
|
+
|
|
145
|
+
return {
|
|
146
|
+
block,
|
|
147
|
+
publicProcessorDuration,
|
|
148
|
+
numTxs: processedTxs.length,
|
|
149
|
+
failedTxs,
|
|
150
|
+
usedTxs,
|
|
151
|
+
};
|
|
152
|
+
} catch (err) {
|
|
153
|
+
// Revert all changes to contracts db
|
|
154
|
+
this.contractsDB.revertCheckpoint();
|
|
155
|
+
// If we reached the point of committing the checkpoint, this does nothing
|
|
156
|
+
// Otherwise it reverts any changes made to the fork for this failed block
|
|
157
|
+
await forkCheckpoint.revert();
|
|
158
|
+
throw err;
|
|
159
|
+
}
|
|
130
160
|
}
|
|
131
161
|
|
|
132
162
|
/** Completes the checkpoint and returns it. */
|
|
@@ -147,11 +177,72 @@ export class CheckpointBuilder implements ICheckpointBlockBuilder {
|
|
|
147
177
|
return this.checkpointBuilder.clone().completeCheckpoint();
|
|
148
178
|
}
|
|
149
179
|
|
|
180
|
+
/**
|
|
181
|
+
* Caps per-block gas and blob field limits by remaining checkpoint-level budgets.
|
|
182
|
+
* When building a proposal (isBuildingProposal=true), computes a fair share of remaining budget
|
|
183
|
+
* across remaining blocks scaled by the multiplier. When validating, only caps by per-block limit
|
|
184
|
+
* and remaining checkpoint budget (no redistribution or multiplier).
|
|
185
|
+
*/
|
|
186
|
+
protected capLimitsByCheckpointBudgets(
|
|
187
|
+
opts: BlockBuilderOptions,
|
|
188
|
+
): Pick<PublicProcessorLimits, 'maxBlockGas' | 'maxBlobFields' | 'maxTransactions'> {
|
|
189
|
+
const existingBlocks = this.checkpointBuilder.getBlocks();
|
|
190
|
+
|
|
191
|
+
// Remaining L2 gas (mana)
|
|
192
|
+
// IMPORTANT: This assumes mana is computed solely based on L2 gas used in transactions.
|
|
193
|
+
// This may change in the future.
|
|
194
|
+
const usedMana = sum(existingBlocks.map(b => b.header.totalManaUsed.toNumber()));
|
|
195
|
+
const remainingMana = this.config.rollupManaLimit - usedMana;
|
|
196
|
+
|
|
197
|
+
// Remaining DA gas
|
|
198
|
+
const usedDAGas = sum(existingBlocks.map(b => b.computeDAGasUsed())) ?? 0;
|
|
199
|
+
const remainingDAGas = MAX_PROCESSABLE_DA_GAS_PER_CHECKPOINT - usedDAGas;
|
|
200
|
+
|
|
201
|
+
// Remaining blob fields (block blob fields include both tx data and block-end overhead)
|
|
202
|
+
const usedBlobFields = sum(existingBlocks.map(b => b.toBlobFields().length));
|
|
203
|
+
const totalBlobCapacity = BLOBS_PER_CHECKPOINT * FIELDS_PER_BLOB - NUM_CHECKPOINT_END_MARKER_FIELDS;
|
|
204
|
+
const isFirstBlock = existingBlocks.length === 0;
|
|
205
|
+
const blockEndOverhead = getNumBlockEndBlobFields(isFirstBlock);
|
|
206
|
+
const maxBlobFieldsForTxs = totalBlobCapacity - usedBlobFields - blockEndOverhead;
|
|
207
|
+
|
|
208
|
+
// Remaining txs
|
|
209
|
+
const usedTxs = sum(existingBlocks.map(b => b.body.txEffects.length));
|
|
210
|
+
const remainingTxs = Math.max(0, (this.config.maxTxsPerCheckpoint ?? Infinity) - usedTxs);
|
|
211
|
+
|
|
212
|
+
// Cap by per-block limit + remaining checkpoint budget
|
|
213
|
+
let cappedL2Gas = Math.min(opts.maxBlockGas?.l2Gas ?? Infinity, remainingMana);
|
|
214
|
+
let cappedDAGas = Math.min(opts.maxBlockGas?.daGas ?? Infinity, remainingDAGas);
|
|
215
|
+
let cappedBlobFields = Math.min(opts.maxBlobFields ?? Infinity, maxBlobFieldsForTxs);
|
|
216
|
+
let cappedMaxTransactions = Math.min(opts.maxTransactions ?? Infinity, remainingTxs);
|
|
217
|
+
|
|
218
|
+
// Proposer mode: further cap by fair share of remaining budget across remaining blocks
|
|
219
|
+
if (opts.isBuildingProposal) {
|
|
220
|
+
const remainingBlocks = Math.max(1, opts.maxBlocksPerCheckpoint - existingBlocks.length);
|
|
221
|
+
const multiplier = opts.perBlockAllocationMultiplier;
|
|
222
|
+
|
|
223
|
+
cappedL2Gas = Math.min(cappedL2Gas, Math.ceil((remainingMana / remainingBlocks) * multiplier));
|
|
224
|
+
cappedDAGas = Math.min(cappedDAGas, Math.ceil((remainingDAGas / remainingBlocks) * multiplier));
|
|
225
|
+
cappedBlobFields = Math.min(cappedBlobFields, Math.ceil((maxBlobFieldsForTxs / remainingBlocks) * multiplier));
|
|
226
|
+
cappedMaxTransactions = Math.min(cappedMaxTransactions, Math.ceil((remainingTxs / remainingBlocks) * multiplier));
|
|
227
|
+
}
|
|
228
|
+
|
|
229
|
+
return {
|
|
230
|
+
maxBlockGas: new Gas(cappedDAGas, cappedL2Gas),
|
|
231
|
+
maxBlobFields: cappedBlobFields,
|
|
232
|
+
maxTransactions: Number.isFinite(cappedMaxTransactions) ? cappedMaxTransactions : undefined,
|
|
233
|
+
};
|
|
234
|
+
}
|
|
235
|
+
|
|
150
236
|
protected async makeBlockBuilderDeps(globalVariables: GlobalVariables, fork: MerkleTreeWriteOperations) {
|
|
151
|
-
const txPublicSetupAllowList =
|
|
152
|
-
|
|
237
|
+
const txPublicSetupAllowList = [
|
|
238
|
+
...(await getDefaultAllowedSetupFunctions()),
|
|
239
|
+
...(this.config.txPublicSetupAllowListExtend ?? []),
|
|
240
|
+
];
|
|
241
|
+
const contractsDB = this.contractsDB;
|
|
153
242
|
const guardedFork = new GuardedMerkleTreeOperations(fork);
|
|
154
243
|
|
|
244
|
+
const collectDebugLogs = this.debugLogStore.isEnabled;
|
|
245
|
+
|
|
155
246
|
const bindings = this.log.getBindings();
|
|
156
247
|
const publicTxSimulator = createPublicTxSimulatorForBlockBuilding(
|
|
157
248
|
guardedFork,
|
|
@@ -159,6 +250,7 @@ export class CheckpointBuilder implements ICheckpointBlockBuilder {
|
|
|
159
250
|
globalVariables,
|
|
160
251
|
this.telemetryClient,
|
|
161
252
|
bindings,
|
|
253
|
+
collectDebugLogs,
|
|
162
254
|
);
|
|
163
255
|
|
|
164
256
|
const processor = new PublicProcessor(
|
|
@@ -170,9 +262,10 @@ export class CheckpointBuilder implements ICheckpointBlockBuilder {
|
|
|
170
262
|
this.telemetryClient,
|
|
171
263
|
createLogger('simulator:public-processor', bindings),
|
|
172
264
|
this.config,
|
|
265
|
+
this.debugLogStore,
|
|
173
266
|
);
|
|
174
267
|
|
|
175
|
-
const validator =
|
|
268
|
+
const validator = createTxValidatorForBlockBuilding(
|
|
176
269
|
fork,
|
|
177
270
|
this.contractDataSource,
|
|
178
271
|
globalVariables,
|
|
@@ -197,6 +290,7 @@ export class FullNodeCheckpointsBuilder implements ICheckpointsBuilder {
|
|
|
197
290
|
private contractDataSource: ContractDataSource,
|
|
198
291
|
private dateProvider: DateProvider,
|
|
199
292
|
private telemetryClient: TelemetryClient = getTelemetryClient(),
|
|
293
|
+
private debugLogStore: DebugLogStore = new NullDebugLogStore(),
|
|
200
294
|
) {
|
|
201
295
|
this.log = createLogger('checkpoint-builder');
|
|
202
296
|
}
|
|
@@ -215,6 +309,7 @@ export class FullNodeCheckpointsBuilder implements ICheckpointsBuilder {
|
|
|
215
309
|
async startCheckpoint(
|
|
216
310
|
checkpointNumber: CheckpointNumber,
|
|
217
311
|
constants: CheckpointGlobalVariables,
|
|
312
|
+
feeAssetPriceModifier: bigint,
|
|
218
313
|
l1ToL2Messages: Fr[],
|
|
219
314
|
previousCheckpointOutHashes: Fr[],
|
|
220
315
|
fork: MerkleTreeWriteOperations,
|
|
@@ -229,6 +324,7 @@ export class FullNodeCheckpointsBuilder implements ICheckpointsBuilder {
|
|
|
229
324
|
initialStateReference: stateReference.toInspect(),
|
|
230
325
|
initialArchiveRoot: bufferToHex(archiveTree.root),
|
|
231
326
|
constants,
|
|
327
|
+
feeAssetPriceModifier,
|
|
232
328
|
});
|
|
233
329
|
|
|
234
330
|
const lightweightBuilder = await LightweightCheckpointBuilder.startNewCheckpoint(
|
|
@@ -238,6 +334,7 @@ export class FullNodeCheckpointsBuilder implements ICheckpointsBuilder {
|
|
|
238
334
|
previousCheckpointOutHashes,
|
|
239
335
|
fork,
|
|
240
336
|
bindings,
|
|
337
|
+
feeAssetPriceModifier,
|
|
241
338
|
);
|
|
242
339
|
|
|
243
340
|
return new CheckpointBuilder(
|
|
@@ -248,6 +345,7 @@ export class FullNodeCheckpointsBuilder implements ICheckpointsBuilder {
|
|
|
248
345
|
this.dateProvider,
|
|
249
346
|
this.telemetryClient,
|
|
250
347
|
bindings,
|
|
348
|
+
this.debugLogStore,
|
|
251
349
|
);
|
|
252
350
|
}
|
|
253
351
|
|
|
@@ -257,6 +355,7 @@ export class FullNodeCheckpointsBuilder implements ICheckpointsBuilder {
|
|
|
257
355
|
async openCheckpoint(
|
|
258
356
|
checkpointNumber: CheckpointNumber,
|
|
259
357
|
constants: CheckpointGlobalVariables,
|
|
358
|
+
feeAssetPriceModifier: bigint,
|
|
260
359
|
l1ToL2Messages: Fr[],
|
|
261
360
|
previousCheckpointOutHashes: Fr[],
|
|
262
361
|
fork: MerkleTreeWriteOperations,
|
|
@@ -270,6 +369,7 @@ export class FullNodeCheckpointsBuilder implements ICheckpointsBuilder {
|
|
|
270
369
|
return this.startCheckpoint(
|
|
271
370
|
checkpointNumber,
|
|
272
371
|
constants,
|
|
372
|
+
feeAssetPriceModifier,
|
|
273
373
|
l1ToL2Messages,
|
|
274
374
|
previousCheckpointOutHashes,
|
|
275
375
|
fork,
|
|
@@ -284,11 +384,13 @@ export class FullNodeCheckpointsBuilder implements ICheckpointsBuilder {
|
|
|
284
384
|
initialStateReference: stateReference.toInspect(),
|
|
285
385
|
initialArchiveRoot: bufferToHex(archiveTree.root),
|
|
286
386
|
constants,
|
|
387
|
+
feeAssetPriceModifier,
|
|
287
388
|
});
|
|
288
389
|
|
|
289
390
|
const lightweightBuilder = await LightweightCheckpointBuilder.resumeCheckpoint(
|
|
290
391
|
checkpointNumber,
|
|
291
392
|
constants,
|
|
393
|
+
feeAssetPriceModifier,
|
|
292
394
|
l1ToL2Messages,
|
|
293
395
|
previousCheckpointOutHashes,
|
|
294
396
|
fork,
|
|
@@ -304,6 +406,7 @@ export class FullNodeCheckpointsBuilder implements ICheckpointsBuilder {
|
|
|
304
406
|
this.dateProvider,
|
|
305
407
|
this.telemetryClient,
|
|
306
408
|
bindings,
|
|
409
|
+
this.debugLogStore,
|
|
307
410
|
);
|
|
308
411
|
}
|
|
309
412
|
|
package/src/config.ts
CHANGED
|
@@ -6,8 +6,8 @@ import {
|
|
|
6
6
|
secretValueConfigHelper,
|
|
7
7
|
} from '@aztec/foundation/config';
|
|
8
8
|
import { EthAddress } from '@aztec/foundation/eth-address';
|
|
9
|
+
import { localSignerConfigMappings, validatorHASignerConfigMappings } from '@aztec/stdlib/ha-signing';
|
|
9
10
|
import type { ValidatorClientConfig } from '@aztec/stdlib/interfaces/server';
|
|
10
|
-
import { validatorHASignerConfigMappings } from '@aztec/validator-ha-signer/config';
|
|
11
11
|
|
|
12
12
|
export type { ValidatorClientConfig };
|
|
13
13
|
|
|
@@ -49,11 +49,6 @@ export const validatorClientConfigMappings: ConfigMappingsType<ValidatorClientCo
|
|
|
49
49
|
description: 'Interval between polling for new attestations',
|
|
50
50
|
...numberConfigHelper(200),
|
|
51
51
|
},
|
|
52
|
-
validatorReexecute: {
|
|
53
|
-
env: 'VALIDATOR_REEXECUTE',
|
|
54
|
-
description: 'Re-execute transactions before attesting',
|
|
55
|
-
...booleanConfigHelper(true),
|
|
56
|
-
},
|
|
57
52
|
alwaysReexecuteBlockProposals: {
|
|
58
53
|
description:
|
|
59
54
|
'Whether to always reexecute block proposals, even for non-validator nodes (useful for monitoring network status).',
|
|
@@ -77,6 +72,27 @@ export const validatorClientConfigMappings: ConfigMappingsType<ValidatorClientCo
|
|
|
77
72
|
description: 'Agree to attest to equivocated checkpoint proposals (for testing purposes only)',
|
|
78
73
|
...booleanConfigHelper(false),
|
|
79
74
|
},
|
|
75
|
+
validateMaxL2BlockGas: {
|
|
76
|
+
env: 'VALIDATOR_MAX_L2_BLOCK_GAS',
|
|
77
|
+
description: 'Maximum L2 block gas for validation. Proposals exceeding this limit are rejected.',
|
|
78
|
+
parseEnv: (val: string) => (val ? parseInt(val, 10) : undefined),
|
|
79
|
+
},
|
|
80
|
+
validateMaxDABlockGas: {
|
|
81
|
+
env: 'VALIDATOR_MAX_DA_BLOCK_GAS',
|
|
82
|
+
description: 'Maximum DA block gas for validation. Proposals exceeding this limit are rejected.',
|
|
83
|
+
parseEnv: (val: string) => (val ? parseInt(val, 10) : undefined),
|
|
84
|
+
},
|
|
85
|
+
validateMaxTxsPerBlock: {
|
|
86
|
+
env: 'VALIDATOR_MAX_TX_PER_BLOCK',
|
|
87
|
+
description: 'Maximum transactions per block for validation. Proposals exceeding this limit are rejected.',
|
|
88
|
+
parseEnv: (val: string) => (val ? parseInt(val, 10) : undefined),
|
|
89
|
+
},
|
|
90
|
+
validateMaxTxsPerCheckpoint: {
|
|
91
|
+
env: 'VALIDATOR_MAX_TX_PER_CHECKPOINT',
|
|
92
|
+
description: 'Maximum transactions per checkpoint for validation. Proposals exceeding this limit are rejected.',
|
|
93
|
+
parseEnv: (val: string) => (val ? parseInt(val, 10) : undefined),
|
|
94
|
+
},
|
|
95
|
+
...localSignerConfigMappings,
|
|
80
96
|
...validatorHASignerConfigMappings,
|
|
81
97
|
};
|
|
82
98
|
|
|
@@ -11,7 +11,6 @@ import type { EthAddress } from '@aztec/foundation/eth-address';
|
|
|
11
11
|
import type { Signature } from '@aztec/foundation/eth-signature';
|
|
12
12
|
import { createLogger } from '@aztec/foundation/log';
|
|
13
13
|
import type { CommitteeAttestationsAndSigners } from '@aztec/stdlib/block';
|
|
14
|
-
import type { CreateCheckpointProposalLastBlockData } from '@aztec/stdlib/interfaces/server';
|
|
15
14
|
import {
|
|
16
15
|
BlockProposal,
|
|
17
16
|
type BlockProposalOptions,
|
|
@@ -86,7 +85,7 @@ export class ValidationService {
|
|
|
86
85
|
*
|
|
87
86
|
* @param checkpointHeader - The checkpoint header containing aggregated data
|
|
88
87
|
* @param archive - The archive of the checkpoint
|
|
89
|
-
* @param
|
|
88
|
+
* @param lastBlockProposal - Signed block proposal for the last block in the checkpoint, or undefined
|
|
90
89
|
* @param proposerAttesterAddress - The address of the proposer
|
|
91
90
|
* @param options - Checkpoint proposal options
|
|
92
91
|
*
|
|
@@ -95,13 +94,16 @@ export class ValidationService {
|
|
|
95
94
|
public createCheckpointProposal(
|
|
96
95
|
checkpointHeader: CheckpointHeader,
|
|
97
96
|
archive: Fr,
|
|
98
|
-
|
|
97
|
+
feeAssetPriceModifier: bigint,
|
|
98
|
+
lastBlockProposal: BlockProposal | undefined,
|
|
99
99
|
proposerAttesterAddress: EthAddress | undefined,
|
|
100
100
|
options: CheckpointProposalOptions,
|
|
101
101
|
): Promise<CheckpointProposal> {
|
|
102
|
-
// For testing: change the archive to trigger state_mismatch validation failure
|
|
102
|
+
// For testing: change the archive to trigger state_mismatch validation failure.
|
|
103
|
+
// If there's a last block proposal, use its (already invalid) archive to keep signatures consistent
|
|
104
|
+
// so P2P validation passes and the slasher can detect the offense.
|
|
103
105
|
if (options.broadcastInvalidCheckpointProposal) {
|
|
104
|
-
archive = Fr.random();
|
|
106
|
+
archive = lastBlockProposal?.archiveRoot ?? Fr.random();
|
|
105
107
|
this.log.warn(`Creating INVALID checkpoint proposal for slot ${checkpointHeader.slotNumber}`);
|
|
106
108
|
}
|
|
107
109
|
|
|
@@ -111,15 +113,13 @@ export class ValidationService {
|
|
|
111
113
|
return this.keyStore.signMessageWithAddress(address, payload, context);
|
|
112
114
|
};
|
|
113
115
|
|
|
114
|
-
|
|
115
|
-
|
|
116
|
-
|
|
117
|
-
|
|
118
|
-
|
|
119
|
-
|
|
120
|
-
|
|
121
|
-
|
|
122
|
-
return CheckpointProposal.createProposalFromSigner(checkpointHeader, archive, lastBlock, payloadSigner);
|
|
116
|
+
return CheckpointProposal.createProposalFromSigner(
|
|
117
|
+
checkpointHeader,
|
|
118
|
+
archive,
|
|
119
|
+
feeAssetPriceModifier,
|
|
120
|
+
lastBlockProposal,
|
|
121
|
+
payloadSigner,
|
|
122
|
+
);
|
|
123
123
|
}
|
|
124
124
|
|
|
125
125
|
/**
|
|
@@ -137,22 +137,16 @@ export class ValidationService {
|
|
|
137
137
|
attestors: EthAddress[],
|
|
138
138
|
): Promise<CheckpointAttestation[]> {
|
|
139
139
|
// Create the attestation payload from the checkpoint proposal
|
|
140
|
-
const payload = new ConsensusPayload(proposal.checkpointHeader, proposal.archive);
|
|
140
|
+
const payload = new ConsensusPayload(proposal.checkpointHeader, proposal.archive, proposal.feeAssetPriceModifier);
|
|
141
141
|
const buf = Buffer32.fromBuffer(
|
|
142
142
|
keccak256(payload.getPayloadToSign(SignatureDomainSeparator.checkpointAttestation)),
|
|
143
143
|
);
|
|
144
144
|
|
|
145
145
|
// TODO(spy/ha): Use checkpointNumber instead of blockNumber once CheckpointHeader includes it.
|
|
146
|
-
//
|
|
146
|
+
// CheckpointProposalCore doesn't have lastBlock info, so use 0 as a proxy.
|
|
147
147
|
// blockNumber is NOT used for the primary key so it's safe to use here.
|
|
148
148
|
// See CheckpointHeader TODO and SigningContext types documentation.
|
|
149
|
-
|
|
150
|
-
try {
|
|
151
|
-
blockNumber = proposal.blockNumber;
|
|
152
|
-
} catch {
|
|
153
|
-
// Checkpoint proposal may not have lastBlock, use 0 as fallback
|
|
154
|
-
blockNumber = BlockNumber(0);
|
|
155
|
-
}
|
|
149
|
+
const blockNumber = BlockNumber(0);
|
|
156
150
|
const context: SigningContext = {
|
|
157
151
|
slot: proposal.slotNumber,
|
|
158
152
|
blockNumber,
|
|
@@ -176,7 +170,7 @@ export class ValidationService {
|
|
|
176
170
|
} else {
|
|
177
171
|
const error = result.reason;
|
|
178
172
|
if (error instanceof DutyAlreadySignedError || error instanceof SlashingProtectionError) {
|
|
179
|
-
this.log.
|
|
173
|
+
this.log.verbose(
|
|
180
174
|
`Attestation for slot ${proposal.slotNumber} by ${attestors[i]} already signed by another High-Availability node`,
|
|
181
175
|
);
|
|
182
176
|
// Continue with remaining attestors
|
package/src/factory.ts
CHANGED
|
@@ -7,13 +7,14 @@ import type { L2BlockSink, L2BlockSource } from '@aztec/stdlib/block';
|
|
|
7
7
|
import type { ValidatorClientFullConfig, WorldStateSynchronizer } from '@aztec/stdlib/interfaces/server';
|
|
8
8
|
import type { L1ToL2MessageSource } from '@aztec/stdlib/messaging';
|
|
9
9
|
import type { TelemetryClient } from '@aztec/telemetry-client';
|
|
10
|
+
import type { SlashingProtectionDatabase } from '@aztec/validator-ha-signer/types';
|
|
10
11
|
|
|
11
|
-
import { BlockProposalHandler } from './block_proposal_handler.js';
|
|
12
12
|
import type { FullNodeCheckpointsBuilder } from './checkpoint_builder.js';
|
|
13
13
|
import { ValidatorMetrics } from './metrics.js';
|
|
14
|
+
import { ProposalHandler } from './proposal_handler.js';
|
|
14
15
|
import { ValidatorClient } from './validator.js';
|
|
15
16
|
|
|
16
|
-
export function
|
|
17
|
+
export function createProposalHandler(
|
|
17
18
|
config: ValidatorClientFullConfig,
|
|
18
19
|
deps: {
|
|
19
20
|
checkpointsBuilder: FullNodeCheckpointsBuilder;
|
|
@@ -22,6 +23,7 @@ export function createBlockProposalHandler(
|
|
|
22
23
|
l1ToL2MessageSource: L1ToL2MessageSource;
|
|
23
24
|
p2pClient: P2PClient;
|
|
24
25
|
epochCache: EpochCache;
|
|
26
|
+
blobClient: BlobClientInterface;
|
|
25
27
|
dateProvider: DateProvider;
|
|
26
28
|
telemetry: TelemetryClient;
|
|
27
29
|
},
|
|
@@ -29,8 +31,9 @@ export function createBlockProposalHandler(
|
|
|
29
31
|
const metrics = new ValidatorMetrics(deps.telemetry);
|
|
30
32
|
const blockProposalValidator = new BlockProposalValidator(deps.epochCache, {
|
|
31
33
|
txsPermitted: !config.disableTransactions,
|
|
34
|
+
maxTxsPerBlock: config.validateMaxTxsPerBlock ?? config.validateMaxTxsPerCheckpoint,
|
|
32
35
|
});
|
|
33
|
-
return new
|
|
36
|
+
return new ProposalHandler(
|
|
34
37
|
deps.checkpointsBuilder,
|
|
35
38
|
deps.worldState,
|
|
36
39
|
deps.blockSource,
|
|
@@ -39,6 +42,7 @@ export function createBlockProposalHandler(
|
|
|
39
42
|
blockProposalValidator,
|
|
40
43
|
deps.epochCache,
|
|
41
44
|
config,
|
|
45
|
+
deps.blobClient,
|
|
42
46
|
metrics,
|
|
43
47
|
deps.dateProvider,
|
|
44
48
|
deps.telemetry,
|
|
@@ -58,6 +62,7 @@ export function createValidatorClient(
|
|
|
58
62
|
epochCache: EpochCache;
|
|
59
63
|
keyStoreManager: KeystoreManager | undefined;
|
|
60
64
|
blobClient: BlobClientInterface;
|
|
65
|
+
slashingProtectionDb?: SlashingProtectionDatabase;
|
|
61
66
|
},
|
|
62
67
|
) {
|
|
63
68
|
if (config.disableValidator || !deps.keyStoreManager) {
|
|
@@ -78,5 +83,6 @@ export function createValidatorClient(
|
|
|
78
83
|
deps.blobClient,
|
|
79
84
|
deps.dateProvider,
|
|
80
85
|
deps.telemetry,
|
|
86
|
+
deps.slashingProtectionDb,
|
|
81
87
|
);
|
|
82
88
|
}
|
package/src/index.ts
CHANGED
|
@@ -1,7 +1,6 @@
|
|
|
1
|
-
export * from './
|
|
1
|
+
export * from './proposal_handler.js';
|
|
2
2
|
export * from './checkpoint_builder.js';
|
|
3
3
|
export * from './config.js';
|
|
4
4
|
export * from './factory.js';
|
|
5
5
|
export * from './validator.js';
|
|
6
6
|
export * from './key_store/index.js';
|
|
7
|
-
export * from './tx_validator/index.js';
|
|
@@ -240,7 +240,7 @@ export class HAKeyStore implements ExtendedValidatorKeyStore {
|
|
|
240
240
|
}
|
|
241
241
|
|
|
242
242
|
if (error instanceof SlashingProtectionError) {
|
|
243
|
-
this.log.
|
|
243
|
+
this.log.info(`Duty already signed by another node with different payload`, {
|
|
244
244
|
dutyType: context.dutyType,
|
|
245
245
|
slot: context.slot,
|
|
246
246
|
existingMessageHash: error.existingMessageHash,
|
package/src/metrics.ts
CHANGED
|
@@ -1,3 +1,5 @@
|
|
|
1
|
+
import type { EpochNumber } from '@aztec/foundation/branded-types';
|
|
2
|
+
import type { EthAddress } from '@aztec/foundation/eth-address';
|
|
1
3
|
import type { BlockProposal } from '@aztec/stdlib/p2p';
|
|
2
4
|
import {
|
|
3
5
|
Attributes,
|
|
@@ -9,13 +11,15 @@ import {
|
|
|
9
11
|
createUpDownCounterWithDefault,
|
|
10
12
|
} from '@aztec/telemetry-client';
|
|
11
13
|
|
|
12
|
-
import type { BlockProposalValidationFailureReason } from './
|
|
14
|
+
import type { BlockProposalValidationFailureReason } from './proposal_handler.js';
|
|
13
15
|
|
|
14
16
|
export class ValidatorMetrics {
|
|
15
17
|
private failedReexecutionCounter: UpDownCounter;
|
|
16
18
|
private successfulAttestationsCount: UpDownCounter;
|
|
17
19
|
private failedAttestationsBadProposalCount: UpDownCounter;
|
|
18
20
|
private failedAttestationsNodeIssueCount: UpDownCounter;
|
|
21
|
+
private currentEpoch: Gauge;
|
|
22
|
+
private attestedEpochCount: UpDownCounter;
|
|
19
23
|
|
|
20
24
|
private reexMana: Histogram;
|
|
21
25
|
private reexTx: Histogram;
|
|
@@ -64,6 +68,10 @@ export class ValidatorMetrics {
|
|
|
64
68
|
},
|
|
65
69
|
);
|
|
66
70
|
|
|
71
|
+
this.currentEpoch = meter.createGauge(Metrics.VALIDATOR_CURRENT_EPOCH);
|
|
72
|
+
|
|
73
|
+
this.attestedEpochCount = createUpDownCounterWithDefault(meter, Metrics.VALIDATOR_ATTESTED_EPOCH_COUNT);
|
|
74
|
+
|
|
67
75
|
this.reexMana = meter.createHistogram(Metrics.VALIDATOR_RE_EXECUTION_MANA);
|
|
68
76
|
|
|
69
77
|
this.reexTx = meter.createHistogram(Metrics.VALIDATOR_RE_EXECUTION_TX_COUNT);
|
|
@@ -110,4 +118,14 @@ export class ValidatorMetrics {
|
|
|
110
118
|
[Attributes.IS_COMMITTEE_MEMBER]: inCommittee,
|
|
111
119
|
});
|
|
112
120
|
}
|
|
121
|
+
|
|
122
|
+
/** Update the gauge tracking the current epoch number (proxy for total epochs elapsed). */
|
|
123
|
+
public setCurrentEpoch(epoch: EpochNumber) {
|
|
124
|
+
this.currentEpoch.record(Number(epoch));
|
|
125
|
+
}
|
|
126
|
+
|
|
127
|
+
/** Increment the count of epochs in which the given attester submitted at least one attestation. */
|
|
128
|
+
public incAttestedEpochCount(attester: EthAddress) {
|
|
129
|
+
this.attestedEpochCount.add(1, { [Attributes.ATTESTER_ADDRESS]: attester.toString() });
|
|
130
|
+
}
|
|
113
131
|
}
|