@aztec/ethereum 0.0.1-commit.6d3c34e → 0.0.1-commit.7ac86ea28

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 (138) hide show
  1. package/dest/config.d.ts +15 -28
  2. package/dest/config.d.ts.map +1 -1
  3. package/dest/config.js +50 -57
  4. package/dest/contracts/empire_base.d.ts +3 -1
  5. package/dest/contracts/empire_base.d.ts.map +1 -1
  6. package/dest/contracts/empire_slashing_proposer.d.ts +3 -1
  7. package/dest/contracts/empire_slashing_proposer.d.ts.map +1 -1
  8. package/dest/contracts/empire_slashing_proposer.js +22 -15
  9. package/dest/contracts/fee_asset_handler.d.ts +1 -1
  10. package/dest/contracts/fee_asset_handler.d.ts.map +1 -1
  11. package/dest/contracts/fee_asset_handler.js +2 -0
  12. package/dest/contracts/fee_asset_price_oracle.d.ts +101 -0
  13. package/dest/contracts/fee_asset_price_oracle.d.ts.map +1 -0
  14. package/dest/contracts/fee_asset_price_oracle.js +651 -0
  15. package/dest/contracts/governance.d.ts +3 -1
  16. package/dest/contracts/governance.d.ts.map +1 -1
  17. package/dest/contracts/governance.js +14 -4
  18. package/dest/contracts/governance_proposer.d.ts +3 -1
  19. package/dest/contracts/governance_proposer.d.ts.map +1 -1
  20. package/dest/contracts/governance_proposer.js +13 -1
  21. package/dest/contracts/inbox.d.ts +18 -1
  22. package/dest/contracts/inbox.d.ts.map +1 -1
  23. package/dest/contracts/inbox.js +32 -1
  24. package/dest/contracts/index.d.ts +3 -1
  25. package/dest/contracts/index.d.ts.map +1 -1
  26. package/dest/contracts/index.js +2 -0
  27. package/dest/contracts/log.d.ts +13 -0
  28. package/dest/contracts/log.d.ts.map +1 -0
  29. package/dest/contracts/log.js +1 -0
  30. package/dest/contracts/multicall.d.ts +1 -1
  31. package/dest/contracts/multicall.d.ts.map +1 -1
  32. package/dest/contracts/multicall.js +2 -1
  33. package/dest/contracts/rollup.d.ts +35 -3
  34. package/dest/contracts/rollup.d.ts.map +1 -1
  35. package/dest/contracts/rollup.js +76 -6
  36. package/dest/contracts/tally_slashing_proposer.d.ts +1 -1
  37. package/dest/contracts/tally_slashing_proposer.d.ts.map +1 -1
  38. package/dest/contracts/tally_slashing_proposer.js +8 -1
  39. package/dest/deploy_aztec_l1_contracts.d.ts +7 -3
  40. package/dest/deploy_aztec_l1_contracts.d.ts.map +1 -1
  41. package/dest/deploy_aztec_l1_contracts.js +56 -28
  42. package/dest/deploy_l1_contract.js +3 -3
  43. package/dest/generated/l1-contracts-defaults.d.ts +30 -0
  44. package/dest/generated/l1-contracts-defaults.d.ts.map +1 -0
  45. package/dest/generated/l1-contracts-defaults.js +30 -0
  46. package/dest/l1_artifacts.d.ts +2405 -473
  47. package/dest/l1_artifacts.d.ts.map +1 -1
  48. package/dest/l1_tx_utils/config.d.ts +7 -1
  49. package/dest/l1_tx_utils/config.d.ts.map +1 -1
  50. package/dest/l1_tx_utils/config.js +14 -1
  51. package/dest/l1_tx_utils/constants.d.ts +1 -1
  52. package/dest/l1_tx_utils/constants.js +2 -2
  53. package/dest/l1_tx_utils/factory.d.ts +18 -10
  54. package/dest/l1_tx_utils/factory.d.ts.map +1 -1
  55. package/dest/l1_tx_utils/factory.js +17 -7
  56. package/dest/l1_tx_utils/fee-strategies/p75_competitive.js +1 -1
  57. package/dest/l1_tx_utils/fee-strategies/p75_competitive_blob_txs_only.js +1 -1
  58. package/dest/l1_tx_utils/forwarder_l1_tx_utils.d.ts +15 -15
  59. package/dest/l1_tx_utils/forwarder_l1_tx_utils.d.ts.map +1 -1
  60. package/dest/l1_tx_utils/forwarder_l1_tx_utils.js +9 -15
  61. package/dest/l1_tx_utils/index-blobs.d.ts +3 -3
  62. package/dest/l1_tx_utils/index-blobs.d.ts.map +1 -1
  63. package/dest/l1_tx_utils/index-blobs.js +2 -2
  64. package/dest/l1_tx_utils/index.d.ts +2 -1
  65. package/dest/l1_tx_utils/index.d.ts.map +1 -1
  66. package/dest/l1_tx_utils/index.js +1 -0
  67. package/dest/l1_tx_utils/l1_tx_utils.d.ts +15 -5
  68. package/dest/l1_tx_utils/l1_tx_utils.d.ts.map +1 -1
  69. package/dest/l1_tx_utils/l1_tx_utils.js +47 -13
  70. package/dest/l1_tx_utils/readonly_l1_tx_utils.js +3 -3
  71. package/dest/l1_tx_utils/tx_delayer.d.ts +56 -0
  72. package/dest/l1_tx_utils/tx_delayer.d.ts.map +1 -0
  73. package/dest/{test → l1_tx_utils}/tx_delayer.js +62 -34
  74. package/dest/publisher_manager.d.ts +3 -2
  75. package/dest/publisher_manager.d.ts.map +1 -1
  76. package/dest/publisher_manager.js +2 -2
  77. package/dest/queries.d.ts +2 -2
  78. package/dest/queries.d.ts.map +1 -1
  79. package/dest/queries.js +4 -1
  80. package/dest/test/eth_cheat_codes.d.ts +13 -1
  81. package/dest/test/eth_cheat_codes.d.ts.map +1 -1
  82. package/dest/test/index.d.ts +1 -3
  83. package/dest/test/index.d.ts.map +1 -1
  84. package/dest/test/index.js +0 -2
  85. package/dest/test/rollup_cheat_codes.d.ts +4 -2
  86. package/dest/test/rollup_cheat_codes.d.ts.map +1 -1
  87. package/dest/test/rollup_cheat_codes.js +10 -1
  88. package/dest/test/start_anvil.js +1 -1
  89. package/dest/test/upgrade_utils.js +2 -2
  90. package/dest/utils.d.ts +2 -1
  91. package/dest/utils.d.ts.map +1 -1
  92. package/dest/utils.js +46 -0
  93. package/package.json +8 -7
  94. package/src/config.ts +58 -56
  95. package/src/contracts/README.md +157 -0
  96. package/src/contracts/empire_base.ts +2 -0
  97. package/src/contracts/empire_slashing_proposer.ts +22 -27
  98. package/src/contracts/fee_asset_handler.ts +2 -0
  99. package/src/contracts/fee_asset_price_oracle.ts +280 -0
  100. package/src/contracts/governance.ts +13 -4
  101. package/src/contracts/governance_proposer.ts +10 -1
  102. package/src/contracts/inbox.ts +48 -1
  103. package/src/contracts/index.ts +2 -0
  104. package/src/contracts/log.ts +13 -0
  105. package/src/contracts/multicall.ts +5 -2
  106. package/src/contracts/rollup.ts +111 -11
  107. package/src/contracts/tally_slashing_proposer.ts +5 -1
  108. package/src/deploy_aztec_l1_contracts.ts +80 -35
  109. package/src/deploy_l1_contract.ts +3 -3
  110. package/src/generated/l1-contracts-defaults.ts +32 -0
  111. package/src/l1_tx_utils/config.ts +20 -0
  112. package/src/l1_tx_utils/constants.ts +2 -2
  113. package/src/l1_tx_utils/factory.ts +31 -31
  114. package/src/l1_tx_utils/fee-strategies/p75_competitive.ts +1 -1
  115. package/src/l1_tx_utils/fee-strategies/p75_competitive_blob_txs_only.ts +1 -1
  116. package/src/l1_tx_utils/forwarder_l1_tx_utils.ts +43 -54
  117. package/src/l1_tx_utils/index-blobs.ts +2 -2
  118. package/src/l1_tx_utils/index.ts +1 -0
  119. package/src/l1_tx_utils/l1_tx_utils.ts +52 -17
  120. package/src/l1_tx_utils/readonly_l1_tx_utils.ts +3 -3
  121. package/src/{test → l1_tx_utils}/tx_delayer.ts +78 -50
  122. package/src/publisher_manager.ts +4 -2
  123. package/src/queries.ts +3 -1
  124. package/src/test/index.ts +0 -2
  125. package/src/test/rollup_cheat_codes.ts +11 -2
  126. package/src/test/start_anvil.ts +1 -1
  127. package/src/test/upgrade_utils.ts +2 -2
  128. package/src/utils.ts +53 -0
  129. package/dest/l1_tx_utils/l1_tx_utils_with_blobs.d.ts +0 -26
  130. package/dest/l1_tx_utils/l1_tx_utils_with_blobs.d.ts.map +0 -1
  131. package/dest/l1_tx_utils/l1_tx_utils_with_blobs.js +0 -26
  132. package/dest/test/delayed_tx_utils.d.ts +0 -13
  133. package/dest/test/delayed_tx_utils.d.ts.map +0 -1
  134. package/dest/test/delayed_tx_utils.js +0 -28
  135. package/dest/test/tx_delayer.d.ts +0 -36
  136. package/dest/test/tx_delayer.d.ts.map +0 -1
  137. package/src/l1_tx_utils/l1_tx_utils_with_blobs.ts +0 -77
  138. package/src/test/delayed_tx_utils.ts +0 -52
@@ -1,64 +1,64 @@
1
+ import type { BlobKzgInstance } from '@aztec/blob-lib/types';
1
2
  import { EthAddress } from '@aztec/foundation/eth-address';
2
3
  import type { Logger } from '@aztec/foundation/log';
3
4
  import { DateProvider } from '@aztec/foundation/timer';
4
5
 
5
- import type { TransactionSerializable } from 'viem';
6
-
7
6
  import type { EthSigner } from '../eth-signer/eth-signer.js';
8
7
  import type { ExtendedViemWalletClient, ViemClient } from '../types.js';
9
8
  import type { L1TxUtilsConfig } from './config.js';
10
9
  import type { IL1TxMetrics, IL1TxStore } from './interfaces.js';
11
10
  import { L1TxUtils } from './l1_tx_utils.js';
12
11
  import { createViemSigner } from './signer.js';
12
+ import { Delayer } from './tx_delayer.js';
13
13
  import type { SigningCallback } from './types.js';
14
14
 
15
- export function createL1TxUtilsFromViemWallet(
16
- client: ExtendedViemWalletClient,
17
- deps?: {
18
- logger?: Logger;
19
- dateProvider?: DateProvider;
20
- store?: IL1TxStore;
21
- metrics?: IL1TxMetrics;
22
- },
23
- config?: Partial<L1TxUtilsConfig> & { debugMaxGasLimit?: boolean },
24
- ): L1TxUtils {
25
- return new L1TxUtils(
15
+ /** Source of signing capability: either a wallet client or a separate client + signer. */
16
+ export type L1SignerSource = ExtendedViemWalletClient | { client: ViemClient; signer: EthSigner };
17
+
18
+ export function resolveSignerSource(source: L1SignerSource): {
19
+ client: ViemClient;
20
+ address: EthAddress;
21
+ signingCallback: SigningCallback;
22
+ } {
23
+ if ('account' in source && source.account) {
24
+ return {
25
+ client: source as ExtendedViemWalletClient,
26
+ address: EthAddress.fromString((source as ExtendedViemWalletClient).account.address),
27
+ signingCallback: createViemSigner(source as ExtendedViemWalletClient),
28
+ };
29
+ }
30
+ const { client, signer } = source as { client: ViemClient; signer: EthSigner };
31
+ return {
26
32
  client,
27
- EthAddress.fromString(client.account.address),
28
- createViemSigner(client),
29
- deps?.logger,
30
- deps?.dateProvider,
31
- config,
32
- config?.debugMaxGasLimit ?? false,
33
- deps?.store,
34
- deps?.metrics,
35
- );
33
+ address: signer.address,
34
+ signingCallback: async (tx, _addr) => (await signer.signTransaction(tx)).toViemTransactionSignature(),
35
+ };
36
36
  }
37
37
 
38
- export function createL1TxUtilsFromEthSigner(
39
- client: ViemClient,
40
- signer: EthSigner,
38
+ export function createL1TxUtils(
39
+ source: L1SignerSource,
41
40
  deps?: {
42
41
  logger?: Logger;
43
42
  dateProvider?: DateProvider;
44
43
  store?: IL1TxStore;
45
44
  metrics?: IL1TxMetrics;
45
+ kzg?: BlobKzgInstance;
46
+ delayer?: Delayer;
46
47
  },
47
48
  config?: Partial<L1TxUtilsConfig> & { debugMaxGasLimit?: boolean },
48
49
  ): L1TxUtils {
49
- const callback: SigningCallback = async (transaction: TransactionSerializable, _signingAddress) => {
50
- return (await signer.signTransaction(transaction)).toViemTransactionSignature();
51
- };
52
-
50
+ const { client, address, signingCallback } = resolveSignerSource(source);
53
51
  return new L1TxUtils(
54
52
  client,
55
- signer.address,
56
- callback,
53
+ address,
54
+ signingCallback,
57
55
  deps?.logger,
58
56
  deps?.dateProvider,
59
57
  config,
60
58
  config?.debugMaxGasLimit ?? false,
61
59
  deps?.store,
62
60
  deps?.metrics,
61
+ deps?.kzg,
62
+ deps?.delayer,
63
63
  );
64
64
  }
@@ -134,7 +134,7 @@ export const P75AllTxsPriorityFeeStrategy: PriorityFeeStrategy = {
134
134
  // Sanity check: cap competitive fee at 100x network estimate to avoid using unrealistic fees
135
135
  const maxReasonableFee = networkEstimate * 100n;
136
136
  if (competitiveFee > maxReasonableFee && networkEstimate > 0n) {
137
- logger?.warn('Competitive fee exceeds sanity cap, using capped value', {
137
+ logger?.debug('Competitive fee exceeds sanity cap, using capped value', {
138
138
  competitiveFee: formatGwei(competitiveFee),
139
139
  networkEstimate: formatGwei(networkEstimate),
140
140
  cappedTo: formatGwei(maxReasonableFee),
@@ -207,7 +207,7 @@ export const P75BlobTxsOnlyPriorityFeeStrategy: PriorityFeeStrategy = {
207
207
 
208
208
  // Debug: Log suspicious fees from history
209
209
  if (medianHistoricalFee > 100n * WEI_CONST) {
210
- logger?.warn('Suspicious high fee in history', {
210
+ logger?.debug('Suspicious high fee in history', {
211
211
  historicalMedian: formatGwei(medianHistoricalFee),
212
212
  allP75Fees: percentile75Fees.map(f => formatGwei(f)),
213
213
  });
@@ -1,25 +1,27 @@
1
+ import type { BlobKzgInstance } from '@aztec/blob-lib/types';
1
2
  import { EthAddress } from '@aztec/foundation/eth-address';
2
3
  import type { Logger } from '@aztec/foundation/log';
3
4
  import type { DateProvider } from '@aztec/foundation/timer';
4
5
 
5
6
  import { type Hex, encodeFunctionData } from 'viem';
6
7
 
7
- import type { EthSigner } from '../eth-signer/eth-signer.js';
8
8
  import { FORWARDER_ABI } from '../forwarder_proxy.js';
9
- import type { ExtendedViemWalletClient, ViemClient } from '../types.js';
9
+ import type { ViemClient } from '../types.js';
10
10
  import type { L1TxUtilsConfig } from './config.js';
11
+ import type { L1SignerSource } from './factory.js';
12
+ import { resolveSignerSource } from './factory.js';
11
13
  import type { IL1TxMetrics, IL1TxStore } from './interfaces.js';
12
- import { L1TxUtilsWithBlobs } from './l1_tx_utils_with_blobs.js';
13
- import { createViemSigner } from './signer.js';
14
+ import { L1TxUtils } from './l1_tx_utils.js';
15
+ import { Delayer } from './tx_delayer.js';
14
16
  import type { L1BlobInputs, L1TxConfig, L1TxRequest, SigningCallback } from './types.js';
15
17
 
16
18
  /**
17
- * Extends L1TxUtilsWithBlobs to wrap all transactions through a forwarder contract.
19
+ * Extends L1TxUtils to wrap all transactions through a forwarder contract.
18
20
  * This is mainly used for testing the archiver's ability to decode transactions that go through proxies.
19
21
  */
20
- export class ForwarderL1TxUtils extends L1TxUtilsWithBlobs {
22
+ export class ForwarderL1TxUtils extends L1TxUtils {
21
23
  constructor(
22
- client: ViemClient | ExtendedViemWalletClient,
24
+ client: ViemClient,
23
25
  senderAddress: EthAddress,
24
26
  signingCallback: SigningCallback,
25
27
  logger: Logger | undefined,
@@ -28,9 +30,23 @@ export class ForwarderL1TxUtils extends L1TxUtilsWithBlobs {
28
30
  debugMaxGasLimit: boolean,
29
31
  store: IL1TxStore | undefined,
30
32
  metrics: IL1TxMetrics | undefined,
33
+ kzg: BlobKzgInstance | undefined,
34
+ delayer: Delayer | undefined,
31
35
  private readonly forwarderAddress: EthAddress,
32
36
  ) {
33
- super(client, senderAddress, signingCallback, logger, dateProvider, config, debugMaxGasLimit, store, metrics);
37
+ super(
38
+ client,
39
+ senderAddress,
40
+ signingCallback,
41
+ logger,
42
+ dateProvider,
43
+ config,
44
+ debugMaxGasLimit,
45
+ store,
46
+ metrics,
47
+ kzg,
48
+ delayer,
49
+ );
34
50
  }
35
51
 
36
52
  /**
@@ -61,59 +77,32 @@ export class ForwarderL1TxUtils extends L1TxUtilsWithBlobs {
61
77
  }
62
78
  }
63
79
 
64
- export function createForwarderL1TxUtilsFromViemWallet(
65
- client: ExtendedViemWalletClient,
80
+ export function createForwarderL1TxUtils(
81
+ source: L1SignerSource,
66
82
  forwarderAddress: EthAddress,
67
- deps: {
83
+ deps?: {
68
84
  logger?: Logger;
69
85
  dateProvider?: DateProvider;
70
86
  store?: IL1TxStore;
71
87
  metrics?: IL1TxMetrics;
72
- } = {},
73
- config: Partial<L1TxUtilsConfig> = {},
74
- debugMaxGasLimit: boolean = false,
75
- ) {
88
+ kzg?: BlobKzgInstance;
89
+ delayer?: Delayer;
90
+ },
91
+ config?: Partial<L1TxUtilsConfig> & { debugMaxGasLimit?: boolean },
92
+ ): ForwarderL1TxUtils {
93
+ const { client, address, signingCallback } = resolveSignerSource(source);
76
94
  return new ForwarderL1TxUtils(
77
95
  client,
78
- EthAddress.fromString(client.account.address),
79
- createViemSigner(client),
80
- deps.logger,
81
- deps.dateProvider,
82
- config,
83
- debugMaxGasLimit,
84
- deps.store,
85
- deps.metrics,
86
- forwarderAddress,
87
- );
88
- }
89
-
90
- export function createForwarderL1TxUtilsFromEthSigner(
91
- client: ViemClient,
92
- signer: EthSigner,
93
- forwarderAddress: EthAddress,
94
- deps: {
95
- logger?: Logger;
96
- dateProvider?: DateProvider;
97
- store?: IL1TxStore;
98
- metrics?: IL1TxMetrics;
99
- } = {},
100
- config: Partial<L1TxUtilsConfig> = {},
101
- debugMaxGasLimit: boolean = false,
102
- ) {
103
- const callback: SigningCallback = async (transaction, _signingAddress) => {
104
- return (await signer.signTransaction(transaction)).toViemTransactionSignature();
105
- };
106
-
107
- return new ForwarderL1TxUtils(
108
- client,
109
- signer.address,
110
- callback,
111
- deps.logger,
112
- deps.dateProvider,
113
- config,
114
- debugMaxGasLimit,
115
- deps.store,
116
- deps.metrics,
96
+ address,
97
+ signingCallback,
98
+ deps?.logger,
99
+ deps?.dateProvider,
100
+ config ?? {},
101
+ config?.debugMaxGasLimit ?? false,
102
+ deps?.store,
103
+ deps?.metrics,
104
+ deps?.kzg,
105
+ deps?.delayer,
117
106
  forwarderAddress,
118
107
  );
119
108
  }
@@ -1,2 +1,2 @@
1
- export * from './forwarder_l1_tx_utils.js';
2
- export * from './l1_tx_utils_with_blobs.js';
1
+ export { createForwarderL1TxUtils, ForwarderL1TxUtils } from './forwarder_l1_tx_utils.js';
2
+ export { createL1TxUtils, type L1SignerSource, resolveSignerSource } from './factory.js';
@@ -8,6 +8,7 @@ export * from './l1_tx_utils.js';
8
8
  export * from './readonly_l1_tx_utils.js';
9
9
  export * from './signer.js';
10
10
  export * from './types.js';
11
+ export * from './tx_delayer.js';
11
12
  export * from './utils.js';
12
13
 
13
14
  // Note: We intentionally do not export l1_tx_utils_with_blobs.js
@@ -1,8 +1,9 @@
1
+ import type { BlobKzgInstance } from '@aztec/blob-lib/types';
1
2
  import { maxBigint } from '@aztec/foundation/bigint';
2
3
  import { merge, pick } from '@aztec/foundation/collection';
3
4
  import { InterruptError, TimeoutError } from '@aztec/foundation/error';
4
5
  import { EthAddress } from '@aztec/foundation/eth-address';
5
- import { type Logger, createLogger } from '@aztec/foundation/log';
6
+ import { type Logger, type LoggerBindings, createLogger } from '@aztec/foundation/log';
6
7
  import { retryUntil } from '@aztec/foundation/retry';
7
8
  import { sleep } from '@aztec/foundation/sleep';
8
9
  import { DateProvider } from '@aztec/foundation/timer';
@@ -27,9 +28,10 @@ import { jsonRpc } from 'viem/nonce';
27
28
  import type { ViemClient } from '../types.js';
28
29
  import { formatViemError } from '../utils.js';
29
30
  import { type L1TxUtilsConfig, l1TxUtilsConfigMappings } from './config.js';
30
- import { LARGE_GAS_LIMIT } from './constants.js';
31
+ import { MAX_L1_TX_LIMIT } from './constants.js';
31
32
  import type { IL1TxMetrics, IL1TxStore } from './interfaces.js';
32
33
  import { ReadOnlyL1TxUtils } from './readonly_l1_tx_utils.js';
34
+ import { Delayer, createDelayer, wrapClientWithDelayer } from './tx_delayer.js';
33
35
  import {
34
36
  DroppedTransactionError,
35
37
  type L1BlobInputs,
@@ -47,6 +49,10 @@ const MAX_L1_TX_STATES = 32;
47
49
  export class L1TxUtils extends ReadOnlyL1TxUtils {
48
50
  protected nonceManager: NonceManager;
49
51
  protected txs: L1TxState[] = [];
52
+ /** Tx delayer for testing. Only set when enableDelayer config is true. */
53
+ public delayer?: Delayer;
54
+ /** KZG instance for blob operations. */
55
+ protected kzg?: BlobKzgInstance;
50
56
 
51
57
  constructor(
52
58
  public override client: ViemClient,
@@ -58,9 +64,26 @@ export class L1TxUtils extends ReadOnlyL1TxUtils {
58
64
  debugMaxGasLimit: boolean = false,
59
65
  protected store?: IL1TxStore,
60
66
  protected metrics?: IL1TxMetrics,
67
+ kzg?: BlobKzgInstance,
68
+ delayer?: Delayer,
61
69
  ) {
62
70
  super(client, logger, dateProvider, config, debugMaxGasLimit);
63
71
  this.nonceManager = createNonceManager({ source: jsonRpc() });
72
+ this.kzg = kzg;
73
+
74
+ // Set up delayer: use provided one or create new
75
+ if (config?.enableDelayer && config?.ethereumSlotDuration) {
76
+ this.delayer =
77
+ delayer ?? this.createDelayer({ ethereumSlotDuration: config.ethereumSlotDuration }, logger.getBindings());
78
+ this.client = wrapClientWithDelayer(this.client, this.delayer);
79
+ if (config.txDelayerMaxInclusionTimeIntoSlot !== undefined) {
80
+ this.delayer.setMaxInclusionTimeIntoSlot(config.txDelayerMaxInclusionTimeIntoSlot);
81
+ }
82
+ } else if (delayer) {
83
+ // Delayer provided but enableDelayer not set — just store it without wrapping
84
+ logger.warn('Delayer provided but enableDelayer config is not set; delayer will not be used');
85
+ this.delayer = delayer;
86
+ }
64
87
  }
65
88
 
66
89
  public get state() {
@@ -207,7 +230,7 @@ export class L1TxUtils extends ReadOnlyL1TxUtils {
207
230
 
208
231
  let gasLimit: bigint;
209
232
  if (this.debugMaxGasLimit) {
210
- gasLimit = LARGE_GAS_LIMIT;
233
+ gasLimit = MAX_L1_TX_LIMIT;
211
234
  } else if (gasConfig.gasLimit) {
212
235
  gasLimit = gasConfig.gasLimit;
213
236
  } else {
@@ -221,6 +244,16 @@ export class L1TxUtils extends ReadOnlyL1TxUtils {
221
244
  throw new InterruptError(`Transaction sending is interrupted`);
222
245
  }
223
246
 
247
+ // Check timeout before consuming nonce to avoid leaking a nonce that was never sent.
248
+ // A leaked nonce creates a gap (e.g. nonce 107 consumed but unsent), so all subsequent
249
+ // transactions (108, 109, ...) can never be mined since the chain expects 107 first.
250
+ const now = new Date(await this.getL1Timestamp());
251
+ if (gasConfig.txTimeoutAt && now > gasConfig.txTimeoutAt) {
252
+ throw new TimeoutError(
253
+ `Transaction timed out before sending (now ${now.toISOString()} > timeoutAt ${gasConfig.txTimeoutAt.toISOString()})`,
254
+ );
255
+ }
256
+
224
257
  const nonce = await this.nonceManager.consume({
225
258
  client: this.client,
226
259
  address: account,
@@ -230,13 +263,6 @@ export class L1TxUtils extends ReadOnlyL1TxUtils {
230
263
  const baseState = { request, gasLimit, blobInputs, gasPrice, nonce };
231
264
  const txData = this.makeTxData(baseState, { isCancelTx: false });
232
265
 
233
- const now = new Date(await this.getL1Timestamp());
234
- if (gasConfig.txTimeoutAt && now > gasConfig.txTimeoutAt) {
235
- throw new TimeoutError(
236
- `Transaction timed out before sending (now ${now.toISOString()} > timeoutAt ${gasConfig.txTimeoutAt.toISOString()})`,
237
- );
238
- }
239
-
240
266
  // Send the new tx
241
267
  const signedRequest = await this.prepareSignedTransaction(txData);
242
268
  const txHash = await this.client.sendRawTransaction({ serializedTransaction: signedRequest });
@@ -283,7 +309,7 @@ export class L1TxUtils extends ReadOnlyL1TxUtils {
283
309
  return { txHash, state: l1TxState };
284
310
  } catch (err: any) {
285
311
  const viemError = formatViemError(err, request.abi);
286
- this.logger.error(`Failed to send L1 transaction`, viemError, {
312
+ this.logger.error(`Failed to send L1 transaction: ${viemError.message}`, viemError, {
287
313
  request: pick(request, 'to', 'value'),
288
314
  });
289
315
  throw viemError;
@@ -631,12 +657,12 @@ export class L1TxUtils extends ReadOnlyL1TxUtils {
631
657
  from: request.from ?? this.getSenderAddress().toString(),
632
658
  maxFeePerGas: gasPrice.maxFeePerGas,
633
659
  maxPriorityFeePerGas: gasPrice.maxPriorityFeePerGas,
634
- gas: request.gas ?? LARGE_GAS_LIMIT,
660
+ gas: request.gas ?? MAX_L1_TX_LIMIT,
635
661
  };
636
662
 
637
663
  if (!request.gas && !gasConfig.ignoreBlockGasLimit) {
638
- // LARGE_GAS_LIMIT is set as call.gas, increase block gasLimit
639
- blockOverrides.gasLimit = LARGE_GAS_LIMIT * 2n;
664
+ // MAX_L1_TX_LIMIT is set as call.gas, ensure block gasLimit is sufficient
665
+ blockOverrides.gasLimit = MAX_L1_TX_LIMIT;
640
666
  }
641
667
 
642
668
  return this._simulate(call, blockOverrides, stateOverrides, gasConfig, abi);
@@ -731,8 +757,17 @@ export class L1TxUtils extends ReadOnlyL1TxUtils {
731
757
  return Number(timestamp) * 1000;
732
758
  }
733
759
 
734
- /** Makes empty blob inputs for the cancellation tx. To be overridden in L1TxUtilsWithBlobs. */
735
- protected makeEmptyBlobInputs(_maxFeePerBlobGas: bigint): Required<L1BlobInputs> {
736
- throw new Error('Cannot make empty blob inputs for cancellation');
760
+ /** Makes empty blob inputs for the cancellation tx. */
761
+ protected makeEmptyBlobInputs(maxFeePerBlobGas: bigint): Required<L1BlobInputs> {
762
+ if (!this.kzg) {
763
+ throw new Error('Cannot make empty blob inputs for cancellation without kzg');
764
+ }
765
+ const blobData = new Uint8Array(131072).fill(0);
766
+ return { blobs: [blobData], kzg: this.kzg, maxFeePerBlobGas };
767
+ }
768
+
769
+ /** Creates a new delayer instance. */
770
+ protected createDelayer(opts: { ethereumSlotDuration: bigint | number }, bindings: LoggerBindings): Delayer {
771
+ return createDelayer(this.dateProvider, opts, bindings);
737
772
  }
738
773
  }
@@ -27,7 +27,7 @@ import type { ViemClient } from '../types.js';
27
27
  import { type L1TxUtilsConfig, defaultL1TxUtilsConfig, l1TxUtilsConfigMappings } from './config.js';
28
28
  import {
29
29
  BLOCK_TIME_MS,
30
- LARGE_GAS_LIMIT,
30
+ MAX_L1_TX_LIMIT,
31
31
  MIN_BLOB_REPLACEMENT_BUMP_PERCENTAGE,
32
32
  MIN_REPLACEMENT_BUMP_PERCENTAGE,
33
33
  WEI_CONST,
@@ -249,7 +249,7 @@ export class ReadOnlyL1TxUtils {
249
249
  ...request,
250
250
  ..._blobInputs,
251
251
  maxFeePerBlobGas: gasPrice.maxFeePerBlobGas!,
252
- gas: LARGE_GAS_LIMIT,
252
+ gas: MAX_L1_TX_LIMIT,
253
253
  blockTag: 'latest',
254
254
  });
255
255
 
@@ -258,7 +258,7 @@ export class ReadOnlyL1TxUtils {
258
258
  initialEstimate = await this.client.estimateGas({
259
259
  account,
260
260
  ...request,
261
- gas: LARGE_GAS_LIMIT,
261
+ gas: MAX_L1_TX_LIMIT,
262
262
  blockTag: 'latest',
263
263
  });
264
264
  this.logger?.trace(`Estimated gas for non-blob tx: ${initialEstimate}`);