@aztec/sequencer-client 0.0.1-commit.9b94fc1 → 0.0.1-commit.9ee6fcc6

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 (118) hide show
  1. package/dest/client/sequencer-client.d.ts +24 -16
  2. package/dest/client/sequencer-client.d.ts.map +1 -1
  3. package/dest/client/sequencer-client.js +73 -32
  4. package/dest/config.d.ts +35 -9
  5. package/dest/config.d.ts.map +1 -1
  6. package/dest/config.js +106 -44
  7. package/dest/global_variable_builder/global_builder.d.ts +27 -14
  8. package/dest/global_variable_builder/global_builder.d.ts.map +1 -1
  9. package/dest/global_variable_builder/global_builder.js +65 -54
  10. package/dest/global_variable_builder/index.d.ts +2 -2
  11. package/dest/global_variable_builder/index.d.ts.map +1 -1
  12. package/dest/index.d.ts +2 -3
  13. package/dest/index.d.ts.map +1 -1
  14. package/dest/index.js +1 -2
  15. package/dest/publisher/config.d.ts +53 -20
  16. package/dest/publisher/config.d.ts.map +1 -1
  17. package/dest/publisher/config.js +124 -39
  18. package/dest/publisher/index.d.ts +2 -1
  19. package/dest/publisher/index.d.ts.map +1 -1
  20. package/dest/publisher/l1_tx_failed_store/factory.d.ts +11 -0
  21. package/dest/publisher/l1_tx_failed_store/factory.d.ts.map +1 -0
  22. package/dest/publisher/l1_tx_failed_store/factory.js +22 -0
  23. package/dest/publisher/l1_tx_failed_store/failed_tx_store.d.ts +59 -0
  24. package/dest/publisher/l1_tx_failed_store/failed_tx_store.d.ts.map +1 -0
  25. package/dest/publisher/l1_tx_failed_store/failed_tx_store.js +1 -0
  26. package/dest/publisher/l1_tx_failed_store/file_store_failed_tx_store.d.ts +15 -0
  27. package/dest/publisher/l1_tx_failed_store/file_store_failed_tx_store.d.ts.map +1 -0
  28. package/dest/publisher/l1_tx_failed_store/file_store_failed_tx_store.js +34 -0
  29. package/dest/publisher/l1_tx_failed_store/index.d.ts +4 -0
  30. package/dest/publisher/l1_tx_failed_store/index.d.ts.map +1 -0
  31. package/dest/publisher/l1_tx_failed_store/index.js +2 -0
  32. package/dest/publisher/sequencer-publisher-factory.d.ts +15 -6
  33. package/dest/publisher/sequencer-publisher-factory.d.ts.map +1 -1
  34. package/dest/publisher/sequencer-publisher-factory.js +28 -3
  35. package/dest/publisher/sequencer-publisher-metrics.d.ts +3 -3
  36. package/dest/publisher/sequencer-publisher-metrics.d.ts.map +1 -1
  37. package/dest/publisher/sequencer-publisher-metrics.js +23 -86
  38. package/dest/publisher/sequencer-publisher.d.ts +79 -49
  39. package/dest/publisher/sequencer-publisher.d.ts.map +1 -1
  40. package/dest/publisher/sequencer-publisher.js +932 -155
  41. package/dest/sequencer/checkpoint_proposal_job.d.ts +108 -0
  42. package/dest/sequencer/checkpoint_proposal_job.d.ts.map +1 -0
  43. package/dest/sequencer/checkpoint_proposal_job.js +1289 -0
  44. package/dest/sequencer/checkpoint_voter.d.ts +35 -0
  45. package/dest/sequencer/checkpoint_voter.d.ts.map +1 -0
  46. package/dest/sequencer/checkpoint_voter.js +109 -0
  47. package/dest/sequencer/config.d.ts +3 -2
  48. package/dest/sequencer/config.d.ts.map +1 -1
  49. package/dest/sequencer/events.d.ts +47 -0
  50. package/dest/sequencer/events.d.ts.map +1 -0
  51. package/dest/sequencer/events.js +1 -0
  52. package/dest/sequencer/index.d.ts +4 -2
  53. package/dest/sequencer/index.d.ts.map +1 -1
  54. package/dest/sequencer/index.js +3 -1
  55. package/dest/sequencer/metrics.d.ts +42 -6
  56. package/dest/sequencer/metrics.d.ts.map +1 -1
  57. package/dest/sequencer/metrics.js +227 -72
  58. package/dest/sequencer/sequencer.d.ts +125 -134
  59. package/dest/sequencer/sequencer.d.ts.map +1 -1
  60. package/dest/sequencer/sequencer.js +754 -652
  61. package/dest/sequencer/timetable.d.ts +54 -16
  62. package/dest/sequencer/timetable.d.ts.map +1 -1
  63. package/dest/sequencer/timetable.js +147 -62
  64. package/dest/sequencer/types.d.ts +3 -0
  65. package/dest/sequencer/types.d.ts.map +1 -0
  66. package/dest/sequencer/types.js +1 -0
  67. package/dest/sequencer/utils.d.ts +14 -8
  68. package/dest/sequencer/utils.d.ts.map +1 -1
  69. package/dest/sequencer/utils.js +7 -4
  70. package/dest/test/index.d.ts +6 -7
  71. package/dest/test/index.d.ts.map +1 -1
  72. package/dest/test/mock_checkpoint_builder.d.ts +95 -0
  73. package/dest/test/mock_checkpoint_builder.d.ts.map +1 -0
  74. package/dest/test/mock_checkpoint_builder.js +231 -0
  75. package/dest/test/utils.d.ts +53 -0
  76. package/dest/test/utils.d.ts.map +1 -0
  77. package/dest/test/utils.js +104 -0
  78. package/package.json +31 -30
  79. package/src/client/sequencer-client.ts +97 -55
  80. package/src/config.ts +124 -53
  81. package/src/global_variable_builder/global_builder.ts +76 -73
  82. package/src/global_variable_builder/index.ts +1 -1
  83. package/src/index.ts +1 -7
  84. package/src/publisher/config.ts +163 -50
  85. package/src/publisher/index.ts +3 -0
  86. package/src/publisher/l1_tx_failed_store/factory.ts +32 -0
  87. package/src/publisher/l1_tx_failed_store/failed_tx_store.ts +55 -0
  88. package/src/publisher/l1_tx_failed_store/file_store_failed_tx_store.ts +46 -0
  89. package/src/publisher/l1_tx_failed_store/index.ts +3 -0
  90. package/src/publisher/sequencer-publisher-factory.ts +43 -10
  91. package/src/publisher/sequencer-publisher-metrics.ts +19 -71
  92. package/src/publisher/sequencer-publisher.ts +635 -200
  93. package/src/sequencer/README.md +531 -0
  94. package/src/sequencer/checkpoint_proposal_job.ts +1049 -0
  95. package/src/sequencer/checkpoint_voter.ts +130 -0
  96. package/src/sequencer/config.ts +2 -1
  97. package/src/sequencer/events.ts +27 -0
  98. package/src/sequencer/index.ts +3 -1
  99. package/src/sequencer/metrics.ts +282 -82
  100. package/src/sequencer/sequencer.ts +521 -859
  101. package/src/sequencer/timetable.ts +178 -83
  102. package/src/sequencer/types.ts +6 -0
  103. package/src/sequencer/utils.ts +18 -9
  104. package/src/test/index.ts +5 -6
  105. package/src/test/mock_checkpoint_builder.ts +323 -0
  106. package/src/test/utils.ts +167 -0
  107. package/dest/sequencer/block_builder.d.ts +0 -27
  108. package/dest/sequencer/block_builder.d.ts.map +0 -1
  109. package/dest/sequencer/block_builder.js +0 -134
  110. package/dest/tx_validator/nullifier_cache.d.ts +0 -14
  111. package/dest/tx_validator/nullifier_cache.d.ts.map +0 -1
  112. package/dest/tx_validator/nullifier_cache.js +0 -24
  113. package/dest/tx_validator/tx_validator_factory.d.ts +0 -17
  114. package/dest/tx_validator/tx_validator_factory.d.ts.map +0 -1
  115. package/dest/tx_validator/tx_validator_factory.js +0 -53
  116. package/src/sequencer/block_builder.ts +0 -222
  117. package/src/tx_validator/nullifier_cache.ts +0 -30
  118. package/src/tx_validator/tx_validator_factory.ts +0 -132
@@ -1,4 +1,4 @@
1
- import { Attributes, Metrics, ValueType } from '@aztec/telemetry-client';
1
+ import { Attributes, Metrics, createUpDownCounterWithDefault } from '@aztec/telemetry-client';
2
2
  import { formatUnits } from 'viem';
3
3
  export class SequencerMetrics {
4
4
  rollup;
@@ -17,86 +17,107 @@ export class SequencerMetrics {
17
17
  slots;
18
18
  filledSlots;
19
19
  blockProposalFailed;
20
- blockProposalSuccess;
21
- blockProposalPrecheckFailed;
20
+ checkpointProposalSuccess;
21
+ checkpointPrecheckFailed;
22
+ checkpointProposalFailed;
23
+ checkpointSuccess;
22
24
  slashingAttempts;
25
+ checkpointAttestationDelay;
26
+ checkpointBuildDuration;
27
+ checkpointBlockCount;
28
+ checkpointTxCount;
29
+ checkpointTotalMana;
30
+ pipelineDepth;
31
+ pipelineDiscards;
32
+ // Fisherman fee analysis metrics
33
+ fishermanWouldBeIncluded;
34
+ fishermanTimeBeforeBlock;
35
+ fishermanPendingBlobTxCount;
36
+ fishermanIncludedBlobTxCount;
37
+ fishermanPendingBlobCount;
38
+ fishermanIncludedBlobCount;
39
+ fishermanBlockBlobsFull;
40
+ fishermanMaxBlobCapacity;
41
+ fishermanCalculatedPriorityFee;
42
+ fishermanPriorityFeeDelta;
43
+ fishermanEstimatedCost;
44
+ fishermanEstimatedOverpayment;
45
+ fishermanMinedBlobTxPriorityFee;
46
+ fishermanMinedBlobTxTotalCost;
23
47
  lastSeenSlot;
24
48
  constructor(client, rollup, name = 'Sequencer'){
25
49
  this.rollup = rollup;
26
50
  this.meter = client.getMeter(name);
27
51
  this.tracer = client.getTracer(name);
28
- this.blockCounter = this.meter.createUpDownCounter(Metrics.SEQUENCER_BLOCK_COUNT);
29
- this.blockBuildDuration = this.meter.createHistogram(Metrics.SEQUENCER_BLOCK_BUILD_DURATION, {
30
- unit: 'ms',
31
- description: 'Duration to build a block',
32
- valueType: ValueType.INT
33
- });
34
- this.blockBuildManaPerSecond = this.meter.createGauge(Metrics.SEQUENCER_BLOCK_BUILD_MANA_PER_SECOND, {
35
- unit: 'mana/s',
36
- description: 'Mana per second when building a block',
37
- valueType: ValueType.INT
38
- });
39
- this.stateTransitionBufferDuration = this.meter.createHistogram(Metrics.SEQUENCER_STATE_TRANSITION_BUFFER_DURATION, {
40
- unit: 'ms',
41
- description: 'The time difference between when the sequencer needed to transition to a new state and when it actually did.',
42
- valueType: ValueType.INT
43
- });
44
- // Init gauges and counters
45
- this.blockCounter.add(0, {
46
- [Attributes.STATUS]: 'failed'
47
- });
48
- this.blockCounter.add(0, {
49
- [Attributes.STATUS]: 'built'
50
- });
51
- this.rewards = this.meter.createGauge(Metrics.SEQUENCER_CURRENT_BLOCK_REWARDS, {
52
- valueType: ValueType.DOUBLE,
53
- description: 'The rewards earned'
54
- });
55
- this.slots = this.meter.createUpDownCounter(Metrics.SEQUENCER_SLOT_COUNT, {
56
- valueType: ValueType.INT,
57
- description: 'The number of slots this sequencer was selected for'
52
+ this.blockCounter = createUpDownCounterWithDefault(this.meter, Metrics.SEQUENCER_BLOCK_COUNT, {
53
+ [Attributes.STATUS]: [
54
+ 'failed',
55
+ 'built'
56
+ ]
58
57
  });
58
+ this.blockBuildDuration = this.meter.createHistogram(Metrics.SEQUENCER_BLOCK_BUILD_DURATION);
59
+ this.blockBuildManaPerSecond = this.meter.createGauge(Metrics.SEQUENCER_BLOCK_BUILD_MANA_PER_SECOND);
60
+ this.stateTransitionBufferDuration = this.meter.createHistogram(Metrics.SEQUENCER_STATE_TRANSITION_BUFFER_DURATION);
61
+ this.checkpointAttestationDelay = this.meter.createHistogram(Metrics.SEQUENCER_CHECKPOINT_ATTESTATION_DELAY);
62
+ this.rewards = this.meter.createGauge(Metrics.SEQUENCER_CURRENT_SLOT_REWARDS);
63
+ this.slots = createUpDownCounterWithDefault(this.meter, Metrics.SEQUENCER_SLOT_COUNT);
59
64
  /**
60
65
  * NOTE: we do not track missed slots as a separate metric. That would be difficult to determine
61
66
  * Instead, use a computed metric, `slots - filledSlots` to get the number of slots a sequencer has missed.
62
- */ this.filledSlots = this.meter.createUpDownCounter(Metrics.SEQUENCER_FILLED_SLOT_COUNT, {
63
- valueType: ValueType.INT,
64
- description: 'The number of slots this sequencer has filled'
67
+ */ this.filledSlots = createUpDownCounterWithDefault(this.meter, Metrics.SEQUENCER_FILLED_SLOT_COUNT);
68
+ this.timeToCollectAttestations = this.meter.createGauge(Metrics.SEQUENCER_COLLECT_ATTESTATIONS_DURATION);
69
+ this.allowanceToCollectAttestations = this.meter.createGauge(Metrics.SEQUENCER_COLLECT_ATTESTATIONS_TIME_ALLOWANCE);
70
+ this.requiredAttestions = this.meter.createGauge(Metrics.SEQUENCER_REQUIRED_ATTESTATIONS_COUNT);
71
+ this.collectedAttestions = this.meter.createGauge(Metrics.SEQUENCER_COLLECTED_ATTESTATIONS_COUNT);
72
+ this.blockProposalFailed = createUpDownCounterWithDefault(this.meter, Metrics.SEQUENCER_BLOCK_PROPOSAL_FAILED_COUNT);
73
+ this.checkpointProposalSuccess = createUpDownCounterWithDefault(this.meter, Metrics.SEQUENCER_CHECKPOINT_PROPOSAL_SUCCESS_COUNT);
74
+ this.checkpointSuccess = createUpDownCounterWithDefault(this.meter, Metrics.SEQUENCER_CHECKPOINT_SUCCESS_COUNT);
75
+ this.checkpointPrecheckFailed = createUpDownCounterWithDefault(this.meter, Metrics.SEQUENCER_CHECKPOINT_PRECHECK_FAILED_COUNT, {
76
+ [Attributes.ERROR_TYPE]: [
77
+ 'slot_already_taken',
78
+ 'rollup_contract_check_failed',
79
+ 'slot_mismatch',
80
+ 'block_number_mismatch'
81
+ ]
65
82
  });
66
- this.timeToCollectAttestations = this.meter.createGauge(Metrics.SEQUENCER_COLLECT_ATTESTATIONS_DURATION, {
67
- description: 'The time spent collecting attestations from committee members',
68
- unit: 'ms',
69
- valueType: ValueType.INT
83
+ this.checkpointProposalFailed = createUpDownCounterWithDefault(this.meter, Metrics.SEQUENCER_CHECKPOINT_PROPOSAL_FAILED_COUNT);
84
+ this.checkpointBuildDuration = this.meter.createHistogram(Metrics.SEQUENCER_CHECKPOINT_BUILD_DURATION);
85
+ this.checkpointBlockCount = this.meter.createGauge(Metrics.SEQUENCER_CHECKPOINT_BLOCK_COUNT);
86
+ this.checkpointTxCount = this.meter.createGauge(Metrics.SEQUENCER_CHECKPOINT_TX_COUNT);
87
+ this.checkpointTotalMana = this.meter.createGauge(Metrics.SEQUENCER_CHECKPOINT_TOTAL_MANA);
88
+ this.slashingAttempts = createUpDownCounterWithDefault(this.meter, Metrics.SEQUENCER_SLASHING_ATTEMPTS_COUNT);
89
+ this.pipelineDepth = this.meter.createGauge(Metrics.SEQUENCER_PIPELINE_DEPTH);
90
+ this.pipelineDiscards = createUpDownCounterWithDefault(this.meter, Metrics.SEQUENCER_PIPELINE_DISCARDS_COUNT);
91
+ this.pipelineDepth.record(0);
92
+ // Fisherman fee analysis metrics
93
+ this.fishermanWouldBeIncluded = createUpDownCounterWithDefault(this.meter, Metrics.FISHERMAN_FEE_ANALYSIS_WOULD_BE_INCLUDED, {
94
+ [Attributes.OK]: [
95
+ true,
96
+ false
97
+ ],
98
+ [Attributes.BLOCK_FULL]: [
99
+ 'true',
100
+ 'false'
101
+ ]
70
102
  });
71
- this.allowanceToCollectAttestations = this.meter.createGauge(Metrics.SEQUENCER_COLLECT_ATTESTATIONS_TIME_ALLOWANCE, {
72
- description: 'Maximum amount of time to collect attestations',
73
- unit: 'ms',
74
- valueType: ValueType.INT
75
- });
76
- this.requiredAttestions = this.meter.createGauge(Metrics.SEQUENCER_REQUIRED_ATTESTATIONS_COUNT, {
77
- valueType: ValueType.INT,
78
- description: 'The minimum number of attestations required to publish a block'
79
- });
80
- this.collectedAttestions = this.meter.createGauge(Metrics.SEQUENCER_COLLECTED_ATTESTATIONS_COUNT, {
81
- valueType: ValueType.INT,
82
- description: 'The minimum number of attestations required to publish a block'
83
- });
84
- this.blockProposalFailed = this.meter.createUpDownCounter(Metrics.SEQUENCER_BLOCK_PROPOSAL_FAILED_COUNT, {
85
- valueType: ValueType.INT,
86
- description: 'The number of times block proposal failed (including validation builds)'
87
- });
88
- this.blockProposalSuccess = this.meter.createUpDownCounter(Metrics.SEQUENCER_BLOCK_PROPOSAL_SUCCESS_COUNT, {
89
- valueType: ValueType.INT,
90
- description: 'The number of times block proposal succeeded (including validation builds)'
91
- });
92
- this.blockProposalPrecheckFailed = this.meter.createUpDownCounter(Metrics.SEQUENCER_BLOCK_PROPOSAL_PRECHECK_FAILED_COUNT, {
93
- valueType: ValueType.INT,
94
- description: 'The number of times block proposal pre-build checks failed'
95
- });
96
- this.slashingAttempts = this.meter.createUpDownCounter(Metrics.SEQUENCER_SLASHING_ATTEMPTS_COUNT, {
97
- valueType: ValueType.INT,
98
- description: 'The number of slashing action attempts'
103
+ this.fishermanTimeBeforeBlock = this.meter.createHistogram(Metrics.FISHERMAN_FEE_ANALYSIS_TIME_BEFORE_BLOCK);
104
+ this.fishermanPendingBlobTxCount = this.meter.createHistogram(Metrics.FISHERMAN_FEE_ANALYSIS_PENDING_BLOB_TX_COUNT);
105
+ this.fishermanIncludedBlobTxCount = this.meter.createHistogram(Metrics.FISHERMAN_FEE_ANALYSIS_INCLUDED_BLOB_TX_COUNT);
106
+ this.fishermanCalculatedPriorityFee = this.meter.createHistogram(Metrics.FISHERMAN_FEE_ANALYSIS_CALCULATED_PRIORITY_FEE);
107
+ this.fishermanPriorityFeeDelta = this.meter.createHistogram(Metrics.FISHERMAN_FEE_ANALYSIS_PRIORITY_FEE_DELTA);
108
+ this.fishermanEstimatedCost = this.meter.createHistogram(Metrics.FISHERMAN_FEE_ANALYSIS_ESTIMATED_COST);
109
+ this.fishermanEstimatedOverpayment = this.meter.createHistogram(Metrics.FISHERMAN_FEE_ANALYSIS_ESTIMATED_OVERPAYMENT);
110
+ this.fishermanMinedBlobTxPriorityFee = this.meter.createHistogram(Metrics.FISHERMAN_FEE_ANALYSIS_MINED_BLOB_TX_PRIORITY_FEE);
111
+ this.fishermanMinedBlobTxTotalCost = this.meter.createHistogram(Metrics.FISHERMAN_FEE_ANALYSIS_MINED_BLOB_TX_TOTAL_COST);
112
+ this.fishermanPendingBlobCount = this.meter.createHistogram(Metrics.FISHERMAN_FEE_ANALYSIS_PENDING_BLOB_COUNT);
113
+ this.fishermanIncludedBlobCount = this.meter.createHistogram(Metrics.FISHERMAN_FEE_ANALYSIS_INCLUDED_BLOB_COUNT);
114
+ this.fishermanBlockBlobsFull = createUpDownCounterWithDefault(this.meter, Metrics.FISHERMAN_FEE_ANALYSIS_BLOCK_BLOBS_FULL, {
115
+ [Attributes.OK]: [
116
+ true,
117
+ false
118
+ ]
99
119
  });
120
+ this.fishermanMaxBlobCapacity = this.meter.createHistogram(Metrics.FISHERMAN_FEE_ANALYSIS_MAX_BLOB_CAPACITY);
100
121
  }
101
122
  recordRequiredAttestations(requiredAttestationsCount, allowanceMs) {
102
123
  this.requiredAttestions.record(requiredAttestationsCount);
@@ -105,6 +126,9 @@ export class SequencerMetrics {
105
126
  this.collectedAttestions.record(0);
106
127
  this.timeToCollectAttestations.record(0);
107
128
  }
129
+ recordCheckpointAttestationDelay(duration) {
130
+ this.checkpointAttestationDelay.record(duration);
131
+ }
108
132
  recordCollectedAttestations(count, durationMs) {
109
133
  this.collectedAttestions.record(count);
110
134
  this.timeToCollectAttestations.record(Math.ceil(durationMs));
@@ -126,6 +150,12 @@ export class SequencerMetrics {
126
150
  [Attributes.SEQUENCER_STATE]: state
127
151
  });
128
152
  }
153
+ recordPipelineDepth(depth) {
154
+ this.pipelineDepth.record(depth);
155
+ }
156
+ recordPipelineDiscard(count = 1) {
157
+ this.pipelineDiscards.add(count);
158
+ }
129
159
  incOpenSlot(slot, proposer) {
130
160
  // sequencer went through the loop a second time. Noop
131
161
  if (slot === this.lastSeenSlot) {
@@ -153,6 +183,9 @@ export class SequencerMetrics {
153
183
  }
154
184
  }
155
185
  }
186
+ recordCheckpointSuccess() {
187
+ this.checkpointSuccess.add(1);
188
+ }
156
189
  recordBlockProposalFailed(reason) {
157
190
  this.blockProposalFailed.add(1, {
158
191
  ...reason && {
@@ -160,15 +193,137 @@ export class SequencerMetrics {
160
193
  }
161
194
  });
162
195
  }
163
- recordBlockProposalSuccess() {
164
- this.blockProposalSuccess.add(1);
196
+ recordCheckpointProposalSuccess() {
197
+ this.checkpointProposalSuccess.add(1);
165
198
  }
166
- recordBlockProposalPrecheckFailed(checkType) {
167
- this.blockProposalPrecheckFailed.add(1, {
199
+ recordCheckpointPrecheckFailed(checkType) {
200
+ this.checkpointPrecheckFailed.add(1, {
168
201
  [Attributes.ERROR_TYPE]: checkType
169
202
  });
170
203
  }
204
+ recordCheckpointProposalFailed(reason) {
205
+ this.checkpointProposalFailed.add(1, {
206
+ ...reason && {
207
+ [Attributes.ERROR_TYPE]: reason
208
+ }
209
+ });
210
+ }
211
+ /** Records aggregate metrics for a completed checkpoint build. */ recordCheckpointBuild(durationMs, blockCount, txCount, totalMana) {
212
+ this.checkpointBuildDuration.record(Math.ceil(durationMs));
213
+ this.checkpointBlockCount.record(blockCount);
214
+ this.checkpointTxCount.record(txCount);
215
+ this.checkpointTotalMana.record(totalMana);
216
+ }
171
217
  recordSlashingAttempt(actionCount) {
172
218
  this.slashingAttempts.add(actionCount);
173
219
  }
220
+ /**
221
+ * Records metrics for a completed fisherman fee analysis
222
+ * @param analysis - The completed fee analysis result
223
+ */ recordFishermanFeeAnalysis(analysis) {
224
+ // In fisherman mode, we should always have strategy results
225
+ if (!analysis.computedPrices.strategyResults || analysis.computedPrices.strategyResults.length === 0) {
226
+ // This should never happen in fisherman mode - log an error
227
+ // We don't record metrics without strategy IDs as that defeats the purpose
228
+ throw new Error(`No strategy results found in fisherman fee analysis ${analysis.id}. This indicates a bug in the fee analysis.`);
229
+ }
230
+ // Record metrics for each strategy separately
231
+ for (const strategyResult of analysis.computedPrices.strategyResults){
232
+ const strategyAttributes = {
233
+ [Attributes.FISHERMAN_FEE_STRATEGY_ID]: strategyResult.strategyId
234
+ };
235
+ // Record pending block snapshot data (once per strategy for comparison)
236
+ this.fishermanPendingBlobTxCount.record(analysis.pendingSnapshot.pendingBlobTxCount, strategyAttributes);
237
+ this.fishermanPendingBlobCount.record(analysis.pendingSnapshot.pendingBlobCount, strategyAttributes);
238
+ // Record mined block data if available
239
+ if (analysis.minedBlock) {
240
+ this.fishermanIncludedBlobTxCount.record(analysis.minedBlock.includedBlobTxCount, strategyAttributes);
241
+ this.fishermanIncludedBlobCount.record(analysis.minedBlock.includedBlobCount, strategyAttributes);
242
+ // Record actual fees from blob transactions in the mined block
243
+ for (const blobTx of analysis.minedBlock.includedBlobTxs){
244
+ // Record priority fee per gas in Gwei
245
+ const priorityFeeGwei = Number(blobTx.maxPriorityFeePerGas) / 1e9;
246
+ this.fishermanMinedBlobTxPriorityFee.record(priorityFeeGwei, strategyAttributes);
247
+ // Calculate total cost in ETH
248
+ // Cost = (gas * (baseFee + priorityFee)) + (blobCount * blobGasPerBlob * blobBaseFee)
249
+ const baseFee = analysis.minedBlock.baseFeePerGas;
250
+ const effectiveGasPrice = baseFee + blobTx.maxPriorityFeePerGas;
251
+ // Calculate execution cost using actual gas limit from the transaction
252
+ const executionCost = blobTx.gas * effectiveGasPrice;
253
+ // Calculate blob cost using maxFeePerBlobGas * blobCount * GAS_PER_BLOB
254
+ const blobCost = blobTx.maxFeePerBlobGas * BigInt(blobTx.blobCount) * 131072n; // 128KB per blob
255
+ const totalCostWei = executionCost + blobCost;
256
+ const totalCostEth = Number(totalCostWei) / 1e18;
257
+ this.fishermanMinedBlobTxTotalCost.record(totalCostEth, strategyAttributes);
258
+ }
259
+ }
260
+ // Record the calculated priority fee for this strategy
261
+ const calculatedPriorityFeeGwei = Number(strategyResult.calculatedPriorityFee) / 1e9;
262
+ this.fishermanCalculatedPriorityFee.record(calculatedPriorityFeeGwei, strategyAttributes);
263
+ // Record analysis results if available
264
+ if (analysis.analysis) {
265
+ this.fishermanTimeBeforeBlock.record(Math.ceil(analysis.analysis.timeBeforeBlockMs), strategyAttributes);
266
+ // Record whether the block reached 100% blob capacity
267
+ if (analysis.analysis.blockBlobsFull) {
268
+ this.fishermanBlockBlobsFull.add(1, {
269
+ ...strategyAttributes,
270
+ [Attributes.OK]: true
271
+ });
272
+ } else {
273
+ this.fishermanBlockBlobsFull.add(1, {
274
+ ...strategyAttributes,
275
+ [Attributes.OK]: false
276
+ });
277
+ }
278
+ // Record the max blob capacity for this block
279
+ this.fishermanMaxBlobCapacity.record(analysis.analysis.maxBlobCapacity, strategyAttributes);
280
+ // Record strategy-specific inclusion result
281
+ if (strategyResult.wouldBeIncluded !== undefined) {
282
+ const inclusionAttributes = {
283
+ ...strategyAttributes,
284
+ [Attributes.BLOCK_FULL]: analysis.analysis.blockBlobsFull ? 'true' : 'false'
285
+ };
286
+ if (strategyResult.wouldBeIncluded) {
287
+ this.fishermanWouldBeIncluded.add(1, {
288
+ ...inclusionAttributes,
289
+ [Attributes.OK]: true
290
+ });
291
+ } else {
292
+ this.fishermanWouldBeIncluded.add(1, {
293
+ ...inclusionAttributes,
294
+ [Attributes.OK]: false,
295
+ ...strategyResult.exclusionReason && {
296
+ [Attributes.ERROR_TYPE]: strategyResult.exclusionReason
297
+ }
298
+ });
299
+ }
300
+ }
301
+ // Record strategy-specific priority fee delta
302
+ if (strategyResult.priorityFeeDelta !== undefined) {
303
+ const priorityFeeDeltaGwei = Number(strategyResult.priorityFeeDelta) / 1e9;
304
+ const deltaAttributes = {
305
+ ...strategyAttributes,
306
+ [Attributes.BLOCK_FULL]: analysis.analysis.blockBlobsFull ? 'true' : 'false'
307
+ };
308
+ this.fishermanPriorityFeeDelta.record(priorityFeeDeltaGwei, deltaAttributes);
309
+ }
310
+ // Record estimated cost if available
311
+ if (strategyResult.estimatedCostEth !== undefined) {
312
+ const costAttributes = {
313
+ ...strategyAttributes,
314
+ [Attributes.BLOCK_FULL]: analysis.analysis.blockBlobsFull ? 'true' : 'false'
315
+ };
316
+ this.fishermanEstimatedCost.record(strategyResult.estimatedCostEth, costAttributes);
317
+ }
318
+ // Record estimated overpayment if available
319
+ if (strategyResult.estimatedOverpaymentEth !== undefined) {
320
+ const overpaymentAttributes = {
321
+ ...strategyAttributes,
322
+ [Attributes.BLOCK_FULL]: analysis.analysis.blockBlobsFull ? 'true' : 'false'
323
+ };
324
+ this.fishermanEstimatedOverpayment.record(strategyResult.estimatedOverpaymentEth, overpaymentAttributes);
325
+ }
326
+ }
327
+ }
328
+ }
174
329
  }