@aztec/sequencer-client 0.0.1-commit.d3ec352c → 0.0.1-commit.e3c1de76

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 (96) hide show
  1. package/dest/client/sequencer-client.d.ts +12 -12
  2. package/dest/client/sequencer-client.d.ts.map +1 -1
  3. package/dest/client/sequencer-client.js +33 -26
  4. package/dest/config.d.ts +12 -5
  5. package/dest/config.d.ts.map +1 -1
  6. package/dest/config.js +73 -30
  7. package/dest/global_variable_builder/global_builder.d.ts +21 -12
  8. package/dest/global_variable_builder/global_builder.d.ts.map +1 -1
  9. package/dest/global_variable_builder/global_builder.js +51 -41
  10. package/dest/index.d.ts +2 -3
  11. package/dest/index.d.ts.map +1 -1
  12. package/dest/index.js +1 -2
  13. package/dest/publisher/config.d.ts +7 -4
  14. package/dest/publisher/config.d.ts.map +1 -1
  15. package/dest/publisher/config.js +9 -3
  16. package/dest/publisher/sequencer-publisher-factory.d.ts +5 -4
  17. package/dest/publisher/sequencer-publisher-factory.d.ts.map +1 -1
  18. package/dest/publisher/sequencer-publisher-factory.js +1 -1
  19. package/dest/publisher/sequencer-publisher-metrics.d.ts +3 -3
  20. package/dest/publisher/sequencer-publisher-metrics.d.ts.map +1 -1
  21. package/dest/publisher/sequencer-publisher-metrics.js +23 -86
  22. package/dest/publisher/sequencer-publisher.d.ts +48 -41
  23. package/dest/publisher/sequencer-publisher.d.ts.map +1 -1
  24. package/dest/publisher/sequencer-publisher.js +600 -129
  25. package/dest/sequencer/checkpoint_proposal_job.d.ts +96 -0
  26. package/dest/sequencer/checkpoint_proposal_job.d.ts.map +1 -0
  27. package/dest/sequencer/checkpoint_proposal_job.js +1192 -0
  28. package/dest/sequencer/checkpoint_voter.d.ts +35 -0
  29. package/dest/sequencer/checkpoint_voter.d.ts.map +1 -0
  30. package/dest/sequencer/checkpoint_voter.js +109 -0
  31. package/dest/sequencer/config.d.ts +3 -2
  32. package/dest/sequencer/config.d.ts.map +1 -1
  33. package/dest/sequencer/events.d.ts +46 -0
  34. package/dest/sequencer/events.d.ts.map +1 -0
  35. package/dest/sequencer/events.js +1 -0
  36. package/dest/sequencer/index.d.ts +4 -2
  37. package/dest/sequencer/index.d.ts.map +1 -1
  38. package/dest/sequencer/index.js +3 -1
  39. package/dest/sequencer/metrics.d.ts +23 -3
  40. package/dest/sequencer/metrics.d.ts.map +1 -1
  41. package/dest/sequencer/metrics.js +143 -70
  42. package/dest/sequencer/sequencer.d.ts +107 -131
  43. package/dest/sequencer/sequencer.d.ts.map +1 -1
  44. package/dest/sequencer/sequencer.js +690 -602
  45. package/dest/sequencer/timetable.d.ts +54 -14
  46. package/dest/sequencer/timetable.d.ts.map +1 -1
  47. package/dest/sequencer/timetable.js +148 -59
  48. package/dest/sequencer/types.d.ts +3 -0
  49. package/dest/sequencer/types.d.ts.map +1 -0
  50. package/dest/sequencer/types.js +1 -0
  51. package/dest/sequencer/utils.d.ts +14 -8
  52. package/dest/sequencer/utils.d.ts.map +1 -1
  53. package/dest/sequencer/utils.js +7 -4
  54. package/dest/test/index.d.ts +4 -3
  55. package/dest/test/index.d.ts.map +1 -1
  56. package/dest/test/mock_checkpoint_builder.d.ts +95 -0
  57. package/dest/test/mock_checkpoint_builder.d.ts.map +1 -0
  58. package/dest/test/mock_checkpoint_builder.js +220 -0
  59. package/dest/test/utils.d.ts +53 -0
  60. package/dest/test/utils.d.ts.map +1 -0
  61. package/dest/test/utils.js +103 -0
  62. package/package.json +30 -28
  63. package/src/client/sequencer-client.ts +31 -42
  64. package/src/config.ts +78 -34
  65. package/src/global_variable_builder/global_builder.ts +63 -59
  66. package/src/index.ts +1 -7
  67. package/src/publisher/config.ts +12 -9
  68. package/src/publisher/sequencer-publisher-factory.ts +5 -4
  69. package/src/publisher/sequencer-publisher-metrics.ts +19 -71
  70. package/src/publisher/sequencer-publisher.ts +293 -170
  71. package/src/sequencer/README.md +531 -0
  72. package/src/sequencer/checkpoint_proposal_job.ts +874 -0
  73. package/src/sequencer/checkpoint_voter.ts +130 -0
  74. package/src/sequencer/config.ts +2 -1
  75. package/src/sequencer/events.ts +27 -0
  76. package/src/sequencer/index.ts +3 -1
  77. package/src/sequencer/metrics.ts +190 -78
  78. package/src/sequencer/sequencer.ts +430 -804
  79. package/src/sequencer/timetable.ts +173 -79
  80. package/src/sequencer/types.ts +6 -0
  81. package/src/sequencer/utils.ts +18 -9
  82. package/src/test/index.ts +3 -2
  83. package/src/test/mock_checkpoint_builder.ts +309 -0
  84. package/src/test/utils.ts +164 -0
  85. package/dest/sequencer/block_builder.d.ts +0 -28
  86. package/dest/sequencer/block_builder.d.ts.map +0 -1
  87. package/dest/sequencer/block_builder.js +0 -134
  88. package/dest/tx_validator/nullifier_cache.d.ts +0 -14
  89. package/dest/tx_validator/nullifier_cache.d.ts.map +0 -1
  90. package/dest/tx_validator/nullifier_cache.js +0 -24
  91. package/dest/tx_validator/tx_validator_factory.d.ts +0 -18
  92. package/dest/tx_validator/tx_validator_factory.d.ts.map +0 -1
  93. package/dest/tx_validator/tx_validator_factory.js +0 -53
  94. package/src/sequencer/block_builder.ts +0 -222
  95. package/src/tx_validator/nullifier_cache.ts +0 -30
  96. package/src/tx_validator/tx_validator_factory.ts +0 -133
@@ -0,0 +1,309 @@
1
+ import { type BlockNumber, CheckpointNumber } from '@aztec/foundation/branded-types';
2
+ import { Fr } from '@aztec/foundation/curves/bn254';
3
+ import { L2Block } from '@aztec/stdlib/block';
4
+ import { Checkpoint } from '@aztec/stdlib/checkpoint';
5
+ import { Gas } from '@aztec/stdlib/gas';
6
+ import type {
7
+ FullNodeBlockBuilderConfig,
8
+ ICheckpointBlockBuilder,
9
+ ICheckpointsBuilder,
10
+ MerkleTreeWriteOperations,
11
+ PublicProcessorLimits,
12
+ } from '@aztec/stdlib/interfaces/server';
13
+ import { CheckpointHeader } from '@aztec/stdlib/rollup';
14
+ import { makeAppendOnlyTreeSnapshot } from '@aztec/stdlib/testing';
15
+ import type { CheckpointGlobalVariables, Tx } from '@aztec/stdlib/tx';
16
+ import type { BuildBlockInCheckpointResult } from '@aztec/validator-client';
17
+
18
+ /**
19
+ * A fake CheckpointBuilder for testing that implements the same interface as the real one.
20
+ * Can be seeded with blocks to return sequentially on each `buildBlock` call.
21
+ */
22
+ export class MockCheckpointBuilder implements ICheckpointBlockBuilder {
23
+ private blocks: L2Block[] = [];
24
+ private builtBlocks: L2Block[] = [];
25
+ private usedTxsPerBlock: Tx[][] = [];
26
+ private blockIndex = 0;
27
+
28
+ /** Optional function to dynamically provide the block (alternative to seedBlocks) */
29
+ private blockProvider: (() => L2Block) | undefined = undefined;
30
+
31
+ /** Track calls for assertions */
32
+ public buildBlockCalls: Array<{
33
+ blockNumber: BlockNumber;
34
+ timestamp: bigint;
35
+ opts: PublicProcessorLimits;
36
+ }> = [];
37
+ /** Track all consumed transaction hashes across buildBlock calls */
38
+ public consumedTxHashes: Set<string> = new Set();
39
+ public completeCheckpointCalled = false;
40
+ public getCheckpointCalled = false;
41
+
42
+ /** Set to an error to make buildBlock throw on next call */
43
+ public errorOnBuild: Error | undefined = undefined;
44
+
45
+ constructor(
46
+ private readonly constants: CheckpointGlobalVariables,
47
+ private readonly checkpointNumber: CheckpointNumber,
48
+ ) {}
49
+
50
+ /** Seed the builder with blocks to return on successive buildBlock calls */
51
+ seedBlocks(blocks: L2Block[], usedTxsPerBlock?: Tx[][]): this {
52
+ this.blocks = blocks;
53
+ this.usedTxsPerBlock = usedTxsPerBlock ?? blocks.map(() => []);
54
+ this.blockIndex = 0;
55
+ this.blockProvider = undefined;
56
+ return this;
57
+ }
58
+
59
+ /**
60
+ * Set a function that provides blocks dynamically.
61
+ * Useful for tests where the block is determined at call time (e.g., sequencer tests).
62
+ */
63
+ setBlockProvider(provider: () => L2Block): this {
64
+ this.blockProvider = provider;
65
+ this.blocks = [];
66
+ return this;
67
+ }
68
+
69
+ getConstantData(): CheckpointGlobalVariables {
70
+ return this.constants;
71
+ }
72
+
73
+ async buildBlock(
74
+ pendingTxs: Iterable<Tx> | AsyncIterable<Tx>,
75
+ blockNumber: BlockNumber,
76
+ timestamp: bigint,
77
+ opts: PublicProcessorLimits,
78
+ ): Promise<BuildBlockInCheckpointResult> {
79
+ this.buildBlockCalls.push({ blockNumber, timestamp, opts });
80
+
81
+ if (this.errorOnBuild) {
82
+ throw this.errorOnBuild;
83
+ }
84
+
85
+ let block: L2Block;
86
+ let usedTxs: Tx[];
87
+
88
+ if (this.blockProvider) {
89
+ // Dynamic mode: get block from provider
90
+ block = this.blockProvider();
91
+ usedTxs = [];
92
+ this.builtBlocks.push(block);
93
+ } else {
94
+ // Seeded mode: get block from pre-seeded list
95
+ block = this.blocks[this.blockIndex];
96
+ usedTxs = this.usedTxsPerBlock[this.blockIndex] ?? [];
97
+ this.blockIndex++;
98
+ this.builtBlocks.push(block);
99
+ }
100
+
101
+ // Check that no pending tx has already been consumed
102
+ for await (const tx of pendingTxs) {
103
+ const hash = tx.getTxHash().toString();
104
+ if (this.consumedTxHashes.has(hash)) {
105
+ throw new Error(`Transaction ${hash} was already consumed in a previous block`);
106
+ }
107
+ }
108
+
109
+ // Add used txs to consumed set
110
+ for (const tx of usedTxs) {
111
+ this.consumedTxHashes.add(tx.getTxHash().toString());
112
+ }
113
+
114
+ return {
115
+ block,
116
+ publicGas: Gas.empty(),
117
+ publicProcessorDuration: 0,
118
+ numTxs: block?.body?.txEffects?.length ?? usedTxs.length,
119
+ usedTxs,
120
+ failedTxs: [],
121
+ usedTxBlobFields: block?.body?.txEffects?.reduce((sum, tx) => sum + tx.getNumBlobFields(), 0) ?? 0,
122
+ };
123
+ }
124
+
125
+ completeCheckpoint(): Promise<Checkpoint> {
126
+ this.completeCheckpointCalled = true;
127
+ const allBlocks = this.blockProvider ? this.builtBlocks : this.blocks;
128
+ const lastBlock = allBlocks[allBlocks.length - 1];
129
+ // Create a CheckpointHeader from the last block's header for testing
130
+ const checkpointHeader = this.createCheckpointHeader(lastBlock);
131
+ return Promise.resolve(
132
+ new Checkpoint(
133
+ makeAppendOnlyTreeSnapshot(lastBlock.header.globalVariables.blockNumber + 1),
134
+ checkpointHeader,
135
+ allBlocks,
136
+ this.checkpointNumber,
137
+ ),
138
+ );
139
+ }
140
+
141
+ getCheckpoint(): Promise<Checkpoint> {
142
+ this.getCheckpointCalled = true;
143
+ const builtBlocks = this.blockProvider ? this.builtBlocks : this.blocks.slice(0, this.blockIndex);
144
+ const lastBlock = builtBlocks[builtBlocks.length - 1];
145
+ if (!lastBlock) {
146
+ throw new Error('No blocks built yet');
147
+ }
148
+ // Create a CheckpointHeader from the last block's header for testing
149
+ const checkpointHeader = this.createCheckpointHeader(lastBlock);
150
+ return Promise.resolve(
151
+ new Checkpoint(
152
+ makeAppendOnlyTreeSnapshot(lastBlock.header.globalVariables.blockNumber + 1),
153
+ checkpointHeader,
154
+ builtBlocks,
155
+ this.checkpointNumber,
156
+ ),
157
+ );
158
+ }
159
+
160
+ /**
161
+ * Creates a CheckpointHeader from a block's header for testing.
162
+ * This is a simplified version that creates a minimal CheckpointHeader.
163
+ */
164
+ private createCheckpointHeader(block: L2Block): CheckpointHeader {
165
+ const header = block.header;
166
+ const gv = header.globalVariables;
167
+ return CheckpointHeader.empty({
168
+ lastArchiveRoot: header.lastArchive.root,
169
+ blockHeadersHash: Fr.random(), // Use random for testing
170
+ slotNumber: gv.slotNumber,
171
+ timestamp: gv.timestamp,
172
+ coinbase: gv.coinbase,
173
+ feeRecipient: gv.feeRecipient,
174
+ gasFees: gv.gasFees,
175
+ totalManaUsed: header.totalManaUsed,
176
+ });
177
+ }
178
+
179
+ /** Reset for reuse in another test */
180
+ reset(): void {
181
+ this.blocks = [];
182
+ this.builtBlocks = [];
183
+ this.usedTxsPerBlock = [];
184
+ this.blockIndex = 0;
185
+ this.buildBlockCalls = [];
186
+ this.consumedTxHashes.clear();
187
+ this.completeCheckpointCalled = false;
188
+ this.getCheckpointCalled = false;
189
+ this.errorOnBuild = undefined;
190
+ this.blockProvider = undefined;
191
+ }
192
+ }
193
+
194
+ /**
195
+ * A fake CheckpointsBuilder (factory) for testing that implements the same interface
196
+ * as FullNodeCheckpointsBuilder. Returns MockCheckpointBuilder instances.
197
+ * Does NOT use jest mocks - this is a proper test double.
198
+ */
199
+ export class MockCheckpointsBuilder implements ICheckpointsBuilder {
200
+ private checkpointBuilder: MockCheckpointBuilder | undefined;
201
+
202
+ /** Track calls for assertions */
203
+ public startCheckpointCalls: Array<{
204
+ checkpointNumber: CheckpointNumber;
205
+ constants: CheckpointGlobalVariables;
206
+ l1ToL2Messages: Fr[];
207
+ previousCheckpointOutHashes: Fr[];
208
+ }> = [];
209
+ public openCheckpointCalls: Array<{
210
+ checkpointNumber: CheckpointNumber;
211
+ constants: CheckpointGlobalVariables;
212
+ l1ToL2Messages: Fr[];
213
+ previousCheckpointOutHashes: Fr[];
214
+ existingBlocks: L2Block[];
215
+ }> = [];
216
+ public updateConfigCalls: Array<Partial<FullNodeBlockBuilderConfig>> = [];
217
+
218
+ /**
219
+ * Set the MockCheckpointBuilder to return from startCheckpoint.
220
+ * Must be called before startCheckpoint is invoked.
221
+ */
222
+ setCheckpointBuilder(builder: MockCheckpointBuilder): this {
223
+ this.checkpointBuilder = builder;
224
+ return this;
225
+ }
226
+
227
+ /**
228
+ * Creates a new MockCheckpointBuilder with the given constants.
229
+ * Convenience method that creates and sets the builder in one call.
230
+ */
231
+ createCheckpointBuilder(
232
+ constants: CheckpointGlobalVariables,
233
+ checkpointNumber: CheckpointNumber,
234
+ ): MockCheckpointBuilder {
235
+ this.checkpointBuilder = new MockCheckpointBuilder(constants, checkpointNumber);
236
+ return this.checkpointBuilder;
237
+ }
238
+
239
+ /** Get the current checkpoint builder (for assertions) */
240
+ getCheckpointBuilder(): MockCheckpointBuilder | undefined {
241
+ return this.checkpointBuilder;
242
+ }
243
+
244
+ getConfig(): FullNodeBlockBuilderConfig {
245
+ return {
246
+ l1GenesisTime: 0n,
247
+ slotDuration: 24,
248
+ l1ChainId: 1,
249
+ rollupVersion: 1,
250
+ };
251
+ }
252
+
253
+ updateConfig(config: Partial<FullNodeBlockBuilderConfig>): void {
254
+ this.updateConfigCalls.push(config);
255
+ }
256
+
257
+ startCheckpoint(
258
+ checkpointNumber: CheckpointNumber,
259
+ constants: CheckpointGlobalVariables,
260
+ l1ToL2Messages: Fr[],
261
+ previousCheckpointOutHashes: Fr[],
262
+ _fork: MerkleTreeWriteOperations,
263
+ ): Promise<ICheckpointBlockBuilder> {
264
+ this.startCheckpointCalls.push({ checkpointNumber, constants, l1ToL2Messages, previousCheckpointOutHashes });
265
+
266
+ if (!this.checkpointBuilder) {
267
+ // Auto-create a builder if none was set
268
+ this.checkpointBuilder = new MockCheckpointBuilder(constants, checkpointNumber);
269
+ }
270
+
271
+ return Promise.resolve(this.checkpointBuilder);
272
+ }
273
+
274
+ openCheckpoint(
275
+ checkpointNumber: CheckpointNumber,
276
+ constants: CheckpointGlobalVariables,
277
+ l1ToL2Messages: Fr[],
278
+ previousCheckpointOutHashes: Fr[],
279
+ _fork: MerkleTreeWriteOperations,
280
+ existingBlocks: L2Block[] = [],
281
+ ): Promise<ICheckpointBlockBuilder> {
282
+ this.openCheckpointCalls.push({
283
+ checkpointNumber,
284
+ constants,
285
+ l1ToL2Messages,
286
+ previousCheckpointOutHashes,
287
+ existingBlocks,
288
+ });
289
+
290
+ if (!this.checkpointBuilder) {
291
+ // Auto-create a builder if none was set
292
+ this.checkpointBuilder = new MockCheckpointBuilder(constants, checkpointNumber);
293
+ }
294
+
295
+ return Promise.resolve(this.checkpointBuilder);
296
+ }
297
+
298
+ getFork(_blockNumber: BlockNumber): Promise<MerkleTreeWriteOperations> {
299
+ throw new Error('MockCheckpointsBuilder.getFork not implemented');
300
+ }
301
+
302
+ /** Reset for reuse in another test */
303
+ reset(): void {
304
+ this.checkpointBuilder = undefined;
305
+ this.startCheckpointCalls = [];
306
+ this.openCheckpointCalls = [];
307
+ this.updateConfigCalls = [];
308
+ }
309
+ }
@@ -0,0 +1,164 @@
1
+ import { Body } from '@aztec/aztec.js/block';
2
+ import { CheckpointNumber, IndexWithinCheckpoint } from '@aztec/foundation/branded-types';
3
+ import { times } from '@aztec/foundation/collection';
4
+ import { Secp256k1Signer } from '@aztec/foundation/crypto/secp256k1-signer';
5
+ import { Fr } from '@aztec/foundation/curves/bn254';
6
+ import type { EthAddress } from '@aztec/foundation/eth-address';
7
+ import { Signature } from '@aztec/foundation/eth-signature';
8
+ import type { P2P } from '@aztec/p2p';
9
+ import { PublicDataWrite } from '@aztec/stdlib/avm';
10
+ import { CommitteeAttestation, L2Block } from '@aztec/stdlib/block';
11
+ import { BlockProposal, CheckpointAttestation, CheckpointProposal, ConsensusPayload } from '@aztec/stdlib/p2p';
12
+ import { CheckpointHeader } from '@aztec/stdlib/rollup';
13
+ import { makeAppendOnlyTreeSnapshot, mockTxForRollup } from '@aztec/stdlib/testing';
14
+ import { BlockHeader, GlobalVariables, type Tx, makeProcessedTxFromPrivateOnlyTx } from '@aztec/stdlib/tx';
15
+
16
+ import type { MockProxy } from 'jest-mock-extended';
17
+
18
+ // Re-export mock classes from their dedicated file
19
+ export { MockCheckpointBuilder, MockCheckpointsBuilder } from './mock_checkpoint_builder.js';
20
+
21
+ /**
22
+ * Creates a mock transaction with a specific seed for deterministic testing
23
+ */
24
+ export async function makeTx(seed?: number, chainId?: Fr): Promise<Tx> {
25
+ const tx = await mockTxForRollup(seed);
26
+ if (chainId) {
27
+ tx.data.constants.txContext.chainId = chainId;
28
+ }
29
+ return tx;
30
+ }
31
+
32
+ /**
33
+ * Creates an L2Block from transactions and global variables
34
+ */
35
+ export async function makeBlock(txs: Tx[], globalVariables: GlobalVariables): Promise<L2Block> {
36
+ const processedTxs = await Promise.all(
37
+ txs.map(tx =>
38
+ makeProcessedTxFromPrivateOnlyTx(tx, Fr.ZERO, new PublicDataWrite(Fr.random(), Fr.random()), globalVariables),
39
+ ),
40
+ );
41
+ const body = new Body(processedTxs.map(tx => tx.txEffect));
42
+ const header = BlockHeader.empty({ globalVariables });
43
+ const archive = makeAppendOnlyTreeSnapshot(globalVariables.blockNumber + 1);
44
+ return new L2Block(
45
+ archive,
46
+ header,
47
+ body,
48
+ CheckpointNumber.fromBlockNumber(globalVariables.blockNumber),
49
+ IndexWithinCheckpoint(0),
50
+ );
51
+ }
52
+
53
+ /**
54
+ * Mocks the P2P client to return specific pending transactions
55
+ */
56
+ export function mockPendingTxs(p2p: MockProxy<P2P>, txs: Tx[]): void {
57
+ p2p.getPendingTxCount.mockResolvedValue(txs.length);
58
+ p2p.iteratePendingTxs.mockImplementation(() => mockTxIterator(Promise.resolve(txs)));
59
+ }
60
+
61
+ /**
62
+ * Creates an async iterator for transactions
63
+ */
64
+ export async function* mockTxIterator(txs: Promise<Tx[]>): AsyncIterableIterator<Tx> {
65
+ for (const tx of await txs) {
66
+ yield tx;
67
+ }
68
+ }
69
+
70
+ /**
71
+ * Creates mock committee attestations from a signer
72
+ */
73
+ export function createMockSignatures(signer: Secp256k1Signer): CommitteeAttestation[] {
74
+ const mockedSig = Signature.random();
75
+ return [new CommitteeAttestation(signer.address, mockedSig)];
76
+ }
77
+
78
+ /**
79
+ * Creates a CheckpointHeader from an L2Block for testing purposes.
80
+ * Uses mock values for blockHeadersHash, blobsHash and inHash since L2Block doesn't have these fields.
81
+ */
82
+ function createCheckpointHeaderFromBlock(block: L2Block): CheckpointHeader {
83
+ const gv = block.header.globalVariables;
84
+ return new CheckpointHeader(
85
+ block.header.lastArchive.root,
86
+ Fr.random(), // blockHeadersHash - mock value for testing
87
+ Fr.random(), // blobsHash - mock value for testing
88
+ Fr.random(), // inHash - mock value for testing
89
+ Fr.random(), // outHash - mock value for testing
90
+ gv.slotNumber,
91
+ gv.timestamp,
92
+ gv.coinbase,
93
+ gv.feeRecipient,
94
+ gv.gasFees,
95
+ block.header.totalManaUsed,
96
+ );
97
+ }
98
+
99
+ /**
100
+ * Creates a block proposal from a block and signature
101
+ */
102
+ export function createBlockProposal(block: L2Block, signature: Signature): BlockProposal {
103
+ const txHashes = block.body.txEffects.map(tx => tx.txHash);
104
+ return new BlockProposal(
105
+ block.header,
106
+ block.indexWithinCheckpoint,
107
+ Fr.ZERO, // inHash - using zero for testing
108
+ block.archive.root,
109
+ txHashes,
110
+ signature,
111
+ );
112
+ }
113
+
114
+ /**
115
+ * Creates a checkpoint proposal from a block and signature
116
+ */
117
+ export function createCheckpointProposal(
118
+ block: L2Block,
119
+ checkpointSignature: Signature,
120
+ blockSignature?: Signature,
121
+ ): CheckpointProposal {
122
+ const txHashes = block.body.txEffects.map(tx => tx.txHash);
123
+ const checkpointHeader = createCheckpointHeaderFromBlock(block);
124
+ return new CheckpointProposal(checkpointHeader, block.archive.root, checkpointSignature, {
125
+ blockHeader: block.header,
126
+ indexWithinCheckpoint: block.indexWithinCheckpoint,
127
+ txHashes,
128
+ signature: blockSignature ?? checkpointSignature, // Use checkpoint signature as block signature if not provided
129
+ });
130
+ }
131
+
132
+ /**
133
+ * Creates a checkpoint attestation from a block and signature.
134
+ * Note: We manually set the sender since we use random signatures in tests.
135
+ * In production, the sender is recovered from the signature.
136
+ */
137
+ export function createCheckpointAttestation(
138
+ block: L2Block,
139
+ signature: Signature,
140
+ sender: EthAddress,
141
+ ): CheckpointAttestation {
142
+ const checkpointHeader = createCheckpointHeaderFromBlock(block);
143
+ const payload = new ConsensusPayload(checkpointHeader, block.archive.root);
144
+ const attestation = new CheckpointAttestation(payload, signature, signature);
145
+ // Set sender directly for testing (bypasses signature recovery)
146
+ (attestation as any).sender = sender;
147
+ return attestation;
148
+ }
149
+
150
+ /**
151
+ * Creates transactions and a block, and mocks P2P to return them.
152
+ * Helper for tests that need to set up a block with transactions.
153
+ */
154
+ export async function setupTxsAndBlock(
155
+ p2p: MockProxy<P2P>,
156
+ globalVariables: GlobalVariables,
157
+ txCount: number,
158
+ chainId: Fr,
159
+ ): Promise<{ txs: Tx[]; block: L2Block }> {
160
+ const txs = await Promise.all(times(txCount, i => makeTx(i + 1, chainId)));
161
+ const block = await makeBlock(txs, globalVariables);
162
+ mockPendingTxs(p2p, txs);
163
+ return { txs, block };
164
+ }
@@ -1,28 +0,0 @@
1
- import { BlockNumber } from '@aztec/foundation/branded-types';
2
- import type { Fr } from '@aztec/foundation/fields';
3
- import { DateProvider } from '@aztec/foundation/timer';
4
- import { PublicProcessor } from '@aztec/simulator/server';
5
- import type { ContractDataSource } from '@aztec/stdlib/contract';
6
- import { type L1RollupConstants } from '@aztec/stdlib/epoch-helpers';
7
- import type { BuildBlockResult, FullNodeBlockBuilderConfig, IFullNodeBlockBuilder, MerkleTreeWriteOperations, PublicProcessorLimits, PublicProcessorValidator, WorldStateSynchronizer } from '@aztec/stdlib/interfaces/server';
8
- import { GlobalVariables, Tx } from '@aztec/stdlib/tx';
9
- import { type TelemetryClient } from '@aztec/telemetry-client';
10
- export declare function buildBlock(pendingTxs: Iterable<Tx> | AsyncIterable<Tx>, l1ToL2Messages: Fr[], newGlobalVariables: GlobalVariables, opts: PublicProcessorLimits | undefined, worldStateFork: MerkleTreeWriteOperations, processor: PublicProcessor, validator: PublicProcessorValidator, l1Constants: Pick<L1RollupConstants, 'l1GenesisTime' | 'slotDuration'>, dateProvider: DateProvider, telemetryClient?: TelemetryClient): Promise<BuildBlockResult>;
11
- export declare class FullNodeBlockBuilder implements IFullNodeBlockBuilder {
12
- private config;
13
- private worldState;
14
- private contractDataSource;
15
- private dateProvider;
16
- private telemetryClient;
17
- constructor(config: FullNodeBlockBuilderConfig, worldState: WorldStateSynchronizer, contractDataSource: ContractDataSource, dateProvider: DateProvider, telemetryClient?: TelemetryClient);
18
- getConfig(): FullNodeBlockBuilderConfig;
19
- updateConfig(config: Partial<FullNodeBlockBuilderConfig>): void;
20
- makeBlockBuilderDeps(globalVariables: GlobalVariables, fork: MerkleTreeWriteOperations): Promise<{
21
- processor: PublicProcessor;
22
- validator: PublicProcessorValidator;
23
- }>;
24
- private syncToPreviousBlock;
25
- buildBlock(pendingTxs: Iterable<Tx> | AsyncIterable<Tx>, l1ToL2Messages: Fr[], globalVariables: GlobalVariables, opts: PublicProcessorLimits, suppliedFork?: MerkleTreeWriteOperations): Promise<BuildBlockResult>;
26
- getFork(blockNumber: BlockNumber): Promise<MerkleTreeWriteOperations>;
27
- }
28
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYmxvY2tfYnVpbGRlci5kLnRzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL3NlcXVlbmNlci9ibG9ja19idWlsZGVyLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUNBLE9BQU8sRUFBRSxXQUFXLEVBQUUsTUFBTSxpQ0FBaUMsQ0FBQztBQUU5RCxPQUFPLEtBQUssRUFBRSxFQUFFLEVBQUUsTUFBTSwwQkFBMEIsQ0FBQztBQUluRCxPQUFPLEVBQUUsWUFBWSxFQUFrQixNQUFNLHlCQUF5QixDQUFDO0FBR3ZFLE9BQU8sRUFHTCxlQUFlLEVBRWhCLE1BQU0seUJBQXlCLENBQUM7QUFFakMsT0FBTyxLQUFLLEVBQUUsa0JBQWtCLEVBQUUsTUFBTSx3QkFBd0IsQ0FBQztBQUNqRSxPQUFPLEVBQUUsS0FBSyxpQkFBaUIsRUFBdUIsTUFBTSw2QkFBNkIsQ0FBQztBQUUxRixPQUFPLEtBQUssRUFDVixnQkFBZ0IsRUFDaEIsMEJBQTBCLEVBQzFCLHFCQUFxQixFQUNyQix5QkFBeUIsRUFDekIscUJBQXFCLEVBQ3JCLHdCQUF3QixFQUN4QixzQkFBc0IsRUFDdkIsTUFBTSxpQ0FBaUMsQ0FBQztBQUN6QyxPQUFPLEVBQUUsZUFBZSxFQUFFLEVBQUUsRUFBRSxNQUFNLGtCQUFrQixDQUFDO0FBQ3ZELE9BQU8sRUFBRSxLQUFLLGVBQWUsRUFBc0IsTUFBTSx5QkFBeUIsQ0FBQztBQU1uRix3QkFBc0IsVUFBVSxDQUM5QixVQUFVLEVBQUUsUUFBUSxDQUFDLEVBQUUsQ0FBQyxHQUFHLGFBQWEsQ0FBQyxFQUFFLENBQUMsRUFDNUMsY0FBYyxFQUFFLEVBQUUsRUFBRSxFQUNwQixrQkFBa0IsRUFBRSxlQUFlLEVBQ25DLElBQUksbUNBQTRCLEVBQ2hDLGNBQWMsRUFBRSx5QkFBeUIsRUFDekMsU0FBUyxFQUFFLGVBQWUsRUFDMUIsU0FBUyxFQUFFLHdCQUF3QixFQUNuQyxXQUFXLEVBQUUsSUFBSSxDQUFDLGlCQUFpQixFQUFFLGVBQWUsR0FBRyxjQUFjLENBQUMsRUFDdEUsWUFBWSxFQUFFLFlBQVksRUFDMUIsZUFBZSxHQUFFLGVBQXNDLEdBQ3RELE9BQU8sQ0FBQyxnQkFBZ0IsQ0FBQyxDQTRDM0I7QUFXRCxxQkFBYSxvQkFBcUIsWUFBVyxxQkFBcUI7SUFFOUQsT0FBTyxDQUFDLE1BQU07SUFDZCxPQUFPLENBQUMsVUFBVTtJQUNsQixPQUFPLENBQUMsa0JBQWtCO0lBQzFCLE9BQU8sQ0FBQyxZQUFZO0lBQ3BCLE9BQU8sQ0FBQyxlQUFlO0lBTHpCLFlBQ1UsTUFBTSxFQUFFLDBCQUEwQixFQUNsQyxVQUFVLEVBQUUsc0JBQXNCLEVBQ2xDLGtCQUFrQixFQUFFLGtCQUFrQixFQUN0QyxZQUFZLEVBQUUsWUFBWSxFQUMxQixlQUFlLEdBQUUsZUFBc0MsRUFDN0Q7SUFFRyxTQUFTLElBQUksMEJBQTBCLENBRTdDO0lBRU0sWUFBWSxDQUFDLE1BQU0sRUFBRSxPQUFPLENBQUMsMEJBQTBCLENBQUMsUUFFOUQ7SUFFWSxvQkFBb0IsQ0FBQyxlQUFlLEVBQUUsZUFBZSxFQUFFLElBQUksRUFBRSx5QkFBeUI7OztPQXlDbEc7WUFFYSxtQkFBbUI7SUFVM0IsVUFBVSxDQUNkLFVBQVUsRUFBRSxRQUFRLENBQUMsRUFBRSxDQUFDLEdBQUcsYUFBYSxDQUFDLEVBQUUsQ0FBQyxFQUM1QyxjQUFjLEVBQUUsRUFBRSxFQUFFLEVBQ3BCLGVBQWUsRUFBRSxlQUFlLEVBQ2hDLElBQUksRUFBRSxxQkFBcUIsRUFDM0IsWUFBWSxDQUFDLEVBQUUseUJBQXlCLEdBQ3ZDLE9BQU8sQ0FBQyxnQkFBZ0IsQ0FBQyxDQXNDM0I7SUFFRCxPQUFPLENBQUMsV0FBVyxFQUFFLFdBQVcsR0FBRyxPQUFPLENBQUMseUJBQXlCLENBQUMsQ0FFcEU7Q0FDRiJ9
@@ -1 +0,0 @@
1
- {"version":3,"file":"block_builder.d.ts","sourceRoot":"","sources":["../../src/sequencer/block_builder.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,WAAW,EAAE,MAAM,iCAAiC,CAAC;AAE9D,OAAO,KAAK,EAAE,EAAE,EAAE,MAAM,0BAA0B,CAAC;AAInD,OAAO,EAAE,YAAY,EAAkB,MAAM,yBAAyB,CAAC;AAGvE,OAAO,EAGL,eAAe,EAEhB,MAAM,yBAAyB,CAAC;AAEjC,OAAO,KAAK,EAAE,kBAAkB,EAAE,MAAM,wBAAwB,CAAC;AACjE,OAAO,EAAE,KAAK,iBAAiB,EAAuB,MAAM,6BAA6B,CAAC;AAE1F,OAAO,KAAK,EACV,gBAAgB,EAChB,0BAA0B,EAC1B,qBAAqB,EACrB,yBAAyB,EACzB,qBAAqB,EACrB,wBAAwB,EACxB,sBAAsB,EACvB,MAAM,iCAAiC,CAAC;AACzC,OAAO,EAAE,eAAe,EAAE,EAAE,EAAE,MAAM,kBAAkB,CAAC;AACvD,OAAO,EAAE,KAAK,eAAe,EAAsB,MAAM,yBAAyB,CAAC;AAMnF,wBAAsB,UAAU,CAC9B,UAAU,EAAE,QAAQ,CAAC,EAAE,CAAC,GAAG,aAAa,CAAC,EAAE,CAAC,EAC5C,cAAc,EAAE,EAAE,EAAE,EACpB,kBAAkB,EAAE,eAAe,EACnC,IAAI,mCAA4B,EAChC,cAAc,EAAE,yBAAyB,EACzC,SAAS,EAAE,eAAe,EAC1B,SAAS,EAAE,wBAAwB,EACnC,WAAW,EAAE,IAAI,CAAC,iBAAiB,EAAE,eAAe,GAAG,cAAc,CAAC,EACtE,YAAY,EAAE,YAAY,EAC1B,eAAe,GAAE,eAAsC,GACtD,OAAO,CAAC,gBAAgB,CAAC,CA4C3B;AAWD,qBAAa,oBAAqB,YAAW,qBAAqB;IAE9D,OAAO,CAAC,MAAM;IACd,OAAO,CAAC,UAAU;IAClB,OAAO,CAAC,kBAAkB;IAC1B,OAAO,CAAC,YAAY;IACpB,OAAO,CAAC,eAAe;IALzB,YACU,MAAM,EAAE,0BAA0B,EAClC,UAAU,EAAE,sBAAsB,EAClC,kBAAkB,EAAE,kBAAkB,EACtC,YAAY,EAAE,YAAY,EAC1B,eAAe,GAAE,eAAsC,EAC7D;IAEG,SAAS,IAAI,0BAA0B,CAE7C;IAEM,YAAY,CAAC,MAAM,EAAE,OAAO,CAAC,0BAA0B,CAAC,QAE9D;IAEY,oBAAoB,CAAC,eAAe,EAAE,eAAe,EAAE,IAAI,EAAE,yBAAyB;;;OAyClG;YAEa,mBAAmB;IAU3B,UAAU,CACd,UAAU,EAAE,QAAQ,CAAC,EAAE,CAAC,GAAG,aAAa,CAAC,EAAE,CAAC,EAC5C,cAAc,EAAE,EAAE,EAAE,EACpB,eAAe,EAAE,eAAe,EAChC,IAAI,EAAE,qBAAqB,EAC3B,YAAY,CAAC,EAAE,yBAAyB,GACvC,OAAO,CAAC,gBAAgB,CAAC,CAsC3B;IAED,OAAO,CAAC,WAAW,EAAE,WAAW,GAAG,OAAO,CAAC,yBAAyB,CAAC,CAEpE;CACF"}
@@ -1,134 +0,0 @@
1
- import { MerkleTreeId } from '@aztec/aztec.js/trees';
2
- import { BlockNumber } from '@aztec/foundation/branded-types';
3
- import { merge, pick } from '@aztec/foundation/collection';
4
- import { createLogger } from '@aztec/foundation/log';
5
- import { retryUntil } from '@aztec/foundation/retry';
6
- import { bufferToHex } from '@aztec/foundation/string';
7
- import { Timer, elapsed } from '@aztec/foundation/timer';
8
- import { getDefaultAllowedSetupFunctions } from '@aztec/p2p/msg_validators';
9
- import { LightweightBlockFactory } from '@aztec/prover-client/block-factory';
10
- import { GuardedMerkleTreeOperations, PublicContractsDB, PublicProcessor, TelemetryCppPublicTxSimulator } from '@aztec/simulator/server';
11
- import { PublicSimulatorConfig } from '@aztec/stdlib/avm';
12
- import { getTimestampForSlot } from '@aztec/stdlib/epoch-helpers';
13
- import { Gas } from '@aztec/stdlib/gas';
14
- import { getTelemetryClient } from '@aztec/telemetry-client';
15
- import { createValidatorForBlockBuilding } from '../tx_validator/tx_validator_factory.js';
16
- const log = createLogger('block-builder');
17
- export async function buildBlock(pendingTxs, l1ToL2Messages, newGlobalVariables, opts = {}, worldStateFork, processor, validator, l1Constants, dateProvider, telemetryClient = getTelemetryClient()) {
18
- const blockBuildingTimer = new Timer();
19
- const blockNumber = newGlobalVariables.blockNumber;
20
- const slot = newGlobalVariables.slotNumber;
21
- const msgCount = l1ToL2Messages.length;
22
- const stateReference = await worldStateFork.getStateReference();
23
- const archiveTree = await worldStateFork.getTreeInfo(MerkleTreeId.ARCHIVE);
24
- log.verbose(`Building block ${blockNumber} for slot ${slot}`, {
25
- slot,
26
- slotStart: new Date(Number(getTimestampForSlot(slot, l1Constants)) * 1000),
27
- now: new Date(dateProvider.now()),
28
- blockNumber,
29
- msgCount,
30
- initialStateReference: stateReference.toInspect(),
31
- initialArchiveRoot: bufferToHex(archiveTree.root),
32
- opts
33
- });
34
- const blockFactory = new LightweightBlockFactory(worldStateFork, telemetryClient);
35
- await blockFactory.startNewBlock(newGlobalVariables, l1ToL2Messages);
36
- const [publicProcessorDuration, [processedTxs, failedTxs, usedTxs]] = await elapsed(()=>processor.process(pendingTxs, opts, validator));
37
- // All real transactions have been added, set the block as full and pad if needed
38
- await blockFactory.addTxs(processedTxs);
39
- const block = await blockFactory.setBlockCompleted();
40
- // How much public gas was processed
41
- const publicGas = processedTxs.reduce((acc, tx)=>acc.add(tx.gasUsed.publicGas), Gas.empty());
42
- const res = {
43
- block,
44
- publicGas,
45
- publicProcessorDuration,
46
- numMsgs: l1ToL2Messages.length,
47
- numTxs: processedTxs.length,
48
- failedTxs: failedTxs,
49
- blockBuildingTimer,
50
- usedTxs
51
- };
52
- log.trace('Built block', res.block.header);
53
- return res;
54
- }
55
- const FullNodeBlockBuilderConfigKeys = [
56
- 'l1GenesisTime',
57
- 'slotDuration',
58
- 'l1ChainId',
59
- 'rollupVersion',
60
- 'txPublicSetupAllowList',
61
- 'fakeProcessingDelayPerTxMs'
62
- ];
63
- export class FullNodeBlockBuilder {
64
- config;
65
- worldState;
66
- contractDataSource;
67
- dateProvider;
68
- telemetryClient;
69
- constructor(config, worldState, contractDataSource, dateProvider, telemetryClient = getTelemetryClient()){
70
- this.config = config;
71
- this.worldState = worldState;
72
- this.contractDataSource = contractDataSource;
73
- this.dateProvider = dateProvider;
74
- this.telemetryClient = telemetryClient;
75
- }
76
- getConfig() {
77
- return pick(this.config, ...FullNodeBlockBuilderConfigKeys);
78
- }
79
- updateConfig(config) {
80
- this.config = merge(this.config, pick(config, ...FullNodeBlockBuilderConfigKeys));
81
- }
82
- async makeBlockBuilderDeps(globalVariables, fork) {
83
- const txPublicSetupAllowList = this.config.txPublicSetupAllowList ?? await getDefaultAllowedSetupFunctions();
84
- const contractsDB = new PublicContractsDB(this.contractDataSource);
85
- const guardedFork = new GuardedMerkleTreeOperations(fork);
86
- const publicTxSimulator = new TelemetryCppPublicTxSimulator(guardedFork, contractsDB, globalVariables, this.telemetryClient, PublicSimulatorConfig.from({
87
- skipFeeEnforcement: false,
88
- collectDebugLogs: false,
89
- collectHints: false,
90
- collectStatistics: false,
91
- collectCallMetadata: false
92
- }));
93
- const processor = new PublicProcessor(globalVariables, guardedFork, contractsDB, publicTxSimulator, this.dateProvider, this.telemetryClient, undefined, this.config);
94
- const validator = createValidatorForBlockBuilding(fork, this.contractDataSource, globalVariables, txPublicSetupAllowList);
95
- return {
96
- processor,
97
- validator
98
- };
99
- }
100
- async syncToPreviousBlock(parentBlockNumber, timeout) {
101
- await retryUntil(()=>this.worldState.syncImmediate(parentBlockNumber, true).then((syncedTo)=>syncedTo >= parentBlockNumber), 'sync to previous block', timeout, 0.1);
102
- log.debug(`Synced to previous block ${parentBlockNumber}`);
103
- }
104
- async buildBlock(pendingTxs, l1ToL2Messages, globalVariables, opts, suppliedFork) {
105
- const parentBlockNumber = BlockNumber(globalVariables.blockNumber - 1);
106
- const syncTimeout = opts.deadline ? (opts.deadline.getTime() - this.dateProvider.now()) / 1000 : undefined;
107
- await this.syncToPreviousBlock(parentBlockNumber, syncTimeout);
108
- const fork = suppliedFork ?? await this.worldState.fork(parentBlockNumber);
109
- try {
110
- const { processor, validator } = await this.makeBlockBuilderDeps(globalVariables, fork);
111
- const res = await buildBlock(pendingTxs, l1ToL2Messages, globalVariables, opts, fork, processor, validator, this.config, this.dateProvider, this.telemetryClient);
112
- return res;
113
- } finally{
114
- // If the fork was supplied, we don't close it.
115
- // Otherwise, we wait a bit to close the fork we just created,
116
- // since the processor may still be working on a dangling tx
117
- // which was interrupted due to the processingDeadline being hit.
118
- if (!suppliedFork) {
119
- // eslint-disable-next-line @typescript-eslint/no-misused-promises
120
- setTimeout(async ()=>{
121
- try {
122
- await fork.close();
123
- } catch (err) {
124
- // This can happen if the sequencer is stopped before we hit this timeout.
125
- log.warn(`Error closing forks for block processing`, err);
126
- }
127
- }, 5000);
128
- }
129
- }
130
- }
131
- getFork(blockNumber) {
132
- return this.worldState.fork(blockNumber);
133
- }
134
- }
@@ -1,14 +0,0 @@
1
- import type { NullifierSource } from '@aztec/p2p';
2
- import type { MerkleTreeReadOperations } from '@aztec/stdlib/interfaces/server';
3
- /**
4
- * Implements a nullifier source by checking a DB and an in-memory collection.
5
- * Intended for validating transactions as they are added to a block.
6
- */
7
- export declare class NullifierCache implements NullifierSource {
8
- private db;
9
- nullifiers: Set<string>;
10
- constructor(db: MerkleTreeReadOperations);
11
- nullifiersExist(nullifiers: Buffer[]): Promise<boolean[]>;
12
- addNullifiers(nullifiers: Buffer[]): void;
13
- }
14
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibnVsbGlmaWVyX2NhY2hlLmQudHMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvdHhfdmFsaWRhdG9yL251bGxpZmllcl9jYWNoZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEtBQUssRUFBRSxlQUFlLEVBQUUsTUFBTSxZQUFZLENBQUM7QUFDbEQsT0FBTyxLQUFLLEVBQUUsd0JBQXdCLEVBQUUsTUFBTSxpQ0FBaUMsQ0FBQztBQUdoRjs7O0dBR0c7QUFDSCxxQkFBYSxjQUFlLFlBQVcsZUFBZTtJQUd4QyxPQUFPLENBQUMsRUFBRTtJQUZ0QixVQUFVLEVBQUUsR0FBRyxDQUFDLE1BQU0sQ0FBQyxDQUFDO0lBRXhCLFlBQW9CLEVBQUUsRUFBRSx3QkFBd0IsRUFFL0M7SUFFWSxlQUFlLENBQUMsVUFBVSxFQUFFLE1BQU0sRUFBRSxHQUFHLE9BQU8sQ0FBQyxPQUFPLEVBQUUsQ0FBQyxDQU9yRTtJQUVNLGFBQWEsQ0FBQyxVQUFVLEVBQUUsTUFBTSxFQUFFLFFBSXhDO0NBQ0YifQ==
@@ -1 +0,0 @@
1
- {"version":3,"file":"nullifier_cache.d.ts","sourceRoot":"","sources":["../../src/tx_validator/nullifier_cache.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,eAAe,EAAE,MAAM,YAAY,CAAC;AAClD,OAAO,KAAK,EAAE,wBAAwB,EAAE,MAAM,iCAAiC,CAAC;AAGhF;;;GAGG;AACH,qBAAa,cAAe,YAAW,eAAe;IAGxC,OAAO,CAAC,EAAE;IAFtB,UAAU,EAAE,GAAG,CAAC,MAAM,CAAC,CAAC;IAExB,YAAoB,EAAE,EAAE,wBAAwB,EAE/C;IAEY,eAAe,CAAC,UAAU,EAAE,MAAM,EAAE,GAAG,OAAO,CAAC,OAAO,EAAE,CAAC,CAOrE;IAEM,aAAa,CAAC,UAAU,EAAE,MAAM,EAAE,QAIxC;CACF"}