@aztec/sequencer-client 0.76.4-devnet-test-rc3 → 0.77.0-testnet-ignition.17

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 (87) hide show
  1. package/dest/client/index.js +0 -1
  2. package/dest/client/sequencer-client.d.ts +12 -9
  3. package/dest/client/sequencer-client.d.ts.map +1 -1
  4. package/dest/client/sequencer-client.js +55 -60
  5. package/dest/config.d.ts +2 -2
  6. package/dest/config.d.ts.map +1 -1
  7. package/dest/config.js +41 -46
  8. package/dest/global_variable_builder/global_builder.d.ts +5 -2
  9. package/dest/global_variable_builder/global_builder.d.ts.map +1 -1
  10. package/dest/global_variable_builder/global_builder.js +45 -31
  11. package/dest/global_variable_builder/index.js +0 -1
  12. package/dest/index.js +0 -1
  13. package/dest/publisher/config.d.ts.map +1 -1
  14. package/dest/publisher/config.js +33 -48
  15. package/dest/publisher/index.js +0 -1
  16. package/dest/publisher/sequencer-publisher-metrics.d.ts +6 -5
  17. package/dest/publisher/sequencer-publisher-metrics.d.ts.map +1 -1
  18. package/dest/publisher/sequencer-publisher-metrics.js +46 -28
  19. package/dest/publisher/sequencer-publisher.d.ts +11 -23
  20. package/dest/publisher/sequencer-publisher.d.ts.map +1 -1
  21. package/dest/publisher/sequencer-publisher.js +191 -238
  22. package/dest/sequencer/allowed.d.ts +1 -1
  23. package/dest/sequencer/allowed.d.ts.map +1 -1
  24. package/dest/sequencer/allowed.js +6 -13
  25. package/dest/sequencer/config.d.ts +1 -1
  26. package/dest/sequencer/config.d.ts.map +1 -1
  27. package/dest/sequencer/config.js +1 -2
  28. package/dest/sequencer/index.js +0 -1
  29. package/dest/sequencer/metrics.js +26 -18
  30. package/dest/sequencer/sequencer.d.ts +19 -15
  31. package/dest/sequencer/sequencer.d.ts.map +1 -1
  32. package/dest/sequencer/sequencer.js +589 -625
  33. package/dest/sequencer/timetable.d.ts +1 -1
  34. package/dest/sequencer/timetable.d.ts.map +1 -1
  35. package/dest/sequencer/timetable.js +33 -19
  36. package/dest/sequencer/utils.d.ts +2 -1
  37. package/dest/sequencer/utils.d.ts.map +1 -1
  38. package/dest/sequencer/utils.js +22 -32
  39. package/dest/slasher/factory.d.ts +5 -5
  40. package/dest/slasher/factory.d.ts.map +1 -1
  41. package/dest/slasher/factory.js +5 -4
  42. package/dest/slasher/index.js +0 -1
  43. package/dest/slasher/slasher_client.d.ts +4 -4
  44. package/dest/slasher/slasher_client.d.ts.map +1 -1
  45. package/dest/slasher/slasher_client.js +110 -113
  46. package/dest/test/index.d.ts +3 -3
  47. package/dest/test/index.d.ts.map +1 -1
  48. package/dest/test/index.js +4 -1
  49. package/dest/tx_validator/archive_cache.d.ts +3 -3
  50. package/dest/tx_validator/archive_cache.d.ts.map +1 -1
  51. package/dest/tx_validator/archive_cache.js +8 -8
  52. package/dest/tx_validator/gas_validator.d.ts +5 -3
  53. package/dest/tx_validator/gas_validator.d.ts.map +1 -1
  54. package/dest/tx_validator/gas_validator.js +70 -70
  55. package/dest/tx_validator/nullifier_cache.d.ts +2 -2
  56. package/dest/tx_validator/nullifier_cache.d.ts.map +1 -1
  57. package/dest/tx_validator/nullifier_cache.js +9 -9
  58. package/dest/tx_validator/phases_validator.d.ts +3 -2
  59. package/dest/tx_validator/phases_validator.d.ts.map +1 -1
  60. package/dest/tx_validator/phases_validator.js +28 -20
  61. package/dest/tx_validator/test_utils.d.ts +4 -2
  62. package/dest/tx_validator/test_utils.d.ts.map +1 -1
  63. package/dest/tx_validator/test_utils.js +2 -3
  64. package/dest/tx_validator/tx_validator_factory.d.ts +7 -5
  65. package/dest/tx_validator/tx_validator_factory.d.ts.map +1 -1
  66. package/dest/tx_validator/tx_validator_factory.js +14 -13
  67. package/package.json +26 -27
  68. package/src/client/sequencer-client.ts +17 -14
  69. package/src/config.ts +6 -14
  70. package/src/global_variable_builder/global_builder.ts +18 -19
  71. package/src/publisher/config.ts +9 -17
  72. package/src/publisher/sequencer-publisher-metrics.ts +22 -9
  73. package/src/publisher/sequencer-publisher.ts +16 -101
  74. package/src/sequencer/allowed.ts +4 -4
  75. package/src/sequencer/config.ts +1 -1
  76. package/src/sequencer/sequencer.ts +39 -113
  77. package/src/sequencer/timetable.ts +1 -1
  78. package/src/sequencer/utils.ts +2 -1
  79. package/src/slasher/factory.ts +5 -5
  80. package/src/slasher/slasher_client.ts +16 -21
  81. package/src/test/index.ts +3 -3
  82. package/src/tx_validator/archive_cache.ts +4 -3
  83. package/src/tx_validator/gas_validator.ts +22 -31
  84. package/src/tx_validator/nullifier_cache.ts +3 -2
  85. package/src/tx_validator/phases_validator.ts +7 -7
  86. package/src/tx_validator/test_utils.ts +5 -3
  87. package/src/tx_validator/tx_validator_factory.ts +23 -17
@@ -1,14 +1,22 @@
1
- import { __esDecorate, __runInitializers } from "tslib";
2
- import { SequencerConfigSchema, Tx, } from '@aztec/circuit-types';
3
- import { AppendOnlyTreeSnapshot, BlockHeader, ContentCommitment, GENESIS_ARCHIVE_ROOT, Gas, INITIAL_L2_BLOCK_NUM, StateReference, } from '@aztec/circuits.js';
4
- import { AztecAddress } from '@aztec/foundation/aztec-address';
1
+ function _ts_decorate(decorators, target, key, desc) {
2
+ var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d;
3
+ if (typeof Reflect === "object" && typeof Reflect.decorate === "function") r = Reflect.decorate(decorators, target, key, desc);
4
+ else for(var i = decorators.length - 1; i >= 0; i--)if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r;
5
+ return c > 3 && r && Object.defineProperty(target, key, r), r;
6
+ }
7
+ import { INITIAL_L2_BLOCK_NUM } from '@aztec/constants';
5
8
  import { omit } from '@aztec/foundation/collection';
6
9
  import { EthAddress } from '@aztec/foundation/eth-address';
7
10
  import { Fr } from '@aztec/foundation/fields';
8
11
  import { createLogger } from '@aztec/foundation/log';
9
12
  import { RunningPromise } from '@aztec/foundation/running-promise';
10
- import { pickFromSchema } from '@aztec/foundation/schemas';
11
13
  import { Timer, elapsed } from '@aztec/foundation/timer';
14
+ import { AztecAddress } from '@aztec/stdlib/aztec-address';
15
+ import { Gas } from '@aztec/stdlib/gas';
16
+ import { SequencerConfigSchema } from '@aztec/stdlib/interfaces/server';
17
+ import { pickFromSchema } from '@aztec/stdlib/schemas';
18
+ import { AppendOnlyTreeSnapshot, MerkleTreeId } from '@aztec/stdlib/trees';
19
+ import { BlockHeader, ContentCommitment, StateReference, Tx } from '@aztec/stdlib/tx';
12
20
  import { Attributes, getTelemetryClient, trackSpan } from '@aztec/telemetry-client';
13
21
  import { VoteType } from '../publisher/sequencer-publisher.js';
14
22
  import { createValidatorsForBlockBuilding } from '../tx_validator/tx_validator_factory.js';
@@ -25,635 +33,591 @@ export { SequencerState };
25
33
  * - Adds proof requests to the request pool (not for this milestone).
26
34
  * - Receives results to those proofs from the network (repeats as necessary) (not for this milestone).
27
35
  * - Publishes L1 tx(s) to the rollup contract via RollupPublisher.
28
- */
29
- let Sequencer = (() => {
30
- var _a;
31
- let _instanceExtraInitializers = [];
32
- let _work_decorators;
33
- let _buildBlockAndEnqueuePublish_decorators;
34
- let _collectAttestations_decorators;
35
- let _enqueuePublishL2Block_decorators;
36
- let _claimEpochProofRightIfAvailable_decorators;
37
- return _a = class Sequencer {
38
- constructor(publisher, validatorClient, // During migration the validator client can be inactive
39
- globalsBuilder, p2pClient, worldState, slasherClient, blockBuilderFactory, l2BlockSource, l1ToL2MessageSource, publicProcessorFactory, contractDataSource, l1Constants, dateProvider, config = {}, telemetry = getTelemetryClient(), log = createLogger('sequencer')) {
40
- this.publisher = (__runInitializers(this, _instanceExtraInitializers), publisher);
41
- this.validatorClient = validatorClient;
42
- this.globalsBuilder = globalsBuilder;
43
- this.p2pClient = p2pClient;
44
- this.worldState = worldState;
45
- this.slasherClient = slasherClient;
46
- this.blockBuilderFactory = blockBuilderFactory;
47
- this.l2BlockSource = l2BlockSource;
48
- this.l1ToL2MessageSource = l1ToL2MessageSource;
49
- this.publicProcessorFactory = publicProcessorFactory;
50
- this.contractDataSource = contractDataSource;
51
- this.l1Constants = l1Constants;
52
- this.dateProvider = dateProvider;
53
- this.config = config;
54
- this.log = log;
55
- this.pollingIntervalMs = 1000;
56
- this.maxTxsPerBlock = 32;
57
- this.minTxsPerBlock = 1;
58
- this.maxL1TxInclusionTimeIntoSlot = 0;
59
- // TODO: zero values should not be allowed for the following 2 values in PROD
60
- this._coinbase = EthAddress.ZERO;
61
- this._feeRecipient = AztecAddress.ZERO;
62
- this.state = SequencerState.STOPPED;
63
- this.allowedInSetup = [];
64
- this.maxBlockSizeInBytes = 1024 * 1024;
65
- this.maxBlockGas = new Gas(100e9, 100e9);
66
- this.isFlushing = false;
67
- this.enforceTimeTable = false;
68
- this.metrics = new SequencerMetrics(telemetry, () => this.state, 'Sequencer');
69
- // Register the block builder with the validator client for re-execution
70
- this.validatorClient?.registerBlockBuilder(this.buildBlock.bind(this));
71
- // Register the slasher on the publisher to fetch slashing payloads
72
- this.publisher.registerSlashPayloadGetter(this.slasherClient.getSlashPayload.bind(this.slasherClient));
73
- }
74
- get tracer() {
75
- return this.metrics.tracer;
76
- }
77
- /**
78
- * Updates sequencer config.
79
- * @param config - New parameters.
80
- */
81
- async updateConfig(config) {
82
- this.log.info(`Sequencer config set`, omit(pickFromSchema(config, SequencerConfigSchema), 'allowedInSetup'));
83
- if (config.transactionPollingIntervalMS !== undefined) {
84
- this.pollingIntervalMs = config.transactionPollingIntervalMS;
85
- }
86
- if (config.maxTxsPerBlock !== undefined) {
87
- this.maxTxsPerBlock = config.maxTxsPerBlock;
88
- }
89
- if (config.minTxsPerBlock !== undefined) {
90
- this.minTxsPerBlock = config.minTxsPerBlock;
91
- }
92
- if (config.maxDABlockGas !== undefined) {
93
- this.maxBlockGas = new Gas(config.maxDABlockGas, this.maxBlockGas.l2Gas);
94
- }
95
- if (config.maxL2BlockGas !== undefined) {
96
- this.maxBlockGas = new Gas(this.maxBlockGas.daGas, config.maxL2BlockGas);
97
- }
98
- if (config.coinbase) {
99
- this._coinbase = config.coinbase;
100
- }
101
- if (config.feeRecipient) {
102
- this._feeRecipient = config.feeRecipient;
103
- }
104
- if (config.allowedInSetup) {
105
- this.allowedInSetup = config.allowedInSetup;
106
- }
107
- else {
108
- this.allowedInSetup = await getDefaultAllowedSetupFunctions();
109
- }
110
- if (config.maxBlockSizeInBytes !== undefined) {
111
- this.maxBlockSizeInBytes = config.maxBlockSizeInBytes;
112
- }
113
- if (config.governanceProposerPayload) {
114
- this.publisher.setGovernancePayload(config.governanceProposerPayload);
115
- }
116
- if (config.maxL1TxInclusionTimeIntoSlot !== undefined) {
117
- this.maxL1TxInclusionTimeIntoSlot = config.maxL1TxInclusionTimeIntoSlot;
118
- }
119
- if (config.enforceTimeTable !== undefined) {
120
- this.enforceTimeTable = config.enforceTimeTable;
121
- }
122
- this.setTimeTable();
123
- // TODO: Just read everything from the config object as needed instead of copying everything into local vars.
124
- this.config = config;
125
- }
126
- setTimeTable() {
127
- this.timetable = new SequencerTimetable(this.l1Constants.ethereumSlotDuration, this.aztecSlotDuration, this.maxL1TxInclusionTimeIntoSlot, this.enforceTimeTable, this.metrics, this.log);
128
- this.log.verbose(`Sequencer timetable updated`, { enforceTimeTable: this.enforceTimeTable });
129
- }
130
- /**
131
- * Starts the sequencer and moves to IDLE state.
132
- */
133
- async start() {
134
- await this.updateConfig(this.config);
135
- this.runningPromise = new RunningPromise(this.work.bind(this), this.log, this.pollingIntervalMs);
136
- this.setState(SequencerState.IDLE, 0n, true /** force */);
137
- this.runningPromise.start();
138
- this.log.info(`Sequencer started with address ${this.publisher.getSenderAddress().toString()}`);
139
- }
140
- /**
141
- * Stops the sequencer from processing txs and moves to STOPPED state.
142
- */
143
- async stop() {
144
- this.log.debug(`Stopping sequencer`);
145
- await this.validatorClient?.stop();
146
- await this.runningPromise?.stop();
147
- await this.slasherClient?.stop();
148
- this.publisher.interrupt();
149
- this.setState(SequencerState.STOPPED, 0n, true /** force */);
150
- this.log.info('Stopped sequencer');
151
- }
152
- /**
153
- * Starts a previously stopped sequencer.
154
- */
155
- restart() {
156
- this.log.info('Restarting sequencer');
157
- this.publisher.restart();
158
- this.runningPromise.start();
159
- this.setState(SequencerState.IDLE, 0n, true /** force */);
160
- }
161
- /**
162
- * Returns the current state of the sequencer.
163
- * @returns An object with a state entry with one of SequencerState.
164
- */
165
- status() {
166
- return { state: this.state };
167
- }
168
- /** Forces the sequencer to bypass all time and tx count checks for the next block and build anyway. */
169
- flush() {
170
- this.isFlushing = true;
171
- }
172
- /**
173
- * @notice Performs most of the sequencer duties:
174
- * - Checks if we are up to date
175
- * - If we are and we are the sequencer, collect txs and build a block
176
- * - Collect attestations for the block
177
- * - Submit block
178
- * - If our block for some reason is not included, revert the state
179
- */
180
- async doRealWork() {
181
- this.setState(SequencerState.SYNCHRONIZING, 0n);
182
- // Update state when the previous block has been synced
183
- const chainTip = await this.getChainTip();
184
- // Do not go forward with new block if the previous one has not been mined and processed
185
- if (!chainTip) {
186
- return;
187
- }
188
- this.setState(SequencerState.PROPOSER_CHECK, 0n);
189
- const newBlockNumber = chainTip.blockNumber + 1;
190
- // If we cannot find a tip archive, assume genesis.
191
- const chainTipArchive = chainTip.archive;
192
- const slot = await this.slotForProposal(chainTipArchive.toBuffer(), BigInt(newBlockNumber));
193
- if (!slot) {
194
- this.log.debug(`Cannot propose block ${newBlockNumber}`);
195
- return;
196
- }
197
- this.log.debug(`Can propose block ${newBlockNumber} at slot ${slot}`);
198
- const newGlobalVariables = await this.globalsBuilder.buildGlobalVariables(new Fr(newBlockNumber), this._coinbase, this._feeRecipient, slot);
199
- const enqueueGovernanceVotePromise = this.publisher.enqueueCastVote(slot, newGlobalVariables.timestamp.toBigInt(), VoteType.GOVERNANCE);
200
- const enqueueSlashingVotePromise = this.publisher.enqueueCastVote(slot, newGlobalVariables.timestamp.toBigInt(), VoteType.SLASHING);
201
- // Start collecting proof quotes for the previous epoch if needed in the background
202
- const createProofQuotePromise = this.createProofClaimForPreviousEpoch(slot);
203
- this.setState(SequencerState.INITIALIZING_PROPOSAL, slot);
204
- this.log.verbose(`Preparing proposal for block ${newBlockNumber} at slot ${slot}`, {
205
- chainTipArchive,
36
+ */ export class Sequencer {
37
+ publisher;
38
+ validatorClient;
39
+ globalsBuilder;
40
+ p2pClient;
41
+ worldState;
42
+ slasherClient;
43
+ blockBuilderFactory;
44
+ l2BlockSource;
45
+ l1ToL2MessageSource;
46
+ publicProcessorFactory;
47
+ contractDataSource;
48
+ l1Constants;
49
+ dateProvider;
50
+ config;
51
+ log;
52
+ runningPromise;
53
+ pollingIntervalMs;
54
+ maxTxsPerBlock;
55
+ minTxsPerBlock;
56
+ maxL1TxInclusionTimeIntoSlot;
57
+ // TODO: zero values should not be allowed for the following 2 values in PROD
58
+ _coinbase;
59
+ _feeRecipient;
60
+ state;
61
+ allowedInSetup;
62
+ maxBlockSizeInBytes;
63
+ maxBlockGas;
64
+ metrics;
65
+ isFlushing;
66
+ /** The maximum number of seconds that the sequencer can be into a slot to transition to a particular state. */ timetable;
67
+ enforceTimeTable;
68
+ constructor(publisher, validatorClient, globalsBuilder, p2pClient, worldState, slasherClient, blockBuilderFactory, l2BlockSource, l1ToL2MessageSource, publicProcessorFactory, contractDataSource, l1Constants, dateProvider, config = {}, telemetry = getTelemetryClient(), log = createLogger('sequencer')){
69
+ this.publisher = publisher;
70
+ this.validatorClient = validatorClient;
71
+ this.globalsBuilder = globalsBuilder;
72
+ this.p2pClient = p2pClient;
73
+ this.worldState = worldState;
74
+ this.slasherClient = slasherClient;
75
+ this.blockBuilderFactory = blockBuilderFactory;
76
+ this.l2BlockSource = l2BlockSource;
77
+ this.l1ToL2MessageSource = l1ToL2MessageSource;
78
+ this.publicProcessorFactory = publicProcessorFactory;
79
+ this.contractDataSource = contractDataSource;
80
+ this.l1Constants = l1Constants;
81
+ this.dateProvider = dateProvider;
82
+ this.config = config;
83
+ this.log = log;
84
+ this.pollingIntervalMs = 1000;
85
+ this.maxTxsPerBlock = 32;
86
+ this.minTxsPerBlock = 1;
87
+ this.maxL1TxInclusionTimeIntoSlot = 0;
88
+ this._coinbase = EthAddress.ZERO;
89
+ this._feeRecipient = AztecAddress.ZERO;
90
+ this.state = SequencerState.STOPPED;
91
+ this.allowedInSetup = [];
92
+ this.maxBlockSizeInBytes = 1024 * 1024;
93
+ this.maxBlockGas = new Gas(100e9, 100e9);
94
+ this.isFlushing = false;
95
+ this.enforceTimeTable = false;
96
+ this.metrics = new SequencerMetrics(telemetry, ()=>this.state, 'Sequencer');
97
+ // Register the block builder with the validator client for re-execution
98
+ this.validatorClient?.registerBlockBuilder(this.buildBlock.bind(this));
99
+ // Register the slasher on the publisher to fetch slashing payloads
100
+ this.publisher.registerSlashPayloadGetter(this.slasherClient.getSlashPayload.bind(this.slasherClient));
101
+ }
102
+ get tracer() {
103
+ return this.metrics.tracer;
104
+ }
105
+ /**
106
+ * Updates sequencer config.
107
+ * @param config - New parameters.
108
+ */ async updateConfig(config) {
109
+ this.log.info(`Sequencer config set`, omit(pickFromSchema(config, SequencerConfigSchema), 'allowedInSetup'));
110
+ if (config.transactionPollingIntervalMS !== undefined) {
111
+ this.pollingIntervalMs = config.transactionPollingIntervalMS;
112
+ }
113
+ if (config.maxTxsPerBlock !== undefined) {
114
+ this.maxTxsPerBlock = config.maxTxsPerBlock;
115
+ }
116
+ if (config.minTxsPerBlock !== undefined) {
117
+ this.minTxsPerBlock = config.minTxsPerBlock;
118
+ }
119
+ if (config.maxDABlockGas !== undefined) {
120
+ this.maxBlockGas = new Gas(config.maxDABlockGas, this.maxBlockGas.l2Gas);
121
+ }
122
+ if (config.maxL2BlockGas !== undefined) {
123
+ this.maxBlockGas = new Gas(this.maxBlockGas.daGas, config.maxL2BlockGas);
124
+ }
125
+ if (config.coinbase) {
126
+ this._coinbase = config.coinbase;
127
+ }
128
+ if (config.feeRecipient) {
129
+ this._feeRecipient = config.feeRecipient;
130
+ }
131
+ if (config.allowedInSetup) {
132
+ this.allowedInSetup = config.allowedInSetup;
133
+ } else {
134
+ this.allowedInSetup = await getDefaultAllowedSetupFunctions();
135
+ }
136
+ if (config.maxBlockSizeInBytes !== undefined) {
137
+ this.maxBlockSizeInBytes = config.maxBlockSizeInBytes;
138
+ }
139
+ if (config.governanceProposerPayload) {
140
+ this.publisher.setGovernancePayload(config.governanceProposerPayload);
141
+ }
142
+ if (config.maxL1TxInclusionTimeIntoSlot !== undefined) {
143
+ this.maxL1TxInclusionTimeIntoSlot = config.maxL1TxInclusionTimeIntoSlot;
144
+ }
145
+ if (config.enforceTimeTable !== undefined) {
146
+ this.enforceTimeTable = config.enforceTimeTable;
147
+ }
148
+ this.setTimeTable();
149
+ // TODO: Just read everything from the config object as needed instead of copying everything into local vars.
150
+ this.config = config;
151
+ }
152
+ setTimeTable() {
153
+ this.timetable = new SequencerTimetable(this.l1Constants.ethereumSlotDuration, this.aztecSlotDuration, this.maxL1TxInclusionTimeIntoSlot, this.enforceTimeTable, this.metrics, this.log);
154
+ this.log.verbose(`Sequencer timetable updated`, {
155
+ enforceTimeTable: this.enforceTimeTable
156
+ });
157
+ }
158
+ /**
159
+ * Starts the sequencer and moves to IDLE state.
160
+ */ async start() {
161
+ await this.updateConfig(this.config);
162
+ this.runningPromise = new RunningPromise(this.work.bind(this), this.log, this.pollingIntervalMs);
163
+ this.setState(SequencerState.IDLE, 0n, true);
164
+ this.runningPromise.start();
165
+ this.log.info(`Sequencer started with address ${this.publisher.getSenderAddress().toString()}`);
166
+ }
167
+ /**
168
+ * Stops the sequencer from processing txs and moves to STOPPED state.
169
+ */ async stop() {
170
+ this.log.debug(`Stopping sequencer`);
171
+ await this.validatorClient?.stop();
172
+ await this.runningPromise?.stop();
173
+ await this.slasherClient?.stop();
174
+ this.publisher.interrupt();
175
+ this.setState(SequencerState.STOPPED, 0n, true);
176
+ this.log.info('Stopped sequencer');
177
+ }
178
+ /**
179
+ * Starts a previously stopped sequencer.
180
+ */ restart() {
181
+ this.log.info('Restarting sequencer');
182
+ this.publisher.restart();
183
+ this.runningPromise.start();
184
+ this.setState(SequencerState.IDLE, 0n, true);
185
+ }
186
+ /**
187
+ * Returns the current state of the sequencer.
188
+ * @returns An object with a state entry with one of SequencerState.
189
+ */ status() {
190
+ return {
191
+ state: this.state
192
+ };
193
+ }
194
+ /** Forces the sequencer to bypass all time and tx count checks for the next block and build anyway. */ flush() {
195
+ this.isFlushing = true;
196
+ }
197
+ /**
198
+ * @notice Performs most of the sequencer duties:
199
+ * - Checks if we are up to date
200
+ * - If we are and we are the sequencer, collect txs and build a block
201
+ * - Collect attestations for the block
202
+ * - Submit block
203
+ * - If our block for some reason is not included, revert the state
204
+ */ async doRealWork() {
205
+ this.setState(SequencerState.SYNCHRONIZING, 0n);
206
+ // Update state when the previous block has been synced
207
+ const chainTip = await this.getChainTip();
208
+ // Do not go forward with new block if the previous one has not been mined and processed
209
+ if (!chainTip) {
210
+ return;
211
+ }
212
+ this.setState(SequencerState.PROPOSER_CHECK, 0n);
213
+ const newBlockNumber = chainTip.blockNumber + 1;
214
+ // If we cannot find a tip archive, assume genesis.
215
+ const chainTipArchive = chainTip.archive;
216
+ const slot = await this.slotForProposal(chainTipArchive.toBuffer(), BigInt(newBlockNumber));
217
+ if (!slot) {
218
+ this.log.debug(`Cannot propose block ${newBlockNumber}`);
219
+ return;
220
+ }
221
+ this.log.debug(`Can propose block ${newBlockNumber} at slot ${slot}`);
222
+ const newGlobalVariables = await this.globalsBuilder.buildGlobalVariables(new Fr(newBlockNumber), this._coinbase, this._feeRecipient, slot);
223
+ const enqueueGovernanceVotePromise = this.publisher.enqueueCastVote(slot, newGlobalVariables.timestamp.toBigInt(), VoteType.GOVERNANCE);
224
+ const enqueueSlashingVotePromise = this.publisher.enqueueCastVote(slot, newGlobalVariables.timestamp.toBigInt(), VoteType.SLASHING);
225
+ this.setState(SequencerState.INITIALIZING_PROPOSAL, slot);
226
+ this.log.verbose(`Preparing proposal for block ${newBlockNumber} at slot ${slot}`, {
227
+ chainTipArchive,
228
+ blockNumber: newBlockNumber,
229
+ slot
230
+ });
231
+ // If I created a "partial" header here that should make our job much easier.
232
+ const proposalHeader = new BlockHeader(new AppendOnlyTreeSnapshot(chainTipArchive, 1), ContentCommitment.empty(), StateReference.empty(), newGlobalVariables, Fr.ZERO, Fr.ZERO);
233
+ let finishedFlushing = false;
234
+ const pendingTxCount = await this.p2pClient.getPendingTxCount();
235
+ if (pendingTxCount >= this.minTxsPerBlock || this.isFlushing) {
236
+ // We don't fetch exactly maxTxsPerBlock txs here because we may not need all of them if we hit a limit before,
237
+ // and also we may need to fetch more if we don't have enough valid txs.
238
+ const pendingTxs = this.p2pClient.iteratePendingTxs();
239
+ await this.buildBlockAndEnqueuePublish(pendingTxs, proposalHeader).catch((err)=>{
240
+ this.log.error(`Error building/enqueuing block`, err, {
206
241
  blockNumber: newBlockNumber,
207
- slot,
242
+ slot
208
243
  });
209
- // If I created a "partial" header here that should make our job much easier.
210
- const proposalHeader = new BlockHeader(new AppendOnlyTreeSnapshot(chainTipArchive, 1), ContentCommitment.empty(), StateReference.empty(), newGlobalVariables, Fr.ZERO, Fr.ZERO);
211
- let finishedFlushing = false;
212
- const pendingTxCount = await this.p2pClient.getPendingTxCount();
213
- if (pendingTxCount >= this.minTxsPerBlock || this.isFlushing) {
214
- // We don't fetch exactly maxTxsPerBlock txs here because we may not need all of them if we hit a limit before,
215
- // and also we may need to fetch more if we don't have enough valid txs.
216
- const pendingTxs = this.p2pClient.iteratePendingTxs();
217
- await this.buildBlockAndEnqueuePublish(pendingTxs, proposalHeader).catch(err => {
218
- this.log.error(`Error building/enqueuing block`, err, { blockNumber: newBlockNumber, slot });
219
- });
220
- finishedFlushing = true;
221
- }
222
- else {
223
- this.log.debug(`Not enough txs to build block ${newBlockNumber} at slot ${slot}: got ${pendingTxCount} txs, need ${this.minTxsPerBlock}`);
224
- }
225
- await enqueueGovernanceVotePromise.catch(err => {
226
- this.log.error(`Error enqueuing governance vote`, err, { blockNumber: newBlockNumber, slot });
227
- });
228
- await enqueueSlashingVotePromise.catch(err => {
229
- this.log.error(`Error enqueuing slashing vote`, err, { blockNumber: newBlockNumber, slot });
230
- });
231
- await createProofQuotePromise
232
- .then(quote => (quote ? this.publisher.enqueueClaimEpochProofRight(quote) : undefined))
233
- .catch(err => {
234
- this.log.error(`Error creating proof quote`, err, { blockNumber: newBlockNumber, slot });
235
- });
236
- await this.publisher.sendRequests();
237
- if (finishedFlushing) {
238
- this.isFlushing = false;
239
- }
240
- this.setState(SequencerState.IDLE, 0n);
244
+ });
245
+ finishedFlushing = true;
246
+ } else {
247
+ this.log.debug(`Not enough txs to build block ${newBlockNumber} at slot ${slot}: got ${pendingTxCount} txs, need ${this.minTxsPerBlock}`);
248
+ }
249
+ await enqueueGovernanceVotePromise.catch((err)=>{
250
+ this.log.error(`Error enqueuing governance vote`, err, {
251
+ blockNumber: newBlockNumber,
252
+ slot
253
+ });
254
+ });
255
+ await enqueueSlashingVotePromise.catch((err)=>{
256
+ this.log.error(`Error enqueuing slashing vote`, err, {
257
+ blockNumber: newBlockNumber,
258
+ slot
259
+ });
260
+ });
261
+ await this.publisher.sendRequests();
262
+ if (finishedFlushing) {
263
+ this.isFlushing = false;
264
+ }
265
+ this.setState(SequencerState.IDLE, 0n);
266
+ }
267
+ async work() {
268
+ try {
269
+ await this.doRealWork();
270
+ } catch (err) {
271
+ if (err instanceof SequencerTooSlowError) {
272
+ this.log.warn(err.message);
273
+ } else {
274
+ // Re-throw other errors
275
+ throw err;
241
276
  }
242
- async work() {
243
- try {
244
- await this.doRealWork();
245
- }
246
- catch (err) {
247
- if (err instanceof SequencerTooSlowError) {
248
- this.log.warn(err.message);
249
- }
250
- else {
251
- // Re-throw other errors
252
- throw err;
253
- }
254
- }
255
- finally {
256
- this.setState(SequencerState.IDLE, 0n);
257
- }
277
+ } finally{
278
+ this.setState(SequencerState.IDLE, 0n);
279
+ }
280
+ }
281
+ getForwarderAddress() {
282
+ return this.publisher.getForwarderAddress();
283
+ }
284
+ /**
285
+ * Checks if we can propose at the next block and returns the slot number if we can.
286
+ * @param tipArchive - The archive of the previous block.
287
+ * @param proposalBlockNumber - The block number of the proposal.
288
+ * @returns The slot number if we can propose at the next block, otherwise undefined.
289
+ */ async slotForProposal(tipArchive, proposalBlockNumber) {
290
+ const result = await this.publisher.canProposeAtNextEthBlock(tipArchive);
291
+ if (!result) {
292
+ return undefined;
293
+ }
294
+ const [slot, blockNumber] = result;
295
+ if (proposalBlockNumber !== blockNumber) {
296
+ const msg = `Sequencer block number mismatch. Expected ${proposalBlockNumber} but got ${blockNumber}.`;
297
+ this.log.warn(msg);
298
+ throw new Error(msg);
299
+ }
300
+ return slot;
301
+ }
302
+ /**
303
+ * Sets the sequencer state and checks if we have enough time left in the slot to transition to the new state.
304
+ * @param proposedState - The new state to transition to.
305
+ * @param currentSlotNumber - The current slot number.
306
+ * @param force - Whether to force the transition even if the sequencer is stopped.
307
+ *
308
+ * @dev If the `currentSlotNumber` doesn't matter (e.g. transitioning to IDLE), pass in `0n`;
309
+ * it is only used to check if we have enough time left in the slot to transition to the new state.
310
+ */ setState(proposedState, currentSlotNumber, force = false) {
311
+ if (this.state === SequencerState.STOPPED && force !== true) {
312
+ this.log.warn(`Cannot set sequencer from ${this.state} to ${proposedState} as it is stopped.`);
313
+ return;
314
+ }
315
+ const secondsIntoSlot = this.getSecondsIntoSlot(currentSlotNumber);
316
+ this.timetable.assertTimeLeft(proposedState, secondsIntoSlot);
317
+ this.log.debug(`Transitioning from ${this.state} to ${proposedState}`);
318
+ this.state = proposedState;
319
+ }
320
+ /**
321
+ * Build a block
322
+ *
323
+ * Shared between the sequencer and the validator for re-execution
324
+ *
325
+ * @param pendingTxs - The pending transactions to construct the block from
326
+ * @param newGlobalVariables - The global variables for the new block
327
+ * @param historicalHeader - The historical header of the parent
328
+ * @param opts - Whether to just validate the block as a validator, as opposed to building it as a proposal
329
+ */ async buildBlock(pendingTxs, newGlobalVariables, opts = {}) {
330
+ const blockNumber = newGlobalVariables.blockNumber.toNumber();
331
+ const slot = newGlobalVariables.slotNumber.toBigInt();
332
+ this.log.debug(`Requesting L1 to L2 messages from contract for block ${blockNumber}`);
333
+ const l1ToL2Messages = await this.l1ToL2MessageSource.getL1ToL2Messages(BigInt(blockNumber));
334
+ const msgCount = l1ToL2Messages.length;
335
+ this.log.verbose(`Building block ${blockNumber} for slot ${slot}`, {
336
+ slot,
337
+ blockNumber,
338
+ msgCount,
339
+ validator: opts.validateOnly
340
+ });
341
+ // Sync to the previous block at least
342
+ await this.worldState.syncImmediate(blockNumber - 1);
343
+ this.log.debug(`Synced to previous block ${blockNumber - 1}`);
344
+ // NB: separating the dbs because both should update the state
345
+ const publicProcessorFork = await this.worldState.fork();
346
+ const orchestratorFork = await this.worldState.fork();
347
+ const previousBlockHeader = (await this.l2BlockSource.getBlock(blockNumber - 1))?.header ?? orchestratorFork.getInitialHeader();
348
+ try {
349
+ const processor = this.publicProcessorFactory.create(publicProcessorFork, newGlobalVariables, true);
350
+ const blockBuildingTimer = new Timer();
351
+ const blockBuilder = this.blockBuilderFactory.create(orchestratorFork);
352
+ await blockBuilder.startNewBlock(newGlobalVariables, l1ToL2Messages, previousBlockHeader);
353
+ // Deadline for processing depends on whether we're proposing a block
354
+ const secondsIntoSlot = this.getSecondsIntoSlot(slot);
355
+ const processingEndTimeWithinSlot = opts.validateOnly ? this.timetable.getValidatorReexecTimeEnd(secondsIntoSlot) : this.timetable.getBlockProposalExecTimeEnd(secondsIntoSlot);
356
+ // Deadline is only set if enforceTimeTable is enabled.
357
+ const deadline = this.enforceTimeTable ? new Date((this.getSlotStartTimestamp(slot) + processingEndTimeWithinSlot) * 1000) : undefined;
358
+ this.log.verbose(`Processing pending txs`, {
359
+ slot,
360
+ slotStart: new Date(this.getSlotStartTimestamp(slot) * 1000),
361
+ now: new Date(this.dateProvider.now()),
362
+ deadline
363
+ });
364
+ const validators = createValidatorsForBlockBuilding(publicProcessorFork, this.contractDataSource, newGlobalVariables, this.allowedInSetup);
365
+ // TODO(#11000): Public processor should just handle processing, one tx at a time. It should be responsibility
366
+ // of the sequencer to update world state and iterate over txs. We should refactor this along with unifying the
367
+ // publicProcessorFork and orchestratorFork, to avoid doing tree insertions twice when building the block.
368
+ const proposerLimits = {
369
+ maxTransactions: this.maxTxsPerBlock,
370
+ maxBlockSize: this.maxBlockSizeInBytes,
371
+ maxBlockGas: this.maxBlockGas
372
+ };
373
+ const limits = opts.validateOnly ? {
374
+ deadline
375
+ } : {
376
+ deadline,
377
+ ...proposerLimits
378
+ };
379
+ const [publicProcessorDuration, [processedTxs, failedTxs]] = await elapsed(()=>processor.process(pendingTxs, limits, validators));
380
+ if (!opts.validateOnly && failedTxs.length > 0) {
381
+ const failedTxData = failedTxs.map((fail)=>fail.tx);
382
+ const failedTxHashes = await Tx.getHashes(failedTxData);
383
+ this.log.verbose(`Dropping failed txs ${failedTxHashes.join(', ')}`);
384
+ await this.p2pClient.deleteTxs(failedTxHashes);
258
385
  }
259
- getForwarderAddress() {
260
- return this.publisher.getForwarderAddress();
261
- }
262
- /**
263
- * Checks if we can propose at the next block and returns the slot number if we can.
264
- * @param tipArchive - The archive of the previous block.
265
- * @param proposalBlockNumber - The block number of the proposal.
266
- * @returns The slot number if we can propose at the next block, otherwise undefined.
267
- */
268
- async slotForProposal(tipArchive, proposalBlockNumber) {
269
- const result = await this.publisher.canProposeAtNextEthBlock(tipArchive);
270
- if (!result) {
271
- return undefined;
272
- }
273
- const [slot, blockNumber] = result;
274
- if (proposalBlockNumber !== blockNumber) {
275
- const msg = `Sequencer block number mismatch. Expected ${proposalBlockNumber} but got ${blockNumber}.`;
276
- this.log.warn(msg);
277
- throw new Error(msg);
278
- }
279
- return slot;
280
- }
281
- /**
282
- * Sets the sequencer state and checks if we have enough time left in the slot to transition to the new state.
283
- * @param proposedState - The new state to transition to.
284
- * @param currentSlotNumber - The current slot number.
285
- * @param force - Whether to force the transition even if the sequencer is stopped.
286
- *
287
- * @dev If the `currentSlotNumber` doesn't matter (e.g. transitioning to IDLE), pass in `0n`;
288
- * it is only used to check if we have enough time left in the slot to transition to the new state.
289
- */
290
- setState(proposedState, currentSlotNumber, force = false) {
291
- if (this.state === SequencerState.STOPPED && force !== true) {
292
- this.log.warn(`Cannot set sequencer from ${this.state} to ${proposedState} as it is stopped.`);
293
- return;
294
- }
295
- const secondsIntoSlot = this.getSecondsIntoSlot(currentSlotNumber);
296
- this.timetable.assertTimeLeft(proposedState, secondsIntoSlot);
297
- this.log.debug(`Transitioning from ${this.state} to ${proposedState}`);
298
- this.state = proposedState;
299
- }
300
- /**
301
- * Build a block
302
- *
303
- * Shared between the sequencer and the validator for re-execution
304
- *
305
- * @param pendingTxs - The pending transactions to construct the block from
306
- * @param newGlobalVariables - The global variables for the new block
307
- * @param historicalHeader - The historical header of the parent
308
- * @param opts - Whether to just validate the block as a validator, as opposed to building it as a proposal
309
- */
310
- async buildBlock(pendingTxs, newGlobalVariables, opts = {}) {
311
- const blockNumber = newGlobalVariables.blockNumber.toNumber();
312
- const slot = newGlobalVariables.slotNumber.toBigInt();
313
- this.log.debug(`Requesting L1 to L2 messages from contract for block ${blockNumber}`);
314
- const l1ToL2Messages = await this.l1ToL2MessageSource.getL1ToL2Messages(BigInt(blockNumber));
315
- const msgCount = l1ToL2Messages.length;
316
- this.log.verbose(`Building block ${blockNumber} for slot ${slot}`, {
386
+ if (!opts.validateOnly && // We check for minTxCount only if we are proposing a block, not if we are validating it
387
+ !this.isFlushing && // And we skip the check when flushing, since we want all pending txs to go out, no matter if too few
388
+ this.minTxsPerBlock !== undefined && processedTxs.length < this.minTxsPerBlock) {
389
+ this.log.warn(`Block ${blockNumber} has too few txs to be proposed (got ${processedTxs.length} but required ${this.minTxsPerBlock})`, {
317
390
  slot,
318
391
  blockNumber,
319
- msgCount,
320
- validator: opts.validateOnly,
392
+ processedTxCount: processedTxs.length
321
393
  });
322
- // Sync to the previous block at least
323
- await this.worldState.syncImmediate(blockNumber - 1);
324
- this.log.debug(`Synced to previous block ${blockNumber - 1}`);
325
- // NB: separating the dbs because both should update the state
326
- const publicProcessorFork = await this.worldState.fork();
327
- const orchestratorFork = await this.worldState.fork();
328
- const previousBlockHeader = (await this.l2BlockSource.getBlock(blockNumber - 1))?.header ?? orchestratorFork.getInitialHeader();
329
- try {
330
- const processor = this.publicProcessorFactory.create(publicProcessorFork, newGlobalVariables, true);
331
- const blockBuildingTimer = new Timer();
332
- const blockBuilder = this.blockBuilderFactory.create(orchestratorFork);
333
- await blockBuilder.startNewBlock(newGlobalVariables, l1ToL2Messages, previousBlockHeader);
334
- // Deadline for processing depends on whether we're proposing a block
335
- const secondsIntoSlot = this.getSecondsIntoSlot(slot);
336
- const processingEndTimeWithinSlot = opts.validateOnly
337
- ? this.timetable.getValidatorReexecTimeEnd(secondsIntoSlot)
338
- : this.timetable.getBlockProposalExecTimeEnd(secondsIntoSlot);
339
- // Deadline is only set if enforceTimeTable is enabled.
340
- const deadline = this.enforceTimeTable
341
- ? new Date((this.getSlotStartTimestamp(slot) + processingEndTimeWithinSlot) * 1000)
342
- : undefined;
343
- this.log.verbose(`Processing pending txs`, {
344
- slot,
345
- slotStart: new Date(this.getSlotStartTimestamp(slot) * 1000),
346
- now: new Date(this.dateProvider.now()),
347
- deadline,
348
- });
349
- const validators = createValidatorsForBlockBuilding(publicProcessorFork, this.contractDataSource, newGlobalVariables, !!this.config.enforceFees, this.allowedInSetup);
350
- // TODO(#11000): Public processor should just handle processing, one tx at a time. It should be responsibility
351
- // of the sequencer to update world state and iterate over txs. We should refactor this along with unifying the
352
- // publicProcessorFork and orchestratorFork, to avoid doing tree insertions twice when building the block.
353
- const proposerLimits = {
354
- maxTransactions: this.maxTxsPerBlock,
355
- maxBlockSize: this.maxBlockSizeInBytes,
356
- maxBlockGas: this.maxBlockGas,
357
- };
358
- const limits = opts.validateOnly ? { deadline } : { deadline, ...proposerLimits };
359
- const [publicProcessorDuration, [processedTxs, failedTxs]] = await elapsed(() => processor.process(pendingTxs, limits, validators));
360
- if (!opts.validateOnly && failedTxs.length > 0) {
361
- const failedTxData = failedTxs.map(fail => fail.tx);
362
- const failedTxHashes = await Tx.getHashes(failedTxData);
363
- this.log.verbose(`Dropping failed txs ${failedTxHashes.join(', ')}`);
364
- await this.p2pClient.deleteTxs(failedTxHashes);
365
- }
366
- if (!opts.validateOnly && // We check for minTxCount only if we are proposing a block, not if we are validating it
367
- !this.isFlushing && // And we skip the check when flushing, since we want all pending txs to go out, no matter if too few
368
- this.minTxsPerBlock !== undefined &&
369
- processedTxs.length < this.minTxsPerBlock) {
370
- this.log.warn(`Block ${blockNumber} has too few txs to be proposed (got ${processedTxs.length} but required ${this.minTxsPerBlock})`, { slot, blockNumber, processedTxCount: processedTxs.length });
371
- throw new Error(`Block has too few successful txs to be proposed`);
372
- }
373
- const start = process.hrtime.bigint();
374
- await blockBuilder.addTxs(processedTxs);
375
- const end = process.hrtime.bigint();
376
- const duration = Number(end - start) / 1000;
377
- this.metrics.recordBlockBuilderTreeInsertions(duration);
378
- // All real transactions have been added, set the block as full and pad if needed
379
- const block = await blockBuilder.setBlockCompleted();
380
- // How much public gas was processed
381
- const publicGas = processedTxs.reduce((acc, tx) => acc.add(tx.gasUsed.publicGas), Gas.empty());
382
- return {
383
- block,
384
- publicGas,
385
- publicProcessorDuration,
386
- numMsgs: l1ToL2Messages.length,
387
- numTxs: processedTxs.length,
388
- numFailedTxs: failedTxs.length,
389
- blockBuildingTimer,
390
- };
391
- }
392
- finally {
393
- // We create a fresh processor each time to reset any cached state (eg storage writes)
394
- // We wait a bit to close the forks since the processor may still be working on a dangling tx
395
- // which was interrupted due to the processingDeadline being hit.
396
- // eslint-disable-next-line @typescript-eslint/no-misused-promises
397
- setTimeout(async () => {
398
- try {
399
- await publicProcessorFork.close();
400
- await orchestratorFork.close();
401
- }
402
- catch (err) {
403
- // This can happen if the sequencer is stopped before we hit this timeout.
404
- this.log.warn(`Error closing forks for block processing`, err);
405
- }
406
- }, 5000);
407
- }
408
- }
409
- /**
410
- * @notice Build and propose a block to the chain
411
- *
412
- * @dev MUST throw instead of exiting early to ensure that world-state
413
- * is being rolled back if the block is dropped.
414
- *
415
- * @param pendingTxs - Iterable of pending transactions to construct the block from
416
- * @param proposalHeader - The partial header constructed for the proposal
417
- */
418
- async buildBlockAndEnqueuePublish(pendingTxs, proposalHeader) {
419
- await this.publisher.validateBlockForSubmission(proposalHeader);
420
- const newGlobalVariables = proposalHeader.globalVariables;
421
- const blockNumber = newGlobalVariables.blockNumber.toNumber();
422
- const slot = newGlobalVariables.slotNumber.toBigInt();
423
- // this.metrics.recordNewBlock(blockNumber, validTxs.length);
424
- const workTimer = new Timer();
425
- this.setState(SequencerState.CREATING_BLOCK, slot);
426
- try {
427
- const buildBlockRes = await this.buildBlock(pendingTxs, newGlobalVariables);
428
- const { publicGas, block, publicProcessorDuration, numTxs, numMsgs, blockBuildingTimer } = buildBlockRes;
429
- this.metrics.recordBuiltBlock(workTimer.ms(), publicGas.l2Gas);
430
- // TODO(@PhilWindle) We should probably periodically check for things like another
431
- // block being published before ours instead of just waiting on our block
432
- await this.publisher.validateBlockForSubmission(block.header);
433
- const blockStats = {
434
- eventName: 'l2-block-built',
435
- creator: this.publisher.getSenderAddress().toString(),
436
- duration: workTimer.ms(),
437
- publicProcessDuration: publicProcessorDuration,
438
- rollupCircuitsDuration: blockBuildingTimer.ms(),
439
- ...block.getStats(),
440
- };
441
- const blockHash = await block.hash();
442
- const txHashes = block.body.txEffects.map(tx => tx.txHash);
443
- this.log.info(`Built block ${block.number} for slot ${slot} with ${numTxs} txs and ${numMsgs} messages. ${publicGas.l2Gas / workTimer.s()} mana/s`, {
444
- blockHash,
445
- globalVariables: block.header.globalVariables.toInspect(),
446
- txHashes,
447
- ...blockStats,
448
- });
449
- this.log.debug('Collecting attestations');
450
- const stopCollectingAttestationsTimer = this.metrics.startCollectingAttestationsTimer();
451
- const attestations = await this.collectAttestations(block, txHashes);
452
- if (attestations !== undefined) {
453
- this.log.verbose(`Collected ${attestations.length} attestations`, { blockHash, blockNumber });
454
- }
455
- stopCollectingAttestationsTimer();
456
- return this.enqueuePublishL2Block(block, attestations, txHashes);
457
- }
458
- catch (err) {
459
- this.metrics.recordFailedBlock();
460
- throw err;
461
- }
462
- }
463
- async collectAttestations(block, txHashes) {
464
- // TODO(https://github.com/AztecProtocol/aztec-packages/issues/7962): inefficient to have a round trip in here - this should be cached
465
- const committee = await this.publisher.getCurrentEpochCommittee();
466
- if (committee.length === 0) {
467
- this.log.verbose(`Attesting committee is empty`);
468
- return undefined;
469
- }
470
- else {
471
- this.log.debug(`Attesting committee length is ${committee.length}`);
472
- }
473
- if (!this.validatorClient) {
474
- const msg = 'Missing validator client: Cannot collect attestations';
475
- this.log.error(msg);
476
- throw new Error(msg);
477
- }
478
- const numberOfRequiredAttestations = Math.floor((committee.length * 2) / 3) + 1;
479
- const slotNumber = block.header.globalVariables.slotNumber.toBigInt();
480
- this.setState(SequencerState.COLLECTING_ATTESTATIONS, slotNumber);
481
- this.log.debug('Creating block proposal for validators');
482
- const proposal = await this.validatorClient.createBlockProposal(block.header, block.archive.root, txHashes);
483
- if (!proposal) {
484
- const msg = `Failed to create block proposal`;
485
- throw new Error(msg);
486
- }
487
- this.log.debug('Broadcasting block proposal to validators');
488
- this.validatorClient.broadcastBlockProposal(proposal);
489
- const attestationTimeAllowed = this.enforceTimeTable
490
- ? this.timetable.getMaxAllowedTime(SequencerState.PUBLISHING_BLOCK)
491
- : this.aztecSlotDuration;
492
- const attestationDeadline = new Date(this.dateProvider.now() + attestationTimeAllowed * 1000);
493
- const attestations = await this.validatorClient.collectAttestations(proposal, numberOfRequiredAttestations, attestationDeadline);
494
- // note: the smart contract requires that the signatures are provided in the order of the committee
495
- return orderAttestations(attestations, committee);
394
+ throw new Error(`Block has too few successful txs to be proposed`);
496
395
  }
497
- async createProofClaimForPreviousEpoch(slotNumber) {
396
+ const start = process.hrtime.bigint();
397
+ await blockBuilder.addTxs(processedTxs);
398
+ const end = process.hrtime.bigint();
399
+ const duration = Number(end - start) / 1_000;
400
+ this.metrics.recordBlockBuilderTreeInsertions(duration);
401
+ // All real transactions have been added, set the block as full and pad if needed
402
+ const block = await blockBuilder.setBlockCompleted();
403
+ // How much public gas was processed
404
+ const publicGas = processedTxs.reduce((acc, tx)=>acc.add(tx.gasUsed.publicGas), Gas.empty());
405
+ return {
406
+ block,
407
+ publicGas,
408
+ publicProcessorDuration,
409
+ numMsgs: l1ToL2Messages.length,
410
+ numTxs: processedTxs.length,
411
+ numFailedTxs: failedTxs.length,
412
+ blockBuildingTimer
413
+ };
414
+ } finally{
415
+ // We create a fresh processor each time to reset any cached state (eg storage writes)
416
+ // We wait a bit to close the forks since the processor may still be working on a dangling tx
417
+ // which was interrupted due to the processingDeadline being hit.
418
+ // eslint-disable-next-line @typescript-eslint/no-misused-promises
419
+ setTimeout(async ()=>{
498
420
  try {
499
- // Find out which epoch we are currently in
500
- const epochToProve = await this.publisher.getClaimableEpoch();
501
- if (epochToProve === undefined) {
502
- this.log.trace(`No epoch to claim at slot ${slotNumber}`);
503
- return undefined;
504
- }
505
- // Get quotes for the epoch to be proven
506
- this.log.debug(`Collecting proof quotes for epoch ${epochToProve}`);
507
- const p2pQuotes = await this.p2pClient
508
- .getEpochProofQuotes(epochToProve)
509
- .then(quotes => quotes
510
- .filter(x => x.payload.validUntilSlot >= slotNumber)
511
- .filter(x => x.payload.epochToProve === epochToProve));
512
- this.log.verbose(`Retrieved ${p2pQuotes.length} quotes for slot ${slotNumber} epoch ${epochToProve}`, {
513
- epochToProve,
514
- slotNumber,
515
- quotes: p2pQuotes.map(q => q.payload),
516
- });
517
- if (!p2pQuotes.length) {
518
- return undefined;
519
- }
520
- // ensure these quotes are still valid for the slot and have the contract validate them
521
- const validQuotes = await this.publisher.filterValidQuotes(p2pQuotes);
522
- if (!validQuotes.length) {
523
- this.log.warn(`Failed to find any valid proof quotes`);
524
- return undefined;
525
- }
526
- // pick the quote with the lowest fee
527
- const sortedQuotes = validQuotes.sort((a, b) => a.payload.basisPointFee - b.payload.basisPointFee);
528
- const quote = sortedQuotes[0];
529
- this.log.info(`Selected proof quote for proof claim`, { quote: quote.toInspect() });
530
- return quote;
531
- }
532
- catch (err) {
533
- this.log.error(`Failed to create proof claim for previous epoch`, err, { slotNumber });
534
- return undefined;
535
- }
536
- }
537
- /**
538
- * Publishes the L2Block to the rollup contract.
539
- * @param block - The L2Block to be published.
540
- */
541
- async enqueuePublishL2Block(block, attestations, txHashes) {
542
- // Publishes new block to the network and awaits the tx to be mined
543
- this.setState(SequencerState.PUBLISHING_BLOCK, block.header.globalVariables.slotNumber.toBigInt());
544
- // Time out tx at the end of the slot
545
- const slot = block.header.globalVariables.slotNumber.toNumber();
546
- const txTimeoutAt = new Date((this.getSlotStartTimestamp(slot) + this.aztecSlotDuration) * 1000);
547
- const enqueued = await this.publisher.enqueueProposeL2Block(block, attestations, txHashes, {
548
- txTimeoutAt,
421
+ await publicProcessorFork.close();
422
+ await orchestratorFork.close();
423
+ } catch (err) {
424
+ // This can happen if the sequencer is stopped before we hit this timeout.
425
+ this.log.warn(`Error closing forks for block processing`, err);
426
+ }
427
+ }, 5000);
428
+ }
429
+ }
430
+ /**
431
+ * @notice Build and propose a block to the chain
432
+ *
433
+ * @dev MUST throw instead of exiting early to ensure that world-state
434
+ * is being rolled back if the block is dropped.
435
+ *
436
+ * @param pendingTxs - Iterable of pending transactions to construct the block from
437
+ * @param proposalHeader - The partial header constructed for the proposal
438
+ */ async buildBlockAndEnqueuePublish(pendingTxs, proposalHeader) {
439
+ await this.publisher.validateBlockForSubmission(proposalHeader);
440
+ const newGlobalVariables = proposalHeader.globalVariables;
441
+ const blockNumber = newGlobalVariables.blockNumber.toNumber();
442
+ const slot = newGlobalVariables.slotNumber.toBigInt();
443
+ // this.metrics.recordNewBlock(blockNumber, validTxs.length);
444
+ const workTimer = new Timer();
445
+ this.setState(SequencerState.CREATING_BLOCK, slot);
446
+ try {
447
+ const buildBlockRes = await this.buildBlock(pendingTxs, newGlobalVariables);
448
+ const { publicGas, block, publicProcessorDuration, numTxs, numMsgs, blockBuildingTimer } = buildBlockRes;
449
+ this.metrics.recordBuiltBlock(workTimer.ms(), publicGas.l2Gas);
450
+ // TODO(@PhilWindle) We should probably periodically check for things like another
451
+ // block being published before ours instead of just waiting on our block
452
+ await this.publisher.validateBlockForSubmission(block.header);
453
+ const blockStats = {
454
+ eventName: 'l2-block-built',
455
+ creator: this.publisher.getSenderAddress().toString(),
456
+ duration: workTimer.ms(),
457
+ publicProcessDuration: publicProcessorDuration,
458
+ rollupCircuitsDuration: blockBuildingTimer.ms(),
459
+ ...block.getStats()
460
+ };
461
+ const blockHash = await block.hash();
462
+ const txHashes = block.body.txEffects.map((tx)=>tx.txHash);
463
+ this.log.info(`Built block ${block.number} for slot ${slot} with ${numTxs} txs and ${numMsgs} messages. ${publicGas.l2Gas / workTimer.s()} mana/s`, {
464
+ blockHash,
465
+ globalVariables: block.header.globalVariables.toInspect(),
466
+ txHashes,
467
+ ...blockStats
468
+ });
469
+ this.log.debug('Collecting attestations');
470
+ const stopCollectingAttestationsTimer = this.metrics.startCollectingAttestationsTimer();
471
+ const attestations = await this.collectAttestations(block, txHashes);
472
+ if (attestations !== undefined) {
473
+ this.log.verbose(`Collected ${attestations.length} attestations`, {
474
+ blockHash,
475
+ blockNumber
549
476
  });
550
- if (!enqueued) {
551
- throw new Error(`Failed to enqueue publish of block ${block.number}`);
552
- }
553
- }
554
- async claimEpochProofRightIfAvailable(slotNumber) {
555
- const proofQuote = await this.createProofClaimForPreviousEpoch(slotNumber);
556
- if (proofQuote === undefined) {
557
- return;
558
- }
559
- const epoch = proofQuote.payload.epochToProve;
560
- const ctx = { slotNumber, epoch, quote: proofQuote.toInspect() };
561
- this.log.verbose(`Claiming proof right for epoch ${epoch}`, ctx);
562
- const enqueued = this.publisher.enqueueClaimEpochProofRight(proofQuote);
563
- if (!enqueued) {
564
- throw new Error(`Failed to enqueue claim of proof right for epoch ${epoch}`);
565
- }
566
- this.log.info(`Enqueued claim of proof right for epoch ${epoch}`, ctx);
567
- return epoch;
568
- }
569
- /**
570
- * Returns whether all dependencies have caught up.
571
- * We don't check against the previous block submitted since it may have been reorg'd out.
572
- * @returns Boolean indicating if our dependencies are synced to the latest block.
573
- */
574
- async getChainTip() {
575
- const syncedBlocks = await Promise.all([
576
- this.worldState.status().then((s) => s.syncedToL2Block),
577
- this.l2BlockSource.getL2Tips().then(t => t.latest),
578
- this.p2pClient.getStatus().then(p2p => p2p.syncedToL2Block),
579
- this.l1ToL2MessageSource.getBlockNumber(),
580
- ]);
581
- const [worldState, l2BlockSource, p2p, l1ToL2MessageSource] = syncedBlocks;
582
- const result =
583
- // check that world state has caught up with archiver
584
- // note that the archiver reports undefined hash for the genesis block
585
- // because it doesn't have access to world state to compute it (facepalm)
586
- (l2BlockSource.hash === undefined || worldState.hash === l2BlockSource.hash) &&
587
- // and p2p client and message source are at least at the same block
588
- // this should change to hashes once p2p client handles reorgs
589
- // and once we stop pretending that the l1tol2message source is not
590
- // just the archiver under a different name
591
- (!l2BlockSource.hash || p2p.hash === l2BlockSource.hash) &&
592
- l1ToL2MessageSource === l2BlockSource.number;
593
- this.log.debug(`Sequencer sync check ${result ? 'succeeded' : 'failed'}`, {
594
- worldStateNumber: worldState.number,
595
- worldStateHash: worldState.hash,
596
- l2BlockSourceNumber: l2BlockSource.number,
597
- l2BlockSourceHash: l2BlockSource.hash,
598
- p2pNumber: p2p.number,
599
- p2pHash: p2p.hash,
600
- l1ToL2MessageSourceNumber: l1ToL2MessageSource,
601
- });
602
- if (!result) {
603
- return undefined;
604
- }
605
- if (worldState.number >= INITIAL_L2_BLOCK_NUM) {
606
- const block = await this.l2BlockSource.getBlock(worldState.number);
607
- if (!block) {
608
- // this shouldn't really happen because a moment ago we checked that all components were in synch
609
- return undefined;
610
- }
611
- return { blockNumber: block.number, archive: block.archive.root };
612
- }
613
- else {
614
- return { blockNumber: INITIAL_L2_BLOCK_NUM - 1, archive: new Fr(GENESIS_ARCHIVE_ROOT) };
615
- }
616
- }
617
- getSlotStartTimestamp(slotNumber) {
618
- return Number(this.l1Constants.l1GenesisTime) + Number(slotNumber) * this.l1Constants.slotDuration;
619
- }
620
- getSecondsIntoSlot(slotNumber) {
621
- const slotStartTimestamp = this.getSlotStartTimestamp(slotNumber);
622
- return Number((this.dateProvider.now() / 1000 - slotStartTimestamp).toFixed(3));
623
- }
624
- get aztecSlotDuration() {
625
- return this.l1Constants.slotDuration;
626
- }
627
- get coinbase() {
628
- return this._coinbase;
629
477
  }
630
- get feeRecipient() {
631
- return this._feeRecipient;
478
+ stopCollectingAttestationsTimer();
479
+ return this.enqueuePublishL2Block(block, attestations, txHashes);
480
+ } catch (err) {
481
+ this.metrics.recordFailedBlock();
482
+ throw err;
483
+ }
484
+ }
485
+ async collectAttestations(block, txHashes) {
486
+ // TODO(https://github.com/AztecProtocol/aztec-packages/issues/7962): inefficient to have a round trip in here - this should be cached
487
+ const committee = await this.publisher.getCurrentEpochCommittee();
488
+ if (committee.length === 0) {
489
+ this.log.verbose(`Attesting committee is empty`);
490
+ return undefined;
491
+ } else {
492
+ this.log.debug(`Attesting committee length is ${committee.length}`);
493
+ }
494
+ if (!this.validatorClient) {
495
+ const msg = 'Missing validator client: Cannot collect attestations';
496
+ this.log.error(msg);
497
+ throw new Error(msg);
498
+ }
499
+ const numberOfRequiredAttestations = Math.floor(committee.length * 2 / 3) + 1;
500
+ const slotNumber = block.header.globalVariables.slotNumber.toBigInt();
501
+ this.setState(SequencerState.COLLECTING_ATTESTATIONS, slotNumber);
502
+ this.log.debug('Creating block proposal for validators');
503
+ const proposal = await this.validatorClient.createBlockProposal(block.header, block.archive.root, txHashes);
504
+ if (!proposal) {
505
+ const msg = `Failed to create block proposal`;
506
+ throw new Error(msg);
507
+ }
508
+ this.log.debug('Broadcasting block proposal to validators');
509
+ this.validatorClient.broadcastBlockProposal(proposal);
510
+ const attestationTimeAllowed = this.enforceTimeTable ? this.timetable.getMaxAllowedTime(SequencerState.PUBLISHING_BLOCK) : this.aztecSlotDuration;
511
+ const attestationDeadline = new Date(this.dateProvider.now() + attestationTimeAllowed * 1000);
512
+ const attestations = await this.validatorClient.collectAttestations(proposal, numberOfRequiredAttestations, attestationDeadline);
513
+ // note: the smart contract requires that the signatures are provided in the order of the committee
514
+ return orderAttestations(attestations, committee);
515
+ }
516
+ /**
517
+ * Publishes the L2Block to the rollup contract.
518
+ * @param block - The L2Block to be published.
519
+ */ async enqueuePublishL2Block(block, attestations, txHashes) {
520
+ // Publishes new block to the network and awaits the tx to be mined
521
+ this.setState(SequencerState.PUBLISHING_BLOCK, block.header.globalVariables.slotNumber.toBigInt());
522
+ // Time out tx at the end of the slot
523
+ const slot = block.header.globalVariables.slotNumber.toNumber();
524
+ const txTimeoutAt = new Date((this.getSlotStartTimestamp(slot) + this.aztecSlotDuration) * 1000);
525
+ const enqueued = await this.publisher.enqueueProposeL2Block(block, attestations, txHashes, {
526
+ txTimeoutAt
527
+ });
528
+ if (!enqueued) {
529
+ throw new Error(`Failed to enqueue publish of block ${block.number}`);
530
+ }
531
+ }
532
+ /**
533
+ * Returns whether all dependencies have caught up.
534
+ * We don't check against the previous block submitted since it may have been reorg'd out.
535
+ * @returns Boolean indicating if our dependencies are synced to the latest block.
536
+ */ async getChainTip() {
537
+ const syncedBlocks = await Promise.all([
538
+ this.worldState.status().then((s)=>{
539
+ return {
540
+ number: s.syncSummary.latestBlockNumber,
541
+ hash: s.syncSummary.latestBlockHash
542
+ };
543
+ }),
544
+ this.l2BlockSource.getL2Tips().then((t)=>t.latest),
545
+ this.p2pClient.getStatus().then((p2p)=>p2p.syncedToL2Block),
546
+ this.l1ToL2MessageSource.getBlockNumber()
547
+ ]);
548
+ const [worldState, l2BlockSource, p2p, l1ToL2MessageSource] = syncedBlocks;
549
+ const result = // check that world state has caught up with archiver
550
+ // note that the archiver reports undefined hash for the genesis block
551
+ // because it doesn't have access to world state to compute it (facepalm)
552
+ (l2BlockSource.hash === undefined || worldState.hash === l2BlockSource.hash) && // and p2p client and message source are at least at the same block
553
+ // this should change to hashes once p2p client handles reorgs
554
+ // and once we stop pretending that the l1tol2message source is not
555
+ // just the archiver under a different name
556
+ (!l2BlockSource.hash || p2p.hash === l2BlockSource.hash) && l1ToL2MessageSource === l2BlockSource.number;
557
+ this.log.debug(`Sequencer sync check ${result ? 'succeeded' : 'failed'}`, {
558
+ worldStateNumber: worldState.number,
559
+ worldStateHash: worldState.hash,
560
+ l2BlockSourceNumber: l2BlockSource.number,
561
+ l2BlockSourceHash: l2BlockSource.hash,
562
+ p2pNumber: p2p.number,
563
+ p2pHash: p2p.hash,
564
+ l1ToL2MessageSourceNumber: l1ToL2MessageSource
565
+ });
566
+ if (!result) {
567
+ return undefined;
568
+ }
569
+ if (worldState.number >= INITIAL_L2_BLOCK_NUM) {
570
+ const block = await this.l2BlockSource.getBlock(worldState.number);
571
+ if (!block) {
572
+ // this shouldn't really happen because a moment ago we checked that all components were in synch
573
+ return undefined;
632
574
  }
633
- },
634
- (() => {
635
- const _metadata = typeof Symbol === "function" && Symbol.metadata ? Object.create(null) : void 0;
636
- _work_decorators = [trackSpan('Sequencer.work')];
637
- _buildBlockAndEnqueuePublish_decorators = [trackSpan('Sequencer.buildBlockAndEnqueuePublish', (_validTxs, proposalHeader) => ({
638
- [Attributes.BLOCK_NUMBER]: proposalHeader.globalVariables.blockNumber.toNumber(),
639
- }))];
640
- _collectAttestations_decorators = [trackSpan('Sequencer.collectAttestations', (block, txHashes) => ({
641
- [Attributes.BLOCK_NUMBER]: block.number,
642
- [Attributes.BLOCK_ARCHIVE]: block.archive.toString(),
643
- [Attributes.BLOCK_TXS_COUNT]: txHashes.length,
644
- }))];
645
- _enqueuePublishL2Block_decorators = [trackSpan('Sequencer.enqueuePublishL2Block', block => ({
646
- [Attributes.BLOCK_NUMBER]: block.number,
647
- }))];
648
- _claimEpochProofRightIfAvailable_decorators = [trackSpan('Sequencer.claimEpochProofRightIfAvailable', slotNumber => ({ [Attributes.SLOT_NUMBER]: Number(slotNumber) }), epoch => ({ [Attributes.EPOCH_NUMBER]: Number(epoch) }))];
649
- __esDecorate(_a, null, _work_decorators, { kind: "method", name: "work", static: false, private: false, access: { has: obj => "work" in obj, get: obj => obj.work }, metadata: _metadata }, null, _instanceExtraInitializers);
650
- __esDecorate(_a, null, _buildBlockAndEnqueuePublish_decorators, { kind: "method", name: "buildBlockAndEnqueuePublish", static: false, private: false, access: { has: obj => "buildBlockAndEnqueuePublish" in obj, get: obj => obj.buildBlockAndEnqueuePublish }, metadata: _metadata }, null, _instanceExtraInitializers);
651
- __esDecorate(_a, null, _collectAttestations_decorators, { kind: "method", name: "collectAttestations", static: false, private: false, access: { has: obj => "collectAttestations" in obj, get: obj => obj.collectAttestations }, metadata: _metadata }, null, _instanceExtraInitializers);
652
- __esDecorate(_a, null, _enqueuePublishL2Block_decorators, { kind: "method", name: "enqueuePublishL2Block", static: false, private: false, access: { has: obj => "enqueuePublishL2Block" in obj, get: obj => obj.enqueuePublishL2Block }, metadata: _metadata }, null, _instanceExtraInitializers);
653
- __esDecorate(_a, null, _claimEpochProofRightIfAvailable_decorators, { kind: "method", name: "claimEpochProofRightIfAvailable", static: false, private: false, access: { has: obj => "claimEpochProofRightIfAvailable" in obj, get: obj => obj.claimEpochProofRightIfAvailable }, metadata: _metadata }, null, _instanceExtraInitializers);
654
- if (_metadata) Object.defineProperty(_a, Symbol.metadata, { enumerable: true, configurable: true, writable: true, value: _metadata });
655
- })(),
656
- _a;
657
- })();
658
- export { Sequencer };
659
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2VxdWVuY2VyLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vc3JjL3NlcXVlbmNlci9zZXF1ZW5jZXIudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IjtBQUFBLE9BQU8sRUFNTCxxQkFBcUIsRUFDckIsRUFBRSxHQUdILE1BQU0sc0JBQXNCLENBQUM7QUFHOUIsT0FBTyxFQUNMLHNCQUFzQixFQUN0QixXQUFXLEVBQ1gsaUJBQWlCLEVBRWpCLG9CQUFvQixFQUNwQixHQUFHLEVBRUgsb0JBQW9CLEVBQ3BCLGNBQWMsR0FDZixNQUFNLG9CQUFvQixDQUFDO0FBQzVCLE9BQU8sRUFBRSxZQUFZLEVBQUUsTUFBTSxpQ0FBaUMsQ0FBQztBQUMvRCxPQUFPLEVBQUUsSUFBSSxFQUFFLE1BQU0sOEJBQThCLENBQUM7QUFDcEQsT0FBTyxFQUFFLFVBQVUsRUFBRSxNQUFNLCtCQUErQixDQUFDO0FBQzNELE9BQU8sRUFBRSxFQUFFLEVBQUUsTUFBTSwwQkFBMEIsQ0FBQztBQUM5QyxPQUFPLEVBQUUsWUFBWSxFQUFFLE1BQU0sdUJBQXVCLENBQUM7QUFDckQsT0FBTyxFQUFFLGNBQWMsRUFBRSxNQUFNLG1DQUFtQyxDQUFDO0FBQ25FLE9BQU8sRUFBRSxjQUFjLEVBQUUsTUFBTSwyQkFBMkIsQ0FBQztBQUMzRCxPQUFPLEVBQXFCLEtBQUssRUFBRSxPQUFPLEVBQUUsTUFBTSx5QkFBeUIsQ0FBQztBQUk1RSxPQUFPLEVBQUUsVUFBVSxFQUFxQyxrQkFBa0IsRUFBRSxTQUFTLEVBQUUsTUFBTSx5QkFBeUIsQ0FBQztBQUl2SCxPQUFPLEVBQTJCLFFBQVEsRUFBRSxNQUFNLHFDQUFxQyxDQUFDO0FBRXhGLE9BQU8sRUFBRSxnQ0FBZ0MsRUFBRSxNQUFNLHlDQUF5QyxDQUFDO0FBQzNGLE9BQU8sRUFBRSwrQkFBK0IsRUFBRSxNQUFNLGNBQWMsQ0FBQztBQUUvRCxPQUFPLEVBQUUsZ0JBQWdCLEVBQUUsTUFBTSxjQUFjLENBQUM7QUFDaEQsT0FBTyxF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
575
+ return {
576
+ blockNumber: block.number,
577
+ archive: block.archive.root
578
+ };
579
+ } else {
580
+ const archive = new Fr((await this.worldState.getCommitted().getTreeInfo(MerkleTreeId.ARCHIVE)).root);
581
+ return {
582
+ blockNumber: INITIAL_L2_BLOCK_NUM - 1,
583
+ archive
584
+ };
585
+ }
586
+ }
587
+ getSlotStartTimestamp(slotNumber) {
588
+ return Number(this.l1Constants.l1GenesisTime) + Number(slotNumber) * this.l1Constants.slotDuration;
589
+ }
590
+ getSecondsIntoSlot(slotNumber) {
591
+ const slotStartTimestamp = this.getSlotStartTimestamp(slotNumber);
592
+ return Number((this.dateProvider.now() / 1000 - slotStartTimestamp).toFixed(3));
593
+ }
594
+ get aztecSlotDuration() {
595
+ return this.l1Constants.slotDuration;
596
+ }
597
+ get coinbase() {
598
+ return this._coinbase;
599
+ }
600
+ get feeRecipient() {
601
+ return this._feeRecipient;
602
+ }
603
+ }
604
+ _ts_decorate([
605
+ trackSpan('Sequencer.work')
606
+ ], Sequencer.prototype, "work", null);
607
+ _ts_decorate([
608
+ trackSpan('Sequencer.buildBlockAndEnqueuePublish', (_validTxs, proposalHeader)=>({
609
+ [Attributes.BLOCK_NUMBER]: proposalHeader.globalVariables.blockNumber.toNumber()
610
+ }))
611
+ ], Sequencer.prototype, "buildBlockAndEnqueuePublish", null);
612
+ _ts_decorate([
613
+ trackSpan('Sequencer.collectAttestations', (block, txHashes)=>({
614
+ [Attributes.BLOCK_NUMBER]: block.number,
615
+ [Attributes.BLOCK_ARCHIVE]: block.archive.toString(),
616
+ [Attributes.BLOCK_TXS_COUNT]: txHashes.length
617
+ }))
618
+ ], Sequencer.prototype, "collectAttestations", null);
619
+ _ts_decorate([
620
+ trackSpan('Sequencer.enqueuePublishL2Block', (block)=>({
621
+ [Attributes.BLOCK_NUMBER]: block.number
622
+ }))
623
+ ], Sequencer.prototype, "enqueuePublishL2Block", null);