@aztec/ethereum 0.0.1-commit.24de95ac → 0.0.1-commit.3469e52

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 (202) hide show
  1. package/dest/account.d.ts +1 -1
  2. package/dest/chain.d.ts +1 -1
  3. package/dest/client.d.ts +1 -1
  4. package/dest/client.js +6 -2
  5. package/dest/config.d.ts +20 -68
  6. package/dest/config.d.ts.map +1 -1
  7. package/dest/config.js +52 -378
  8. package/dest/constants.d.ts +1 -1
  9. package/dest/contracts/empire_base.d.ts +7 -5
  10. package/dest/contracts/empire_base.d.ts.map +1 -1
  11. package/dest/contracts/empire_base.js +1 -1
  12. package/dest/contracts/empire_slashing_proposer.d.ts +6 -4
  13. package/dest/contracts/empire_slashing_proposer.d.ts.map +1 -1
  14. package/dest/contracts/empire_slashing_proposer.js +17 -2
  15. package/dest/contracts/errors.d.ts +1 -1
  16. package/dest/contracts/errors.d.ts.map +1 -1
  17. package/dest/contracts/fee_asset_handler.d.ts +6 -5
  18. package/dest/contracts/fee_asset_handler.d.ts.map +1 -1
  19. package/dest/contracts/fee_asset_handler.js +9 -9
  20. package/dest/contracts/fee_juice.d.ts +1 -1
  21. package/dest/contracts/fee_juice.d.ts.map +1 -1
  22. package/dest/contracts/governance.d.ts +16 -16
  23. package/dest/contracts/governance.d.ts.map +1 -1
  24. package/dest/contracts/governance_proposer.d.ts +6 -4
  25. package/dest/contracts/governance_proposer.d.ts.map +1 -1
  26. package/dest/contracts/governance_proposer.js +399 -10
  27. package/dest/contracts/gse.d.ts +1 -1
  28. package/dest/contracts/gse.d.ts.map +1 -1
  29. package/dest/contracts/inbox.d.ts +24 -3
  30. package/dest/contracts/inbox.d.ts.map +1 -1
  31. package/dest/contracts/inbox.js +36 -1
  32. package/dest/contracts/index.d.ts +3 -1
  33. package/dest/contracts/index.d.ts.map +1 -1
  34. package/dest/contracts/index.js +2 -0
  35. package/dest/contracts/log.d.ts +13 -0
  36. package/dest/contracts/log.d.ts.map +1 -0
  37. package/dest/contracts/log.js +1 -0
  38. package/dest/contracts/multicall.d.ts +2 -2
  39. package/dest/contracts/multicall.d.ts.map +1 -1
  40. package/dest/contracts/outbox.d.ts +41 -0
  41. package/dest/contracts/outbox.d.ts.map +1 -0
  42. package/dest/contracts/outbox.js +86 -0
  43. package/dest/contracts/registry.d.ts +1 -1
  44. package/dest/contracts/registry.d.ts.map +1 -1
  45. package/dest/contracts/rollup.d.ts +193 -122
  46. package/dest/contracts/rollup.d.ts.map +1 -1
  47. package/dest/contracts/rollup.js +744 -182
  48. package/dest/contracts/slasher_contract.d.ts +1 -1
  49. package/dest/contracts/slasher_contract.d.ts.map +1 -1
  50. package/dest/contracts/tally_slashing_proposer.d.ts +9 -7
  51. package/dest/contracts/tally_slashing_proposer.d.ts.map +1 -1
  52. package/dest/contracts/tally_slashing_proposer.js +4 -4
  53. package/dest/contracts/utils.d.ts +1 -1
  54. package/dest/deploy_aztec_l1_contracts.d.ts +258 -0
  55. package/dest/deploy_aztec_l1_contracts.d.ts.map +1 -0
  56. package/dest/deploy_aztec_l1_contracts.js +397 -0
  57. package/dest/deploy_l1_contract.d.ts +68 -0
  58. package/dest/deploy_l1_contract.d.ts.map +1 -0
  59. package/dest/deploy_l1_contract.js +312 -0
  60. package/dest/eth-signer/eth-signer.d.ts +1 -1
  61. package/dest/eth-signer/index.d.ts +1 -1
  62. package/dest/forwarder_proxy.d.ts +32 -0
  63. package/dest/forwarder_proxy.d.ts.map +1 -0
  64. package/dest/forwarder_proxy.js +93 -0
  65. package/dest/generated/l1-contracts-defaults.d.ts +29 -0
  66. package/dest/generated/l1-contracts-defaults.d.ts.map +1 -0
  67. package/dest/generated/l1-contracts-defaults.js +29 -0
  68. package/dest/l1_artifacts.d.ts +5791 -2036
  69. package/dest/l1_artifacts.d.ts.map +1 -1
  70. package/dest/l1_contract_addresses.d.ts +3 -3
  71. package/dest/l1_contract_addresses.d.ts.map +1 -1
  72. package/dest/l1_contract_addresses.js +3 -3
  73. package/dest/l1_reader.d.ts +3 -1
  74. package/dest/l1_reader.d.ts.map +1 -1
  75. package/dest/l1_reader.js +6 -0
  76. package/dest/l1_tx_utils/config.d.ts +5 -5
  77. package/dest/l1_tx_utils/config.d.ts.map +1 -1
  78. package/dest/l1_tx_utils/config.js +30 -7
  79. package/dest/l1_tx_utils/constants.d.ts +7 -1
  80. package/dest/l1_tx_utils/constants.d.ts.map +1 -1
  81. package/dest/l1_tx_utils/constants.js +25 -0
  82. package/dest/l1_tx_utils/factory.d.ts +1 -1
  83. package/dest/l1_tx_utils/fee-strategies/index.d.ts +10 -0
  84. package/dest/l1_tx_utils/fee-strategies/index.d.ts.map +1 -0
  85. package/dest/l1_tx_utils/fee-strategies/index.js +12 -0
  86. package/dest/l1_tx_utils/fee-strategies/p75_competitive.d.ts +8 -0
  87. package/dest/l1_tx_utils/fee-strategies/p75_competitive.d.ts.map +1 -0
  88. package/dest/l1_tx_utils/fee-strategies/p75_competitive.js +129 -0
  89. package/dest/l1_tx_utils/fee-strategies/p75_competitive_blob_txs_only.d.ts +23 -0
  90. package/dest/l1_tx_utils/fee-strategies/p75_competitive_blob_txs_only.d.ts.map +1 -0
  91. package/dest/l1_tx_utils/fee-strategies/p75_competitive_blob_txs_only.js +191 -0
  92. package/dest/l1_tx_utils/fee-strategies/types.d.ts +51 -0
  93. package/dest/l1_tx_utils/fee-strategies/types.d.ts.map +1 -0
  94. package/dest/l1_tx_utils/fee-strategies/types.js +3 -0
  95. package/dest/l1_tx_utils/forwarder_l1_tx_utils.d.ts +41 -0
  96. package/dest/l1_tx_utils/forwarder_l1_tx_utils.d.ts.map +1 -0
  97. package/dest/l1_tx_utils/forwarder_l1_tx_utils.js +48 -0
  98. package/dest/l1_tx_utils/index-blobs.d.ts +3 -0
  99. package/dest/l1_tx_utils/index-blobs.d.ts.map +1 -0
  100. package/dest/l1_tx_utils/index-blobs.js +2 -0
  101. package/dest/l1_tx_utils/index.d.ts +3 -1
  102. package/dest/l1_tx_utils/index.d.ts.map +1 -1
  103. package/dest/l1_tx_utils/index.js +2 -0
  104. package/dest/l1_tx_utils/interfaces.d.ts +2 -2
  105. package/dest/l1_tx_utils/interfaces.d.ts.map +1 -1
  106. package/dest/l1_tx_utils/l1_fee_analyzer.d.ts +233 -0
  107. package/dest/l1_tx_utils/l1_fee_analyzer.d.ts.map +1 -0
  108. package/dest/l1_tx_utils/l1_fee_analyzer.js +506 -0
  109. package/dest/l1_tx_utils/l1_tx_utils.d.ts +1 -2
  110. package/dest/l1_tx_utils/l1_tx_utils.d.ts.map +1 -1
  111. package/dest/l1_tx_utils/l1_tx_utils.js +17 -4
  112. package/dest/l1_tx_utils/l1_tx_utils_with_blobs.d.ts +1 -1
  113. package/dest/l1_tx_utils/l1_tx_utils_with_blobs.d.ts.map +1 -1
  114. package/dest/l1_tx_utils/readonly_l1_tx_utils.d.ts +19 -30
  115. package/dest/l1_tx_utils/readonly_l1_tx_utils.d.ts.map +1 -1
  116. package/dest/l1_tx_utils/readonly_l1_tx_utils.js +54 -162
  117. package/dest/l1_tx_utils/signer.d.ts +1 -1
  118. package/dest/l1_tx_utils/types.d.ts +1 -1
  119. package/dest/l1_tx_utils/types.d.ts.map +1 -1
  120. package/dest/l1_tx_utils/utils.d.ts +1 -1
  121. package/dest/l1_types.d.ts +1 -1
  122. package/dest/publisher_manager.d.ts +1 -1
  123. package/dest/publisher_manager.d.ts.map +1 -1
  124. package/dest/queries.d.ts +2 -2
  125. package/dest/queries.d.ts.map +1 -1
  126. package/dest/queries.js +14 -5
  127. package/dest/test/chain_monitor.d.ts +27 -24
  128. package/dest/test/chain_monitor.d.ts.map +1 -1
  129. package/dest/test/chain_monitor.js +33 -36
  130. package/dest/test/delayed_tx_utils.d.ts +1 -1
  131. package/dest/test/delayed_tx_utils.d.ts.map +1 -1
  132. package/dest/test/eth_cheat_codes.d.ts +11 -3
  133. package/dest/test/eth_cheat_codes.d.ts.map +1 -1
  134. package/dest/test/eth_cheat_codes.js +11 -3
  135. package/dest/test/eth_cheat_codes_with_state.d.ts +1 -1
  136. package/dest/test/eth_cheat_codes_with_state.d.ts.map +1 -1
  137. package/dest/test/index.d.ts +1 -1
  138. package/dest/test/rollup_cheat_codes.d.ts +15 -13
  139. package/dest/test/rollup_cheat_codes.d.ts.map +1 -1
  140. package/dest/test/rollup_cheat_codes.js +54 -39
  141. package/dest/test/start_anvil.d.ts +4 -1
  142. package/dest/test/start_anvil.d.ts.map +1 -1
  143. package/dest/test/start_anvil.js +2 -1
  144. package/dest/test/tx_delayer.d.ts +1 -1
  145. package/dest/test/tx_delayer.d.ts.map +1 -1
  146. package/dest/test/tx_delayer.js +4 -3
  147. package/dest/test/upgrade_utils.d.ts +1 -1
  148. package/dest/types.d.ts +57 -2
  149. package/dest/types.d.ts.map +1 -1
  150. package/dest/utils.d.ts +15 -3
  151. package/dest/utils.d.ts.map +1 -1
  152. package/dest/utils.js +18 -0
  153. package/dest/zkPassportVerifierAddress.d.ts +1 -1
  154. package/package.json +34 -14
  155. package/src/client.ts +2 -2
  156. package/src/config.ts +64 -458
  157. package/src/contracts/README.md +157 -0
  158. package/src/contracts/empire_base.ts +6 -5
  159. package/src/contracts/empire_slashing_proposer.ts +16 -5
  160. package/src/contracts/fee_asset_handler.ts +8 -7
  161. package/src/contracts/governance_proposer.ts +16 -5
  162. package/src/contracts/inbox.ts +55 -3
  163. package/src/contracts/index.ts +2 -0
  164. package/src/contracts/log.ts +13 -0
  165. package/src/contracts/outbox.ts +98 -0
  166. package/src/contracts/rollup.ts +416 -159
  167. package/src/contracts/tally_slashing_proposer.ts +11 -8
  168. package/src/deploy_aztec_l1_contracts.ts +618 -0
  169. package/src/deploy_l1_contract.ts +362 -0
  170. package/src/forwarder_proxy.ts +108 -0
  171. package/src/generated/l1-contracts-defaults.ts +31 -0
  172. package/src/l1_contract_addresses.ts +22 -20
  173. package/src/l1_reader.ts +8 -0
  174. package/src/l1_tx_utils/config.ts +32 -11
  175. package/src/l1_tx_utils/constants.ts +11 -0
  176. package/src/l1_tx_utils/fee-strategies/index.ts +22 -0
  177. package/src/l1_tx_utils/fee-strategies/p75_competitive.ts +163 -0
  178. package/src/l1_tx_utils/fee-strategies/p75_competitive_blob_txs_only.ts +245 -0
  179. package/src/l1_tx_utils/fee-strategies/types.ts +56 -0
  180. package/src/l1_tx_utils/forwarder_l1_tx_utils.ts +119 -0
  181. package/src/l1_tx_utils/index-blobs.ts +2 -0
  182. package/src/l1_tx_utils/index.ts +2 -0
  183. package/src/l1_tx_utils/interfaces.ts +1 -1
  184. package/src/l1_tx_utils/l1_fee_analyzer.ts +803 -0
  185. package/src/l1_tx_utils/l1_tx_utils.ts +24 -4
  186. package/src/l1_tx_utils/readonly_l1_tx_utils.ts +67 -207
  187. package/src/queries.ts +16 -6
  188. package/src/test/chain_monitor.ts +51 -49
  189. package/src/test/eth_cheat_codes.ts +9 -3
  190. package/src/test/rollup_cheat_codes.ts +54 -43
  191. package/src/test/start_anvil.ts +4 -0
  192. package/src/test/tx_delayer.ts +5 -3
  193. package/src/types.ts +62 -0
  194. package/src/utils.ts +30 -1
  195. package/dest/deploy_l1_contracts.d.ts +0 -226
  196. package/dest/deploy_l1_contracts.d.ts.map +0 -1
  197. package/dest/deploy_l1_contracts.js +0 -1473
  198. package/dest/index.d.ts +0 -18
  199. package/dest/index.d.ts.map +0 -1
  200. package/dest/index.js +0 -17
  201. package/src/deploy_l1_contracts.ts +0 -1849
  202. package/src/index.ts +0 -17
@@ -0,0 +1,157 @@
1
+ # L1 Contract Wrappers
2
+
3
+ This folder contains TypeScript wrappers for L1 contracts defined in `l1-contracts/`. These wrappers are used by the node client to interact with the rollup and related contracts on Ethereum.
4
+
5
+ ## Purpose
6
+
7
+ The goal of wrapping is to shield consumers from L1-specific and viem-specific details. Clients using these wrappers interact with domain types and branded types native to the Aztec codebase, without needing to understand viem's ABI type system or deal with raw types like `Hex` and `bigint`.
8
+
9
+ ## Type Safety
10
+
11
+ ### Explicit Return Types
12
+
13
+ Every function in the contract wrappers must declare its return type explicitly. This is critical because viem's type inference over ABI types is slow and significantly impacts IDE performance.
14
+
15
+ ```typescript
16
+ // Good: Explicit return type
17
+ async getSlotNumber(): Promise<SlotNumber> {
18
+ return SlotNumber.fromBigInt(await this.rollup.read.getCurrentSlot());
19
+ }
20
+
21
+ // Bad: Inferred return type (slow)
22
+ async getSlotNumber() {
23
+ return SlotNumber.fromBigInt(await this.rollup.read.getCurrentSlot());
24
+ }
25
+ ```
26
+
27
+ ### Branded and Domain Types
28
+
29
+ Use branded types and domain-specific types instead of viem's autogenerated types for both arguments and return values:
30
+
31
+ - `CheckpointNumber`, `EpochNumber`, `SlotNumber` instead of `bigint`
32
+ - `EthAddress` instead of `` `0x${string}` `` or `Hex`
33
+ - `Fr`, `Buffer32` instead of `Hex` for hashes and field elements
34
+ - Custom domain types (e.g., `CheckpointLog`, `AttesterView`) instead of raw tuples
35
+
36
+ Type conversions happen inside wrapper methods, not at the call site:
37
+
38
+ ```typescript
39
+ async getCheckpoint(checkpointNumber: CheckpointNumber): Promise<CheckpointLog> {
40
+ const result = await this.rollup.read.getCheckpoint([BigInt(checkpointNumber)]);
41
+ return {
42
+ archive: Fr.fromString(result.archive),
43
+ headerHash: Buffer32.fromString(result.headerHash),
44
+ blockCount: result.blockCount,
45
+ };
46
+ }
47
+ ```
48
+
49
+ ## Wrapper Pattern
50
+
51
+ ### Basic Structure
52
+
53
+ Each wrapper follows a consistent structure:
54
+
55
+ ```typescript
56
+ export class FooContract {
57
+ private readonly foo: GetContractReturnType<typeof FooAbi, ViemClient>;
58
+
59
+ constructor(
60
+ public readonly client: ViemClient,
61
+ address: Hex | EthAddress,
62
+ ) {
63
+ if (address instanceof EthAddress) {
64
+ address = address.toString();
65
+ }
66
+ this.foo = getContract({ address, abi: FooAbi, client });
67
+ }
68
+
69
+ public get address(): Hex {
70
+ return this.foo.address;
71
+ }
72
+
73
+ public getContract(): GetContractReturnType<typeof FooAbi, ViemClient> {
74
+ return this.foo;
75
+ }
76
+ }
77
+ ```
78
+
79
+ The raw contract is exposed via `getContract()` for cases where direct access is needed, but most consumers should use the typed wrapper methods. Relying on `getContract()` is a code smell and should be avoided.
80
+
81
+ ### Static Factory Methods
82
+
83
+ Wrappers may provide static factory methods for common initialization patterns:
84
+
85
+ - `getFromConfig(config)` - construct from configuration object
86
+ - `getFromL1ContractsValues(deployResult)` - construct from deployment result
87
+
88
+ ### Wallet Assertions
89
+
90
+ For write operations that require a wallet, use an assertion helper:
91
+
92
+ ```typescript
93
+ private assertWallet(): GetContractReturnType<typeof FooAbi, ExtendedViemWalletClient> {
94
+ if (!isExtendedClient(this.client)) {
95
+ throw new Error('Wallet client is required for this operation');
96
+ }
97
+ return this.foo as GetContractReturnType<typeof FooAbi, ExtendedViemWalletClient>;
98
+ }
99
+ ```
100
+
101
+ ## Event Handling
102
+
103
+ ### Event Log Types
104
+
105
+ Event logs are wrapped in `L1EventLog<T>` to include L1 block context:
106
+
107
+ ```typescript
108
+ type L1EventLog<T> = {
109
+ l1BlockNumber: bigint;
110
+ l1BlockHash: Buffer32;
111
+ l1TransactionHash: Hex;
112
+ args: T;
113
+ };
114
+ ```
115
+
116
+ ### Event Fetching
117
+
118
+ Methods that fetch events convert viem's raw logs to domain types:
119
+
120
+ ```typescript
121
+ async getCheckpointProposedEvents(fromBlock: bigint, toBlock: bigint): Promise<CheckpointProposedLog[]> {
122
+ const logs = await this.rollup.getEvents.CheckpointProposed({}, { fromBlock, toBlock });
123
+ return logs.map(log => ({
124
+ l1BlockNumber: log.blockNumber!,
125
+ l1BlockHash: Buffer32.fromString(log.blockHash!),
126
+ l1TransactionHash: log.transactionHash!,
127
+ args: {
128
+ checkpointNumber: CheckpointNumber.fromBigInt(log.args.checkpointNumber!),
129
+ // ... convert other fields
130
+ },
131
+ }));
132
+ }
133
+ ```
134
+
135
+ ### Event Listeners
136
+
137
+ For reactive event handling, wrapper methods convert arguments before invoking callbacks:
138
+
139
+ ```typescript
140
+ public listenToCheckpointInvalidated(
141
+ callback: (args: { checkpointNumber: CheckpointNumber }) => unknown,
142
+ ): WatchContractEventReturnType {
143
+ return this.rollup.watchEvent.CheckpointInvalidated({}, {
144
+ onLogs: logs => {
145
+ for (const log of logs) {
146
+ if (log.args.checkpointNumber !== undefined) {
147
+ callback({ checkpointNumber: CheckpointNumber.fromBigInt(log.args.checkpointNumber) });
148
+ }
149
+ }
150
+ },
151
+ });
152
+ }
153
+ ```
154
+
155
+ ## Error Handling
156
+
157
+ Custom error classes in `errors.ts` extend `Error` and set `this.name` for proper error identification. Include relevant context as public readonly properties.
@@ -1,3 +1,4 @@
1
+ import type { SlotNumber } from '@aztec/foundation/branded-types';
1
2
  import type { EthAddress } from '@aztec/foundation/eth-address';
2
3
  import { Signature } from '@aztec/foundation/eth-signature';
3
4
  import { EmpireBaseAbi } from '@aztec/l1-artifacts/EmpireBaseAbi';
@@ -11,12 +12,12 @@ export interface IEmpireBase {
11
12
  getRoundInfo(
12
13
  rollupAddress: Hex,
13
14
  round: bigint,
14
- ): Promise<{ lastSignalSlot: bigint; payloadWithMostSignals: Hex; executed: boolean }>;
15
- computeRound(slot: bigint): Promise<bigint>;
15
+ ): Promise<{ lastSignalSlot: SlotNumber; payloadWithMostSignals: Hex; quorumReached: boolean; executed: boolean }>;
16
+ computeRound(slot: SlotNumber): Promise<bigint>;
16
17
  createSignalRequest(payload: Hex): L1TxRequest;
17
18
  createSignalRequestWithSignature(
18
19
  payload: Hex,
19
- round: bigint,
20
+ slot: SlotNumber,
20
21
  chainId: number,
21
22
  signerAddress: Hex,
22
23
  signer: (msg: TypedDataDefinition) => Promise<Hex>,
@@ -51,7 +52,7 @@ export function encodeSignalWithSignature(payload: Hex, signature: Signature) {
51
52
  export async function signSignalWithSig(
52
53
  signer: (msg: TypedDataDefinition) => Promise<Hex>,
53
54
  payload: Hex,
54
- slot: bigint,
55
+ slot: SlotNumber,
55
56
  instance: Hex,
56
57
  verifyingContract: Hex,
57
58
  chainId: number,
@@ -79,7 +80,7 @@ export async function signSignalWithSig(
79
80
 
80
81
  const message = {
81
82
  payload,
82
- slot,
83
+ slot: BigInt(slot),
83
84
  instance,
84
85
  };
85
86
 
@@ -1,3 +1,4 @@
1
+ import { SlotNumber } from '@aztec/foundation/branded-types';
1
2
  import { EthAddress } from '@aztec/foundation/eth-address';
2
3
  import { createLogger } from '@aztec/foundation/log';
3
4
  import { retryUntil } from '@aztec/foundation/retry';
@@ -67,8 +68,8 @@ export class EmpireSlashingProposerContract extends EventEmitter implements IEmp
67
68
  return this.proposer.read.getCurrentRound();
68
69
  }
69
70
 
70
- public computeRound(slot: bigint): Promise<bigint> {
71
- return this.proposer.read.computeRound([slot]);
71
+ public computeRound(slot: SlotNumber): Promise<bigint> {
72
+ return this.proposer.read.computeRound([BigInt(slot)]);
72
73
  }
73
74
 
74
75
  public getInstance() {
@@ -78,8 +79,18 @@ export class EmpireSlashingProposerContract extends EventEmitter implements IEmp
78
79
  public async getRoundInfo(
79
80
  rollupAddress: Hex,
80
81
  round: bigint,
81
- ): Promise<{ lastSignalSlot: bigint; payloadWithMostSignals: Hex; executed: boolean }> {
82
- return await this.proposer.read.getRoundData([rollupAddress, round]);
82
+ ): Promise<{ lastSignalSlot: SlotNumber; payloadWithMostSignals: Hex; quorumReached: boolean; executed: boolean }> {
83
+ const result = await this.proposer.read.getRoundData([rollupAddress, round]);
84
+ const [signalCount, quorum] = await Promise.all([
85
+ this.proposer.read.signalCount([rollupAddress, round, result.payloadWithMostSignals]),
86
+ this.getQuorumSize(),
87
+ ]);
88
+ return {
89
+ lastSignalSlot: SlotNumber.fromBigInt(result.lastSignalSlot),
90
+ payloadWithMostSignals: result.payloadWithMostSignals,
91
+ quorumReached: signalCount >= quorum,
92
+ executed: result.executed,
93
+ };
83
94
  }
84
95
 
85
96
  public getPayloadSignals(rollupAddress: Hex, round: bigint, payload: Hex): Promise<bigint> {
@@ -95,7 +106,7 @@ export class EmpireSlashingProposerContract extends EventEmitter implements IEmp
95
106
 
96
107
  public async createSignalRequestWithSignature(
97
108
  payload: Hex,
98
- slot: bigint,
109
+ slot: SlotNumber,
99
110
  chainId: number,
100
111
  signerAddress: Hex,
101
112
  signer: (msg: TypedDataDefinition) => Promise<Hex>,
@@ -4,13 +4,14 @@ import { FeeAssetHandlerAbi } from '@aztec/l1-artifacts/FeeAssetHandlerAbi';
4
4
  import { type Hex, encodeFunctionData, getContract } from 'viem';
5
5
 
6
6
  import type { L1TxUtils } from '../l1_tx_utils/index.js';
7
+ import type { ViemClient } from '../types.js';
7
8
 
8
9
  export class FeeAssetHandlerContract {
9
10
  public address: EthAddress;
10
11
 
11
12
  constructor(
13
+ public readonly client: ViemClient,
12
14
  address: Hex | EthAddress,
13
- public readonly txUtils: L1TxUtils,
14
15
  ) {
15
16
  if (address instanceof EthAddress) {
16
17
  address = address.toString();
@@ -22,7 +23,7 @@ export class FeeAssetHandlerContract {
22
23
  const contract = getContract({
23
24
  abi: FeeAssetHandlerAbi,
24
25
  address: this.address.toString(),
25
- client: this.txUtils.client,
26
+ client: this.client,
26
27
  });
27
28
  return EthAddress.fromString(await contract.read.owner());
28
29
  }
@@ -31,16 +32,16 @@ export class FeeAssetHandlerContract {
31
32
  const contract = getContract({
32
33
  abi: FeeAssetHandlerAbi,
33
34
  address: this.address.toString(),
34
- client: this.txUtils.client,
35
+ client: this.client,
35
36
  });
36
37
  return contract.read.mintAmount();
37
38
  }
38
39
 
39
- public mint(recipient: Hex | EthAddress) {
40
+ public mint(txUtils: L1TxUtils, recipient: Hex | EthAddress) {
40
41
  if (recipient instanceof EthAddress) {
41
42
  recipient = recipient.toString();
42
43
  }
43
- return this.txUtils.sendAndMonitorTransaction({
44
+ return txUtils.sendAndMonitorTransaction({
44
45
  to: this.address.toString(),
45
46
  data: encodeFunctionData({
46
47
  abi: FeeAssetHandlerAbi,
@@ -50,8 +51,8 @@ export class FeeAssetHandlerContract {
50
51
  });
51
52
  }
52
53
 
53
- public setMintAmount(amount: bigint) {
54
- return this.txUtils.sendAndMonitorTransaction({
54
+ public setMintAmount(txUtils: L1TxUtils, amount: bigint) {
55
+ return txUtils.sendAndMonitorTransaction({
55
56
  to: this.address.toString(),
56
57
  data: encodeFunctionData({
57
58
  abi: FeeAssetHandlerAbi,
@@ -1,3 +1,4 @@
1
+ import { SlotNumber } from '@aztec/foundation/branded-types';
1
2
  import { memoize } from '@aztec/foundation/decorators';
2
3
  import { EthAddress } from '@aztec/foundation/eth-address';
3
4
  import { GovernanceProposerAbi } from '@aztec/l1-artifacts/GovernanceProposerAbi';
@@ -54,15 +55,25 @@ export class GovernanceProposerContract implements IEmpireBase {
54
55
  return this.proposer.read.getInstance();
55
56
  }
56
57
 
57
- public computeRound(slot: bigint): Promise<bigint> {
58
- return this.proposer.read.computeRound([slot]);
58
+ public computeRound(slot: SlotNumber): Promise<bigint> {
59
+ return this.proposer.read.computeRound([BigInt(slot)]);
59
60
  }
60
61
 
61
62
  public async getRoundInfo(
62
63
  rollupAddress: Hex,
63
64
  round: bigint,
64
- ): Promise<{ lastSignalSlot: bigint; payloadWithMostSignals: Hex; executed: boolean }> {
65
- return await this.proposer.read.getRoundData([rollupAddress, round]);
65
+ ): Promise<{ lastSignalSlot: SlotNumber; payloadWithMostSignals: Hex; quorumReached: boolean; executed: boolean }> {
66
+ const result = await this.proposer.read.getRoundData([rollupAddress, round]);
67
+ const [signalCount, quorum] = await Promise.all([
68
+ this.proposer.read.signalCount([rollupAddress, round, result.payloadWithMostSignals]),
69
+ this.getQuorumSize(),
70
+ ]);
71
+ return {
72
+ lastSignalSlot: SlotNumber.fromBigInt(result.lastSignalSlot),
73
+ payloadWithMostSignals: result.payloadWithMostSignals,
74
+ quorumReached: signalCount >= quorum,
75
+ executed: result.executed,
76
+ };
66
77
  }
67
78
 
68
79
  public getPayloadSignals(rollupAddress: Hex, round: bigint, payload: Hex): Promise<bigint> {
@@ -78,7 +89,7 @@ export class GovernanceProposerContract implements IEmpireBase {
78
89
 
79
90
  public async createSignalRequestWithSignature(
80
91
  payload: Hex,
81
- slot: bigint,
92
+ slot: SlotNumber,
82
93
  chainId: number,
83
94
  signerAddress: Hex,
84
95
  signer: (msg: TypedDataDefinition) => Promise<Hex>,
@@ -1,19 +1,33 @@
1
- import { Buffer16 } from '@aztec/foundation/buffer';
1
+ import { CheckpointNumber } from '@aztec/foundation/branded-types';
2
+ import { Buffer16, Buffer32 } from '@aztec/foundation/buffer';
3
+ import { Fr } from '@aztec/foundation/curves/bn254';
2
4
  import { EthAddress } from '@aztec/foundation/eth-address';
3
5
  import { InboxAbi } from '@aztec/l1-artifacts/InboxAbi';
4
6
 
5
7
  import { type BlockTag, type GetContractReturnType, type Hex, getContract } from 'viem';
6
8
 
7
9
  import { getPublicClient } from '../client.js';
8
- import type { DeployL1ContractsReturnType } from '../deploy_l1_contracts.js';
10
+ import type { DeployAztecL1ContractsReturnType } from '../deploy_aztec_l1_contracts.js';
9
11
  import type { L1ReaderConfig } from '../l1_reader.js';
10
12
  import type { ViemClient } from '../types.js';
13
+ import type { L1EventLog } from './log.js';
11
14
  import { checkBlockTag } from './utils.js';
12
15
 
16
+ /** Arguments for the MessageSent event. */
17
+ export type MessageSentArgs = {
18
+ index: bigint;
19
+ leaf: Fr;
20
+ checkpointNumber: CheckpointNumber;
21
+ rollingHash: Buffer16;
22
+ };
23
+
24
+ /** Log type for MessageSent events. */
25
+ export type MessageSentLog = L1EventLog<MessageSentArgs>;
26
+
13
27
  export class InboxContract {
14
28
  private readonly inbox: GetContractReturnType<typeof InboxAbi, ViemClient>;
15
29
 
16
- static getFromL1ContractsValues(deployL1ContractsValues: DeployL1ContractsReturnType) {
30
+ static getFromL1ContractsValues(deployL1ContractsValues: DeployAztecL1ContractsReturnType) {
17
31
  const {
18
32
  l1Client,
19
33
  l1ContractAddresses: { inboxAddress },
@@ -45,6 +59,11 @@ export class InboxContract {
45
59
  return this.inbox;
46
60
  }
47
61
 
62
+ public async getLag(opts: { blockTag?: BlockTag; blockNumber?: bigint } = {}): Promise<bigint> {
63
+ await checkBlockTag(opts.blockNumber, this.client);
64
+ return await this.inbox.read.LAG(opts);
65
+ }
66
+
48
67
  public async getState(opts: { blockTag?: BlockTag; blockNumber?: bigint } = {}): Promise<InboxContractState> {
49
68
  await checkBlockTag(opts.blockNumber, this.client);
50
69
  const state = await this.inbox.read.getState(opts);
@@ -54,6 +73,39 @@ export class InboxContract {
54
73
  treeInProgress: state.inProgress,
55
74
  };
56
75
  }
76
+
77
+ /** Fetches MessageSent events within the given block range. */
78
+ async getMessageSentEvents(fromBlock: bigint, toBlock: bigint): Promise<MessageSentLog[]> {
79
+ const logs = await this.inbox.getEvents.MessageSent({}, { fromBlock, toBlock });
80
+ return logs
81
+ .filter(log => log.blockNumber! >= fromBlock && log.blockNumber! <= toBlock)
82
+ .map(log => this.mapMessageSentLog(log));
83
+ }
84
+
85
+ /** Fetches MessageSent events for a specific message hash within the given block range. */
86
+ async getMessageSentEventByHash(hash: Hex, fromBlock: bigint, toBlock: bigint): Promise<MessageSentLog[]> {
87
+ const logs = await this.inbox.getEvents.MessageSent({ hash }, { fromBlock, toBlock });
88
+ return logs.map(log => this.mapMessageSentLog(log));
89
+ }
90
+
91
+ private mapMessageSentLog(log: {
92
+ blockNumber: bigint | null;
93
+ blockHash: `0x${string}` | null;
94
+ transactionHash: `0x${string}` | null;
95
+ args: { index?: bigint; hash?: `0x${string}`; checkpointNumber?: bigint; rollingHash?: `0x${string}` };
96
+ }): MessageSentLog {
97
+ return {
98
+ l1BlockNumber: log.blockNumber!,
99
+ l1BlockHash: Buffer32.fromString(log.blockHash!),
100
+ l1TransactionHash: log.transactionHash!,
101
+ args: {
102
+ index: log.args.index!,
103
+ leaf: Fr.fromString(log.args.hash!),
104
+ checkpointNumber: CheckpointNumber.fromBigInt(log.args.checkpointNumber!),
105
+ rollingHash: Buffer16.fromString(log.args.rollingHash!),
106
+ },
107
+ };
108
+ }
57
109
  }
58
110
 
59
111
  export type InboxContractState = {
@@ -6,7 +6,9 @@ export * from './governance.js';
6
6
  export * from './governance_proposer.js';
7
7
  export * from './gse.js';
8
8
  export * from './inbox.js';
9
+ export * from './log.js';
9
10
  export * from './multicall.js';
11
+ export * from './outbox.js';
10
12
  export * from './registry.js';
11
13
  export * from './rollup.js';
12
14
  export * from './empire_slashing_proposer.js';
@@ -0,0 +1,13 @@
1
+ import type { Buffer32 } from '@aztec/foundation/buffer';
2
+
3
+ /** Base L1 event log with common fields. */
4
+ export type L1EventLog<T> = {
5
+ /** L1 block number where the event was emitted. */
6
+ l1BlockNumber: bigint;
7
+ /** L1 block hash. */
8
+ l1BlockHash: Buffer32;
9
+ /** L1 transaction hash that emitted the event. */
10
+ l1TransactionHash: `0x${string}`;
11
+ /** Event-specific arguments. */
12
+ args: T;
13
+ };
@@ -0,0 +1,98 @@
1
+ import type { EpochNumber } from '@aztec/foundation/branded-types';
2
+ import { EthAddress } from '@aztec/foundation/eth-address';
3
+ import { OutboxAbi } from '@aztec/l1-artifacts/OutboxAbi';
4
+
5
+ import { type GetContractReturnType, type Hex, encodeAbiParameters, getContract, hexToBigInt, keccak256 } from 'viem';
6
+
7
+ import { getPublicClient } from '../client.js';
8
+ import type { DeployAztecL1ContractsReturnType } from '../deploy_aztec_l1_contracts.js';
9
+ import type { L1ReaderConfig } from '../l1_reader.js';
10
+ import { type ExtendedViemWalletClient, type ViemClient, isExtendedClient } from '../types.js';
11
+
12
+ export type ViemL1Actor = {
13
+ actor: Hex;
14
+ chainId: bigint;
15
+ };
16
+
17
+ export type ViemL2Actor = {
18
+ actor: Hex;
19
+ version: bigint;
20
+ };
21
+
22
+ export type ViemL2ToL1Msg = {
23
+ sender: ViemL2Actor;
24
+ recipient: ViemL1Actor;
25
+ content: Hex;
26
+ };
27
+
28
+ export class OutboxContract {
29
+ private readonly outbox: GetContractReturnType<typeof OutboxAbi, ViemClient>;
30
+
31
+ static getFromL1ContractsValues(deployL1ContractsValues: DeployAztecL1ContractsReturnType) {
32
+ const {
33
+ l1Client,
34
+ l1ContractAddresses: { outboxAddress },
35
+ } = deployL1ContractsValues;
36
+ return new OutboxContract(l1Client, outboxAddress.toString());
37
+ }
38
+
39
+ static getFromConfig(config: L1ReaderConfig) {
40
+ const client = getPublicClient(config);
41
+ const address = config.l1Contracts.outboxAddress.toString();
42
+ return new OutboxContract(client, address);
43
+ }
44
+
45
+ static getEpochRootStorageSlot(epoch: EpochNumber) {
46
+ return hexToBigInt(keccak256(encodeAbiParameters([{ type: 'uint256' }, { type: 'uint256' }], [BigInt(epoch), 0n])));
47
+ }
48
+
49
+ constructor(
50
+ public readonly client: ViemClient,
51
+ address: Hex | EthAddress,
52
+ ) {
53
+ if (address instanceof EthAddress) {
54
+ address = address.toString();
55
+ }
56
+ this.outbox = getContract({ address, abi: OutboxAbi, client });
57
+ }
58
+
59
+ public get address() {
60
+ return this.outbox.address;
61
+ }
62
+
63
+ public getContract(): GetContractReturnType<typeof OutboxAbi, ViemClient> {
64
+ return this.outbox;
65
+ }
66
+
67
+ public hasMessageBeenConsumedAtEpoch(epoch: EpochNumber, leafId: bigint) {
68
+ return this.outbox.read.hasMessageBeenConsumedAtEpoch([BigInt(epoch), leafId]);
69
+ }
70
+
71
+ public getRootData(epoch: EpochNumber) {
72
+ return this.outbox.read.getRootData([BigInt(epoch)]);
73
+ }
74
+
75
+ public consume(message: ViemL2ToL1Msg, epoch: EpochNumber, leafIndex: bigint, path: Hex[]) {
76
+ const wallet = this.assertWallet();
77
+ return wallet.write.consume([message, BigInt(epoch), leafIndex, path]);
78
+ }
79
+
80
+ public async getMessageConsumedEvents(
81
+ l1BlockHash: Hex,
82
+ ): Promise<{ epoch: bigint; root: Hex; messageHash: Hex; leafId: bigint }[]> {
83
+ const events = await this.outbox.getEvents.MessageConsumed({}, { blockHash: l1BlockHash, strict: true });
84
+ return events.map(event => ({
85
+ epoch: event.args.epoch!,
86
+ root: event.args.root!,
87
+ messageHash: event.args.messageHash!,
88
+ leafId: event.args.leafId!,
89
+ }));
90
+ }
91
+
92
+ private assertWallet(): GetContractReturnType<typeof OutboxAbi, ExtendedViemWalletClient> {
93
+ if (!isExtendedClient(this.client)) {
94
+ throw new Error('Wallet client is required for this operation');
95
+ }
96
+ return this.outbox as GetContractReturnType<typeof OutboxAbi, ExtendedViemWalletClient>;
97
+ }
98
+ }