@aztec/ethereum 0.0.0-test.1 → 0.0.1-commit.b655e406

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 (229) hide show
  1. package/dest/account.d.ts +2 -0
  2. package/dest/account.d.ts.map +1 -0
  3. package/dest/account.js +4 -0
  4. package/dest/client.d.ts +5 -3
  5. package/dest/client.d.ts.map +1 -1
  6. package/dest/client.js +16 -2
  7. package/dest/config.d.ts +107 -16
  8. package/dest/config.d.ts.map +1 -1
  9. package/dest/config.js +456 -22
  10. package/dest/contracts/empire_base.d.ts +21 -6
  11. package/dest/contracts/empire_base.d.ts.map +1 -1
  12. package/dest/contracts/empire_base.js +75 -2
  13. package/dest/contracts/empire_slashing_proposer.d.ts +65 -0
  14. package/dest/contracts/empire_slashing_proposer.d.ts.map +1 -0
  15. package/dest/contracts/empire_slashing_proposer.js +194 -0
  16. package/dest/contracts/errors.d.ts +7 -0
  17. package/dest/contracts/errors.d.ts.map +1 -0
  18. package/dest/contracts/errors.js +12 -0
  19. package/dest/contracts/fee_asset_handler.d.ts +19 -0
  20. package/dest/contracts/fee_asset_handler.d.ts.map +1 -0
  21. package/dest/contracts/fee_asset_handler.js +57 -0
  22. package/dest/contracts/fee_juice.d.ts +5 -6
  23. package/dest/contracts/fee_juice.d.ts.map +1 -1
  24. package/dest/contracts/fee_juice.js +27 -20
  25. package/dest/contracts/governance.d.ts +36 -25
  26. package/dest/contracts/governance.d.ts.map +1 -1
  27. package/dest/contracts/governance.js +87 -84
  28. package/dest/contracts/governance_proposer.d.ts +13 -11
  29. package/dest/contracts/governance_proposer.d.ts.map +1 -1
  30. package/dest/contracts/governance_proposer.js +32 -18
  31. package/dest/contracts/gse.d.ts +32 -0
  32. package/dest/contracts/gse.d.ts.map +1 -0
  33. package/dest/contracts/gse.js +72 -0
  34. package/dest/contracts/inbox.d.ts +26 -0
  35. package/dest/contracts/inbox.d.ts.map +1 -0
  36. package/dest/contracts/inbox.js +45 -0
  37. package/dest/contracts/index.d.ts +8 -2
  38. package/dest/contracts/index.d.ts.map +1 -1
  39. package/dest/contracts/index.js +8 -2
  40. package/dest/contracts/multicall.d.ts +21 -0
  41. package/dest/contracts/multicall.d.ts.map +1 -0
  42. package/dest/contracts/multicall.js +156 -0
  43. package/dest/contracts/registry.d.ts +9 -4
  44. package/dest/contracts/registry.d.ts.map +1 -1
  45. package/dest/contracts/registry.js +44 -16
  46. package/dest/contracts/rollup.d.ts +202 -29
  47. package/dest/contracts/rollup.d.ts.map +1 -1
  48. package/dest/contracts/rollup.js +500 -55
  49. package/dest/contracts/slasher_contract.d.ts +44 -0
  50. package/dest/contracts/slasher_contract.d.ts.map +1 -0
  51. package/dest/contracts/slasher_contract.js +75 -0
  52. package/dest/contracts/tally_slashing_proposer.d.ts +138 -0
  53. package/dest/contracts/tally_slashing_proposer.d.ts.map +1 -0
  54. package/dest/contracts/tally_slashing_proposer.js +313 -0
  55. package/dest/contracts/utils.d.ts +3 -0
  56. package/dest/contracts/utils.d.ts.map +1 -0
  57. package/dest/contracts/utils.js +11 -0
  58. package/dest/deploy_l1_contracts.d.ts +128 -21112
  59. package/dest/deploy_l1_contracts.d.ts.map +1 -1
  60. package/dest/deploy_l1_contracts.js +1204 -418
  61. package/dest/eth-signer/eth-signer.d.ts +21 -0
  62. package/dest/eth-signer/eth-signer.d.ts.map +1 -0
  63. package/dest/eth-signer/eth-signer.js +5 -0
  64. package/dest/eth-signer/index.d.ts +2 -0
  65. package/dest/eth-signer/index.d.ts.map +1 -0
  66. package/dest/eth-signer/index.js +1 -0
  67. package/dest/index.d.ts +6 -2
  68. package/dest/index.d.ts.map +1 -1
  69. package/dest/index.js +6 -2
  70. package/dest/l1_artifacts.d.ts +76184 -0
  71. package/dest/l1_artifacts.d.ts.map +1 -0
  72. package/dest/l1_artifacts.js +166 -0
  73. package/dest/l1_contract_addresses.d.ts +21 -1
  74. package/dest/l1_contract_addresses.d.ts.map +1 -1
  75. package/dest/l1_contract_addresses.js +22 -18
  76. package/dest/l1_reader.d.ts +1 -1
  77. package/dest/l1_reader.d.ts.map +1 -1
  78. package/dest/l1_reader.js +8 -8
  79. package/dest/l1_tx_utils/config.d.ts +59 -0
  80. package/dest/l1_tx_utils/config.d.ts.map +1 -0
  81. package/dest/l1_tx_utils/config.js +73 -0
  82. package/dest/l1_tx_utils/constants.d.ts +6 -0
  83. package/dest/l1_tx_utils/constants.d.ts.map +1 -0
  84. package/dest/l1_tx_utils/constants.js +14 -0
  85. package/dest/l1_tx_utils/factory.d.ts +24 -0
  86. package/dest/l1_tx_utils/factory.d.ts.map +1 -0
  87. package/dest/l1_tx_utils/factory.js +12 -0
  88. package/dest/l1_tx_utils/index.d.ts +10 -0
  89. package/dest/l1_tx_utils/index.d.ts.map +1 -0
  90. package/dest/l1_tx_utils/index.js +10 -0
  91. package/dest/l1_tx_utils/interfaces.d.ts +76 -0
  92. package/dest/l1_tx_utils/interfaces.d.ts.map +1 -0
  93. package/dest/l1_tx_utils/interfaces.js +4 -0
  94. package/dest/l1_tx_utils/l1_tx_utils.d.ts +95 -0
  95. package/dest/l1_tx_utils/l1_tx_utils.d.ts.map +1 -0
  96. package/dest/l1_tx_utils/l1_tx_utils.js +610 -0
  97. package/dest/l1_tx_utils/l1_tx_utils_with_blobs.d.ts +26 -0
  98. package/dest/l1_tx_utils/l1_tx_utils_with_blobs.d.ts.map +1 -0
  99. package/dest/l1_tx_utils/l1_tx_utils_with_blobs.js +26 -0
  100. package/dest/l1_tx_utils/readonly_l1_tx_utils.d.ts +94 -0
  101. package/dest/l1_tx_utils/readonly_l1_tx_utils.d.ts.map +1 -0
  102. package/dest/l1_tx_utils/readonly_l1_tx_utils.js +431 -0
  103. package/dest/l1_tx_utils/signer.d.ts +4 -0
  104. package/dest/l1_tx_utils/signer.d.ts.map +1 -0
  105. package/dest/l1_tx_utils/signer.js +16 -0
  106. package/dest/l1_tx_utils/types.d.ts +67 -0
  107. package/dest/l1_tx_utils/types.d.ts.map +1 -0
  108. package/dest/l1_tx_utils/types.js +26 -0
  109. package/dest/l1_tx_utils/utils.d.ts +4 -0
  110. package/dest/l1_tx_utils/utils.d.ts.map +1 -0
  111. package/dest/l1_tx_utils/utils.js +14 -0
  112. package/dest/l1_types.d.ts +6 -0
  113. package/dest/l1_types.d.ts.map +1 -0
  114. package/dest/l1_types.js +1 -0
  115. package/dest/publisher_manager.d.ts +15 -0
  116. package/dest/publisher_manager.d.ts.map +1 -0
  117. package/dest/publisher_manager.js +88 -0
  118. package/dest/queries.d.ts +3 -1
  119. package/dest/queries.d.ts.map +1 -1
  120. package/dest/queries.js +51 -12
  121. package/dest/test/chain_monitor.d.ts +72 -0
  122. package/dest/test/chain_monitor.d.ts.map +1 -0
  123. package/dest/test/chain_monitor.js +216 -0
  124. package/dest/test/delayed_tx_utils.d.ts +7 -2
  125. package/dest/test/delayed_tx_utils.d.ts.map +1 -1
  126. package/dest/test/delayed_tx_utils.js +13 -6
  127. package/dest/{eth_cheat_codes.d.ts → test/eth_cheat_codes.d.ts} +87 -13
  128. package/dest/test/eth_cheat_codes.d.ts.map +1 -0
  129. package/dest/test/eth_cheat_codes.js +552 -0
  130. package/dest/test/eth_cheat_codes_with_state.d.ts +1 -1
  131. package/dest/test/eth_cheat_codes_with_state.d.ts.map +1 -1
  132. package/dest/test/eth_cheat_codes_with_state.js +1 -1
  133. package/dest/test/index.d.ts +3 -0
  134. package/dest/test/index.d.ts.map +1 -1
  135. package/dest/test/index.js +3 -0
  136. package/dest/test/rollup_cheat_codes.d.ts +86 -0
  137. package/dest/test/rollup_cheat_codes.d.ts.map +1 -0
  138. package/dest/test/rollup_cheat_codes.js +268 -0
  139. package/dest/test/start_anvil.d.ts +5 -0
  140. package/dest/test/start_anvil.d.ts.map +1 -1
  141. package/dest/test/start_anvil.js +15 -7
  142. package/dest/test/tx_delayer.d.ts +17 -6
  143. package/dest/test/tx_delayer.d.ts.map +1 -1
  144. package/dest/test/tx_delayer.js +95 -19
  145. package/dest/test/upgrade_utils.d.ts +5 -4
  146. package/dest/test/upgrade_utils.d.ts.map +1 -1
  147. package/dest/test/upgrade_utils.js +23 -16
  148. package/dest/types.d.ts +6 -7
  149. package/dest/types.d.ts.map +1 -1
  150. package/dest/types.js +3 -1
  151. package/dest/utils.d.ts +1 -0
  152. package/dest/utils.d.ts.map +1 -1
  153. package/dest/utils.js +43 -88
  154. package/dest/zkPassportVerifierAddress.d.ts +15 -0
  155. package/dest/zkPassportVerifierAddress.d.ts.map +1 -0
  156. package/dest/zkPassportVerifierAddress.js +11 -0
  157. package/package.json +24 -16
  158. package/src/account.ts +5 -0
  159. package/src/client.ts +42 -4
  160. package/src/config.ts +584 -31
  161. package/src/contracts/empire_base.ts +75 -6
  162. package/src/contracts/empire_slashing_proposer.ts +259 -0
  163. package/src/contracts/errors.ts +13 -0
  164. package/src/contracts/fee_asset_handler.ts +63 -0
  165. package/src/contracts/fee_juice.ts +29 -15
  166. package/src/contracts/governance.ts +80 -77
  167. package/src/contracts/governance_proposer.ts +60 -24
  168. package/src/contracts/gse.ts +88 -0
  169. package/src/contracts/inbox.ts +63 -0
  170. package/src/contracts/index.ts +8 -2
  171. package/src/contracts/multicall.ts +155 -0
  172. package/src/contracts/registry.ts +51 -26
  173. package/src/contracts/rollup.ts +585 -56
  174. package/src/contracts/slasher_contract.ts +89 -0
  175. package/src/contracts/tally_slashing_proposer.ts +315 -0
  176. package/src/contracts/utils.ts +14 -0
  177. package/src/deploy_l1_contracts.ts +1467 -566
  178. package/src/eth-signer/eth-signer.ts +25 -0
  179. package/src/eth-signer/index.ts +1 -0
  180. package/src/index.ts +6 -2
  181. package/src/l1_artifacts.ts +254 -0
  182. package/src/l1_contract_addresses.ts +32 -19
  183. package/src/l1_reader.ts +9 -9
  184. package/src/l1_tx_utils/README.md +177 -0
  185. package/src/l1_tx_utils/config.ts +140 -0
  186. package/src/l1_tx_utils/constants.ts +18 -0
  187. package/src/l1_tx_utils/factory.ts +64 -0
  188. package/src/l1_tx_utils/index.ts +12 -0
  189. package/src/l1_tx_utils/interfaces.ts +86 -0
  190. package/src/l1_tx_utils/l1_tx_utils.ts +718 -0
  191. package/src/l1_tx_utils/l1_tx_utils_with_blobs.ts +77 -0
  192. package/src/l1_tx_utils/readonly_l1_tx_utils.ts +559 -0
  193. package/src/l1_tx_utils/signer.ts +28 -0
  194. package/src/l1_tx_utils/types.ts +85 -0
  195. package/src/l1_tx_utils/utils.ts +16 -0
  196. package/src/l1_types.ts +6 -0
  197. package/src/publisher_manager.ts +106 -0
  198. package/src/queries.ts +70 -15
  199. package/src/test/chain_monitor.ts +243 -0
  200. package/src/test/delayed_tx_utils.ts +34 -6
  201. package/src/test/eth_cheat_codes.ts +582 -0
  202. package/src/test/eth_cheat_codes_with_state.ts +1 -1
  203. package/src/test/index.ts +3 -0
  204. package/src/test/rollup_cheat_codes.ts +310 -0
  205. package/src/test/start_anvil.ts +20 -5
  206. package/src/test/tx_delayer.ts +127 -26
  207. package/src/test/upgrade_utils.ts +30 -21
  208. package/src/types.ts +10 -8
  209. package/src/utils.ts +49 -90
  210. package/src/zkPassportVerifierAddress.ts +15 -0
  211. package/dest/contracts/forwarder.d.ts +0 -24
  212. package/dest/contracts/forwarder.d.ts.map +0 -1
  213. package/dest/contracts/forwarder.js +0 -101
  214. package/dest/contracts/slashing_proposer.d.ts +0 -21
  215. package/dest/contracts/slashing_proposer.d.ts.map +0 -1
  216. package/dest/contracts/slashing_proposer.js +0 -47
  217. package/dest/eth_cheat_codes.d.ts.map +0 -1
  218. package/dest/eth_cheat_codes.js +0 -303
  219. package/dest/l1_tx_utils.d.ts +0 -192
  220. package/dest/l1_tx_utils.d.ts.map +0 -1
  221. package/dest/l1_tx_utils.js +0 -641
  222. package/dest/l1_tx_utils_with_blobs.d.ts +0 -12
  223. package/dest/l1_tx_utils_with_blobs.d.ts.map +0 -1
  224. package/dest/l1_tx_utils_with_blobs.js +0 -64
  225. package/src/contracts/forwarder.ts +0 -132
  226. package/src/contracts/slashing_proposer.ts +0 -51
  227. package/src/eth_cheat_codes.ts +0 -314
  228. package/src/l1_tx_utils.ts +0 -847
  229. package/src/l1_tx_utils_with_blobs.ts +0 -86
package/src/config.ts CHANGED
@@ -1,11 +1,23 @@
1
1
  import {
2
2
  type ConfigMappingsType,
3
+ type NetworkNames,
3
4
  bigintConfigHelper,
5
+ booleanConfigHelper,
6
+ enumConfigHelper,
4
7
  getConfigFromMappings,
5
8
  numberConfigHelper,
9
+ optionalNumberConfigHelper,
6
10
  } from '@aztec/foundation/config';
11
+ import { EthAddress } from '@aztec/foundation/eth-address';
7
12
 
8
- import { type L1TxUtilsConfig, l1TxUtilsConfigMappings } from './l1_tx_utils.js';
13
+ import { type L1TxUtilsConfig, l1TxUtilsConfigMappings } from './l1_tx_utils/index.js';
14
+
15
+ export type GenesisStateConfig = {
16
+ /** Whether to populate the genesis state with initial fee juice for the test accounts */
17
+ testAccounts: boolean;
18
+ /** Whether to populate the genesis state with initial fee juice for the sponsored FPC */
19
+ sponsoredFPC: boolean;
20
+ };
9
21
 
10
22
  export type L1ContractsConfig = {
11
23
  /** How many seconds an L1 slot lasts. */
@@ -16,33 +28,278 @@ export type L1ContractsConfig = {
16
28
  aztecEpochDuration: number;
17
29
  /** The target validator committee size. */
18
30
  aztecTargetCommitteeSize: number;
19
- /** The number of L2 slots that we can wait for a proof of an epoch to be produced. */
20
- aztecProofSubmissionWindow: number;
31
+ /** The number of epochs to lag behind the current epoch for validator selection. */
32
+ lagInEpochs: number;
33
+ /** The number of epochs after an epoch ends that proofs are still accepted. */
34
+ aztecProofSubmissionEpochs: number;
35
+ /** The deposit amount for a validator */
36
+ activationThreshold: bigint;
21
37
  /** The minimum stake for a validator. */
22
- minimumStake: bigint;
23
- /** The slashing quorum */
24
- slashingQuorum: number;
25
- /** The slashing round size */
26
- slashingRoundSize: number;
27
- /** Governance proposing quorum */
28
- governanceProposerQuorum: number;
38
+ ejectionThreshold: bigint;
39
+ /** The local ejection threshold for a validator. Stricter than ejectionThreshold but local to a specific rollup */
40
+ localEjectionThreshold: bigint;
41
+ /** The slashing quorum, i.e. how many slots must signal for the same payload in a round for it to be submittable to the Slasher (defaults to slashRoundSize / 2 + 1) */
42
+ slashingQuorum?: number;
43
+ /** The slashing round size, i.e. how many epochs are in a slashing round */
44
+ slashingRoundSizeInEpochs: number;
45
+ /** The slashing lifetime in rounds. I.e., if 1, round N must be submitted before round N + 2 */
46
+ slashingLifetimeInRounds: number;
47
+ /** The slashing execution delay in rounds. I.e., if 1, round N may not be submitted until round N + 2 */
48
+ slashingExecutionDelayInRounds: number;
49
+ /** The slashing vetoer. May blacklist a payload from being submitted. */
50
+ slashingVetoer: EthAddress;
51
+ /** How many slashing rounds back we slash (ie when slashing in round N, we slash for offenses committed during epochs of round N-offset) */
52
+ slashingOffsetInRounds: number;
53
+ /** How long slashing can be disabled for in seconds when vetoer disables it */
54
+ slashingDisableDuration: number;
55
+ /** Type of slasher proposer */
56
+ slasherFlavor: 'empire' | 'tally' | 'none';
57
+ /** Minimum amount that can be slashed in tally slashing */
58
+ slashAmountSmall: bigint;
59
+ /** Medium amount to slash in tally slashing */
60
+ slashAmountMedium: bigint;
61
+ /** Largest amount that can be slashed per round in tally slashing */
62
+ slashAmountLarge: bigint;
63
+ /** Governance proposing quorum (defaults to roundSize/2 + 1) */
64
+ governanceProposerQuorum?: number;
29
65
  /** Governance proposing round size */
30
66
  governanceProposerRoundSize: number;
67
+ /** The mana target for the rollup */
68
+ manaTarget: bigint;
69
+ /** The proving cost per mana */
70
+ provingCostPerMana: bigint;
71
+ /** The number of seconds to wait for an exit */
72
+ exitDelaySeconds: number;
31
73
  } & L1TxUtilsConfig;
32
74
 
33
75
  export const DefaultL1ContractsConfig = {
34
76
  ethereumSlotDuration: 12,
35
- aztecSlotDuration: 24,
36
- aztecEpochDuration: 16,
77
+ aztecSlotDuration: 36,
78
+ aztecEpochDuration: 32,
37
79
  aztecTargetCommitteeSize: 48,
38
- aztecProofSubmissionWindow: 31, // you have a full epoch to submit a proof after the epoch to prove ends
39
- minimumStake: BigInt(100e18),
40
- slashingQuorum: 6,
41
- slashingRoundSize: 10,
42
- governanceProposerQuorum: 6,
43
- governanceProposerRoundSize: 10,
80
+ lagInEpochs: 2,
81
+ aztecProofSubmissionEpochs: 1, // you have a full epoch to submit a proof after the epoch to prove ends
82
+ activationThreshold: 100n * 10n ** 18n,
83
+ ejectionThreshold: 50n * 10n ** 18n,
84
+ localEjectionThreshold: 98n * 10n ** 18n,
85
+ slashAmountSmall: 10n * 10n ** 18n,
86
+ slashAmountMedium: 20n * 10n ** 18n,
87
+ slashAmountLarge: 50n * 10n ** 18n,
88
+ slashingRoundSizeInEpochs: 4,
89
+ slashingLifetimeInRounds: 5,
90
+ slashingExecutionDelayInRounds: 0, // round N may be submitted in round N + 1
91
+ slashingVetoer: EthAddress.ZERO,
92
+ governanceProposerRoundSize: 300,
93
+ manaTarget: BigInt(1e10),
94
+ provingCostPerMana: BigInt(100),
95
+ exitDelaySeconds: 2 * 24 * 60 * 60,
96
+ slasherFlavor: 'tally' as const,
97
+ slashingOffsetInRounds: 2,
98
+ slashingDisableDuration: 5 * 24 * 60 * 60, // 5 days in seconds
44
99
  } satisfies L1ContractsConfig;
45
100
 
101
+ const LocalGovernanceConfiguration = {
102
+ proposeConfig: {
103
+ lockDelay: 60n * 60n * 24n * 30n,
104
+ lockAmount: 1n * 10n ** 24n,
105
+ },
106
+ votingDelay: 60n,
107
+ votingDuration: 60n * 60n,
108
+ executionDelay: 60n,
109
+ gracePeriod: 60n * 60n * 24n * 7n,
110
+ quorum: 1n * 10n ** 17n, // 10%
111
+ requiredYeaMargin: 4n * 10n ** 16n, // 4%
112
+ minimumVotes: 400n * 10n ** 18n,
113
+ };
114
+
115
+ const StagingPublicGovernanceConfiguration = {
116
+ proposeConfig: {
117
+ lockDelay: 60n * 60n * 24n * 30n,
118
+ lockAmount: DefaultL1ContractsConfig.activationThreshold * 100n,
119
+ },
120
+ votingDelay: 60n,
121
+ votingDuration: 60n * 60n,
122
+ executionDelay: 60n,
123
+ gracePeriod: 60n * 60n * 24n * 7n,
124
+ quorum: 3n * 10n ** 17n, // 30%
125
+ requiredYeaMargin: 4n * 10n ** 16n, // 4%
126
+ minimumVotes: DefaultL1ContractsConfig.ejectionThreshold * 200n, // >= 200 validators must vote
127
+ };
128
+
129
+ const TestnetGovernanceConfiguration = {
130
+ proposeConfig: {
131
+ lockDelay: 10n * 365n * 24n * 60n * 60n,
132
+ lockAmount: 1250n * 200_000n * 10n ** 18n,
133
+ },
134
+
135
+ votingDelay: 12n * 60n * 60n, // 12 hours
136
+ votingDuration: 1n * 24n * 60n * 60n, // 1 day
137
+ executionDelay: 12n * 60n * 60n, // 12 hours
138
+ gracePeriod: 1n * 24n * 60n * 60n, // 1 day
139
+ quorum: 2n * 10n ** 17n, // 20%
140
+ requiredYeaMargin: 1n * 10n ** 17n, // 10%
141
+ minimumVotes: 100n * 200_000n * 10n ** 18n,
142
+ };
143
+
144
+ const StagingIgnitionGovernanceConfiguration = {
145
+ proposeConfig: {
146
+ lockDelay: 10n * 365n * 24n * 60n * 60n,
147
+ lockAmount: 1250n * 200_000n * 10n ** 18n,
148
+ },
149
+
150
+ votingDelay: 7n * 24n * 60n * 60n,
151
+ votingDuration: 7n * 24n * 60n * 60n,
152
+ executionDelay: 30n * 24n * 60n * 60n,
153
+ gracePeriod: 7n * 24n * 60n * 60n,
154
+ quorum: 2n * 10n ** 17n, // 20%
155
+ requiredYeaMargin: 1n * 10n ** 17n, // 10%
156
+ minimumVotes: 1250n * 200_000n * 10n ** 18n,
157
+ };
158
+
159
+ const MainnetGovernanceConfiguration = {
160
+ proposeConfig: {
161
+ lockDelay: 90n * 24n * 60n * 60n,
162
+ lockAmount: 258_750_000n * 10n ** 18n,
163
+ },
164
+
165
+ votingDelay: 3n * 24n * 60n * 60n,
166
+ votingDuration: 7n * 24n * 60n * 60n,
167
+ executionDelay: 7n * 24n * 60n * 60n,
168
+ gracePeriod: 7n * 24n * 60n * 60n,
169
+ quorum: 2n * 10n ** 17n, // 20%
170
+ requiredYeaMargin: 33n * 10n ** 16n, // 33%
171
+ minimumVotes: 1000n * 200_000n * 10n ** 18n,
172
+ };
173
+
174
+ export const getGovernanceConfiguration = (networkName: NetworkNames) => {
175
+ switch (networkName) {
176
+ case 'local':
177
+ return LocalGovernanceConfiguration;
178
+ case 'next-net':
179
+ return LocalGovernanceConfiguration;
180
+ case 'devnet':
181
+ return LocalGovernanceConfiguration;
182
+ case 'staging-public':
183
+ return StagingPublicGovernanceConfiguration;
184
+ case 'testnet':
185
+ return TestnetGovernanceConfiguration;
186
+ case 'staging-ignition':
187
+ return StagingIgnitionGovernanceConfiguration;
188
+ case 'mainnet':
189
+ return MainnetGovernanceConfiguration;
190
+ default:
191
+ throw new Error(`Unrecognized network name: ${networkName}`);
192
+ }
193
+ };
194
+
195
+ // Making a default config here as we are only using it thought the deployment
196
+ // and do not expect to be using different setups, so having environment variables
197
+ // for it seems overkill
198
+
199
+ const DefaultRewardConfig = {
200
+ sequencerBps: 8000,
201
+ rewardDistributor: EthAddress.ZERO.toString(),
202
+ booster: EthAddress.ZERO.toString(),
203
+ blockReward: 500n * 10n ** 18n,
204
+ };
205
+
206
+ const MainnetRewardConfig = {
207
+ sequencerBps: 7_000,
208
+ rewardDistributor: EthAddress.ZERO.toString(),
209
+ booster: EthAddress.ZERO.toString(),
210
+ blockReward: 400n * 10n ** 18n,
211
+ };
212
+
213
+ export const getRewardConfig = (networkName: NetworkNames) => {
214
+ switch (networkName) {
215
+ case 'local':
216
+ case 'devnet':
217
+ case 'next-net':
218
+ case 'staging-public':
219
+ case 'testnet':
220
+ case 'staging-ignition':
221
+ return DefaultRewardConfig;
222
+ case 'mainnet':
223
+ return MainnetRewardConfig;
224
+ default:
225
+ throw new Error(`Unrecognized network name: ${networkName}`);
226
+ }
227
+ };
228
+
229
+ export const getRewardBoostConfig = () => {
230
+ // The reward configuration is specified with a precision of 1e5, and we use the same across
231
+ // all networks.
232
+ return {
233
+ increment: 125_000, // 1.25
234
+ maxScore: 15_000_000, // 150
235
+ a: 1_000, // 0.01
236
+ k: 1_000_000, // 10
237
+ minimum: 100_000, // 1
238
+ };
239
+ };
240
+
241
+ // Similar to the above, no need for environment variables for this.
242
+ const LocalEntryQueueConfig = {
243
+ bootstrapValidatorSetSize: 0n,
244
+ bootstrapFlushSize: 0n,
245
+ normalFlushSizeMin: 48n,
246
+ normalFlushSizeQuotient: 2n,
247
+ maxQueueFlushSize: 48n,
248
+ };
249
+
250
+ const StagingPublicEntryQueueConfig = {
251
+ bootstrapValidatorSetSize: 48n,
252
+ bootstrapFlushSize: 48n,
253
+ normalFlushSizeMin: 1n,
254
+ normalFlushSizeQuotient: 2475n,
255
+ maxQueueFlushSize: 32n, // Limited to 32 so flush cost are kept below 15M gas.
256
+ };
257
+
258
+ const TestnetEntryQueueConfig = {
259
+ bootstrapValidatorSetSize: 256n,
260
+ bootstrapFlushSize: 256n,
261
+ normalFlushSizeMin: 4n,
262
+ normalFlushSizeQuotient: 2048n,
263
+ maxQueueFlushSize: 8n,
264
+ };
265
+
266
+ const StagingIgnitionEntryQueueConfig = {
267
+ bootstrapValidatorSetSize: 48n,
268
+ bootstrapFlushSize: 48n,
269
+ normalFlushSizeMin: 1n,
270
+ normalFlushSizeQuotient: 2048n,
271
+ maxQueueFlushSize: 24n,
272
+ };
273
+
274
+ const MainnetEntryQueueConfig = {
275
+ bootstrapValidatorSetSize: 1_000n,
276
+ bootstrapFlushSize: 1_000n,
277
+ normalFlushSizeMin: 1n,
278
+ normalFlushSizeQuotient: 2_048n,
279
+ maxQueueFlushSize: 8n,
280
+ };
281
+
282
+ export const getEntryQueueConfig = (networkName: NetworkNames) => {
283
+ switch (networkName) {
284
+ case 'local':
285
+ return LocalEntryQueueConfig;
286
+ case 'next-net':
287
+ return LocalEntryQueueConfig;
288
+ case 'devnet':
289
+ return LocalEntryQueueConfig;
290
+ case 'staging-public':
291
+ return StagingPublicEntryQueueConfig;
292
+ case 'testnet':
293
+ return TestnetEntryQueueConfig;
294
+ case 'staging-ignition':
295
+ return StagingIgnitionEntryQueueConfig;
296
+ case 'mainnet':
297
+ return MainnetEntryQueueConfig;
298
+ default:
299
+ throw new Error(`Unrecognized network name: ${networkName}`);
300
+ }
301
+ };
302
+
46
303
  export const l1ContractsConfigMappings: ConfigMappingsType<L1ContractsConfig> = {
47
304
  ethereumSlotDuration: {
48
305
  env: 'ETHEREUM_SLOT_DURATION',
@@ -64,40 +321,336 @@ export const l1ContractsConfigMappings: ConfigMappingsType<L1ContractsConfig> =
64
321
  description: 'The target validator committee size.',
65
322
  ...numberConfigHelper(DefaultL1ContractsConfig.aztecTargetCommitteeSize),
66
323
  },
67
- aztecProofSubmissionWindow: {
68
- env: 'AZTEC_PROOF_SUBMISSION_WINDOW',
69
- description:
70
- 'The number of L2 slots that a proof for an epoch can be submitted in, starting from the beginning of the epoch.',
71
- ...numberConfigHelper(DefaultL1ContractsConfig.aztecProofSubmissionWindow),
324
+ lagInEpochs: {
325
+ env: 'AZTEC_LAG_IN_EPOCHS',
326
+ description: 'The number of epochs to lag behind the current epoch for validator selection.',
327
+ ...numberConfigHelper(DefaultL1ContractsConfig.lagInEpochs),
328
+ },
329
+ aztecProofSubmissionEpochs: {
330
+ env: 'AZTEC_PROOF_SUBMISSION_EPOCHS',
331
+ description: 'The number of epochs after an epoch ends that proofs are still accepted.',
332
+ ...numberConfigHelper(DefaultL1ContractsConfig.aztecProofSubmissionEpochs),
72
333
  },
73
- minimumStake: {
74
- env: 'AZTEC_MINIMUM_STAKE',
334
+ activationThreshold: {
335
+ env: 'AZTEC_ACTIVATION_THRESHOLD',
336
+ description: 'The deposit amount for a validator',
337
+ ...bigintConfigHelper(DefaultL1ContractsConfig.activationThreshold),
338
+ },
339
+ ejectionThreshold: {
340
+ env: 'AZTEC_EJECTION_THRESHOLD',
75
341
  description: 'The minimum stake for a validator.',
76
- ...bigintConfigHelper(DefaultL1ContractsConfig.minimumStake),
342
+ ...bigintConfigHelper(DefaultL1ContractsConfig.ejectionThreshold),
343
+ },
344
+ localEjectionThreshold: {
345
+ env: 'AZTEC_LOCAL_EJECTION_THRESHOLD',
346
+ description:
347
+ 'The local ejection threshold for a validator. Stricter than ejectionThreshold but local to a specific rollup',
348
+ ...bigintConfigHelper(DefaultL1ContractsConfig.localEjectionThreshold),
349
+ },
350
+ slashingOffsetInRounds: {
351
+ env: 'AZTEC_SLASHING_OFFSET_IN_ROUNDS',
352
+ description:
353
+ 'How many slashing rounds back we slash (ie when slashing in round N, we slash for offenses committed during epochs of round N-offset)',
354
+ ...numberConfigHelper(DefaultL1ContractsConfig.slashingOffsetInRounds),
355
+ },
356
+ slasherFlavor: {
357
+ env: 'AZTEC_SLASHER_FLAVOR',
358
+ description: 'Type of slasher proposer (empire, tally, or none)',
359
+ ...enumConfigHelper(['empire', 'tally', 'none'] as const, DefaultL1ContractsConfig.slasherFlavor),
360
+ },
361
+ slashAmountSmall: {
362
+ env: 'AZTEC_SLASH_AMOUNT_SMALL',
363
+ description: 'Small slashing amount for light offenses',
364
+ ...bigintConfigHelper(DefaultL1ContractsConfig.slashAmountSmall),
365
+ },
366
+ slashAmountMedium: {
367
+ env: 'AZTEC_SLASH_AMOUNT_MEDIUM',
368
+ description: 'Medium slashing amount for moderate offenses',
369
+ ...bigintConfigHelper(DefaultL1ContractsConfig.slashAmountMedium),
370
+ },
371
+ slashAmountLarge: {
372
+ env: 'AZTEC_SLASH_AMOUNT_LARGE',
373
+ description: 'Large slashing amount for severe offenses',
374
+ ...bigintConfigHelper(DefaultL1ContractsConfig.slashAmountLarge),
77
375
  },
78
376
  slashingQuorum: {
79
377
  env: 'AZTEC_SLASHING_QUORUM',
80
378
  description: 'The slashing quorum',
81
- ...numberConfigHelper(DefaultL1ContractsConfig.slashingQuorum),
379
+ ...optionalNumberConfigHelper(),
82
380
  },
83
- slashingRoundSize: {
84
- env: 'AZTEC_SLASHING_ROUND_SIZE',
381
+ slashingRoundSizeInEpochs: {
382
+ env: 'AZTEC_SLASHING_ROUND_SIZE_IN_EPOCHS',
85
383
  description: 'The slashing round size',
86
- ...numberConfigHelper(DefaultL1ContractsConfig.slashingRoundSize),
384
+ ...numberConfigHelper(DefaultL1ContractsConfig.slashingRoundSizeInEpochs),
385
+ },
386
+ slashingLifetimeInRounds: {
387
+ env: 'AZTEC_SLASHING_LIFETIME_IN_ROUNDS',
388
+ description: 'The slashing lifetime in rounds',
389
+ ...numberConfigHelper(DefaultL1ContractsConfig.slashingLifetimeInRounds),
390
+ },
391
+ slashingExecutionDelayInRounds: {
392
+ env: 'AZTEC_SLASHING_EXECUTION_DELAY_IN_ROUNDS',
393
+ description: 'The slashing execution delay in rounds',
394
+ ...numberConfigHelper(DefaultL1ContractsConfig.slashingExecutionDelayInRounds),
395
+ },
396
+ slashingVetoer: {
397
+ env: 'AZTEC_SLASHING_VETOER',
398
+ description: 'The slashing vetoer',
399
+ parseEnv: (val: string) => EthAddress.fromString(val),
400
+ defaultValue: DefaultL1ContractsConfig.slashingVetoer,
401
+ },
402
+ slashingDisableDuration: {
403
+ env: 'AZTEC_SLASHING_DISABLE_DURATION',
404
+ description: 'How long slashing can be disabled for in seconds when vetoer disables it',
405
+ ...numberConfigHelper(DefaultL1ContractsConfig.slashingDisableDuration),
87
406
  },
88
407
  governanceProposerQuorum: {
89
408
  env: 'AZTEC_GOVERNANCE_PROPOSER_QUORUM',
90
409
  description: 'The governance proposing quorum',
91
- ...numberConfigHelper(DefaultL1ContractsConfig.governanceProposerQuorum),
410
+ ...optionalNumberConfigHelper(),
92
411
  },
93
412
  governanceProposerRoundSize: {
94
413
  env: 'AZTEC_GOVERNANCE_PROPOSER_ROUND_SIZE',
95
414
  description: 'The governance proposing round size',
96
415
  ...numberConfigHelper(DefaultL1ContractsConfig.governanceProposerRoundSize),
97
416
  },
417
+ manaTarget: {
418
+ env: 'AZTEC_MANA_TARGET',
419
+ description: 'The mana target for the rollup',
420
+ ...bigintConfigHelper(DefaultL1ContractsConfig.manaTarget),
421
+ },
422
+ provingCostPerMana: {
423
+ env: 'AZTEC_PROVING_COST_PER_MANA',
424
+ description: 'The proving cost per mana',
425
+ ...bigintConfigHelper(DefaultL1ContractsConfig.provingCostPerMana),
426
+ },
427
+ exitDelaySeconds: {
428
+ env: 'AZTEC_EXIT_DELAY_SECONDS',
429
+ description: 'The delay before a validator can exit the set',
430
+ ...numberConfigHelper(DefaultL1ContractsConfig.exitDelaySeconds),
431
+ },
98
432
  ...l1TxUtilsConfigMappings,
99
433
  };
100
434
 
435
+ export const genesisStateConfigMappings: ConfigMappingsType<GenesisStateConfig> = {
436
+ testAccounts: {
437
+ env: 'TEST_ACCOUNTS',
438
+ description: 'Whether to populate the genesis state with initial fee juice for the test accounts.',
439
+ ...booleanConfigHelper(false),
440
+ },
441
+ sponsoredFPC: {
442
+ env: 'SPONSORED_FPC',
443
+ description: 'Whether to populate the genesis state with initial fee juice for the sponsored FPC.',
444
+ ...booleanConfigHelper(false),
445
+ },
446
+ };
447
+
101
448
  export function getL1ContractsConfigEnvVars(): L1ContractsConfig {
102
449
  return getConfigFromMappings(l1ContractsConfigMappings);
103
450
  }
451
+
452
+ export function getGenesisStateConfigEnvVars(): GenesisStateConfig {
453
+ return getConfigFromMappings(genesisStateConfigMappings);
454
+ }
455
+
456
+ /**
457
+ * Validates the L1 contracts configuration to ensure all requirements enforced by L1 contracts
458
+ * during construction are satisfied before deployment.
459
+ * Accumulates all validation errors and throws an exception listing them all if any are found.
460
+ */
461
+ export function validateConfig(config: Omit<L1ContractsConfig, keyof L1TxUtilsConfig>): void {
462
+ const errors: string[] = [];
463
+
464
+ // RollupCore constructor validation: normalFlushSizeMin > 0
465
+ // From: require(_config.stakingQueueConfig.normalFlushSizeMin > 0, Errors.Staking__InvalidStakingQueueConfig());
466
+ const entryQueueConfig = getEntryQueueConfig('testnet'); // Get config to check normalFlushSizeMin
467
+ if (entryQueueConfig.normalFlushSizeMin <= 0n) {
468
+ errors.push('normalFlushSizeMin must be greater than 0');
469
+ }
470
+
471
+ // TimeLib initialization validation: aztecSlotDuration should be a multiple of ethereumSlotDuration
472
+ // While not explicitly required in constructor, this is a common validation for time-based systems
473
+ if (config.aztecSlotDuration % config.ethereumSlotDuration !== 0) {
474
+ errors.push(
475
+ `aztecSlotDuration (${config.aztecSlotDuration}) must be a multiple of ethereumSlotDuration (${config.ethereumSlotDuration})`,
476
+ );
477
+ }
478
+
479
+ // EmpireBase constructor validations for governance/slashing proposers
480
+ // From: require(QUORUM_SIZE > ROUND_SIZE / 2, Errors.EmpireBase__InvalidQuorumAndRoundSize(QUORUM_SIZE, ROUND_SIZE));
481
+ const { governanceProposerQuorum, governanceProposerRoundSize } = config;
482
+ if (
483
+ governanceProposerQuorum !== undefined &&
484
+ governanceProposerQuorum <= Math.floor(governanceProposerRoundSize / 2)
485
+ ) {
486
+ errors.push(
487
+ `governanceProposerQuorum (${governanceProposerQuorum}) must be greater than half of governanceProposerRoundSize (${Math.floor(governanceProposerRoundSize / 2)})`,
488
+ );
489
+ }
490
+
491
+ // From: require(QUORUM_SIZE <= ROUND_SIZE, Errors.EmpireBase__QuorumCannotBeLargerThanRoundSize(QUORUM_SIZE, ROUND_SIZE));
492
+ if (governanceProposerQuorum !== undefined && governanceProposerQuorum > governanceProposerRoundSize) {
493
+ errors.push(
494
+ `governanceProposerQuorum (${governanceProposerQuorum}) cannot be larger than governanceProposerRoundSize (${governanceProposerRoundSize})`,
495
+ );
496
+ }
497
+
498
+ // Slashing quorum validations (similar to governance quorum)
499
+ const slashingRoundSize = config.slashingRoundSizeInEpochs * config.aztecEpochDuration;
500
+ const { slashingQuorum } = config;
501
+ if (slashingQuorum !== undefined && slashingQuorum <= Math.floor(slashingRoundSize / 2)) {
502
+ errors.push(
503
+ `slashingQuorum (${slashingQuorum}) must be greater than half of slashingRoundSizeInEpochs (${Math.floor(slashingRoundSize / 2)})`,
504
+ );
505
+ }
506
+
507
+ if (slashingQuorum !== undefined && slashingQuorum > slashingRoundSize) {
508
+ errors.push(
509
+ `slashingQuorum (${slashingQuorum}) cannot be larger than slashingRoundSizeInEpochs (${slashingRoundSize})`,
510
+ );
511
+ }
512
+
513
+ // EmpireBase and TallySlashingProposer lifetime and execution delay validation
514
+ // From: require(LIFETIME_IN_ROUNDS > EXECUTION_DELAY_IN_ROUNDS);
515
+ if (config.slashingLifetimeInRounds <= config.slashingExecutionDelayInRounds) {
516
+ errors.push(
517
+ `slashingLifetimeInRounds (${config.slashingLifetimeInRounds}) must be greater than slashingExecutionDelayInRounds (${config.slashingExecutionDelayInRounds})`,
518
+ );
519
+ }
520
+
521
+ // Staking asset validation: activationThreshold > ejectionThreshold
522
+ if (config.activationThreshold < config.ejectionThreshold) {
523
+ errors.push(
524
+ `activationThreshold (${config.activationThreshold}) must be greater than ejectionThreshold (${config.ejectionThreshold})`,
525
+ );
526
+ }
527
+
528
+ // TallySlashingProposer constructor validations
529
+ if (config.slasherFlavor === 'tally') {
530
+ validateTallySlasherConfig(config, errors);
531
+ }
532
+
533
+ // Epoch and slot duration validations
534
+ if (config.aztecSlotDuration <= 0) {
535
+ errors.push('aztecSlotDuration must be greater than 0');
536
+ }
537
+
538
+ if (config.ethereumSlotDuration <= 0) {
539
+ errors.push('ethereumSlotDuration must be greater than 0');
540
+ }
541
+
542
+ if (config.aztecEpochDuration <= 0) {
543
+ errors.push('aztecEpochDuration must be greater than 0');
544
+ }
545
+
546
+ // Committee size validation
547
+ if (config.aztecTargetCommitteeSize < 0) {
548
+ errors.push('aztecTargetCommitteeSize cannot be negative');
549
+ }
550
+
551
+ // Proof submission epochs validation
552
+ if (config.aztecProofSubmissionEpochs < 0) {
553
+ errors.push('aztecProofSubmissionEpochs cannot be negative');
554
+ }
555
+
556
+ // Exit delay validation
557
+ if (config.exitDelaySeconds < 0) {
558
+ errors.push('exitDelaySeconds cannot be negative');
559
+ }
560
+
561
+ // Mana validation
562
+ if (config.manaTarget < 0n) {
563
+ errors.push('manaTarget cannot be negative');
564
+ }
565
+
566
+ if (config.provingCostPerMana < 0n) {
567
+ errors.push('provingCostPerMana cannot be negative');
568
+ }
569
+
570
+ // If any errors were found, throw an exception with all of them
571
+ if (errors.length > 0) {
572
+ throw new Error(
573
+ `L1 contracts configuration validation failed with ${errors.length} error(s):\n${errors.map((error, index) => `${index + 1}. ${error}`).join('\n')}`,
574
+ );
575
+ }
576
+ }
577
+
578
+ function validateTallySlasherConfig(config: L1ContractsConfig, errors: string[]) {
579
+ if (config.slasherFlavor !== 'tally') {
580
+ return;
581
+ }
582
+
583
+ // From: require(SLASH_OFFSET_IN_ROUNDS > 0, Errors.TallySlashingProposer__SlashOffsetMustBeGreaterThanZero(...));
584
+ if (config.slashingOffsetInRounds <= 0) {
585
+ errors.push(`slashingOffsetInRounds (${config.slashingOffsetInRounds}) must be greater than 0`);
586
+ }
587
+
588
+ // From: require(ROUND_SIZE_IN_EPOCHS * _epochDuration == ROUND_SIZE, Errors.TallySlashingProposer__RoundSizeMustBeMultipleOfEpochDuration(...));
589
+ const roundSizeInSlots = config.slashingRoundSizeInEpochs * config.aztecEpochDuration;
590
+
591
+ // From: require(QUORUM > 0, Errors.TallySlashingProposer__QuorumMustBeGreaterThanZero());
592
+ const { slashingQuorum } = config;
593
+ if (slashingQuorum !== undefined && slashingQuorum <= 0) {
594
+ errors.push(`slashingQuorum (${slashingQuorum}) must be greater than 0`);
595
+ }
596
+
597
+ // From: require(ROUND_SIZE > 1, Errors.TallySlashingProposer__InvalidQuorumAndRoundSize(QUORUM, ROUND_SIZE));
598
+ if (roundSizeInSlots <= 1) {
599
+ errors.push(`slashing round size in slots (${roundSizeInSlots}) must be greater than 1`);
600
+ }
601
+
602
+ // From: require(_slashAmounts[0] <= _slashAmounts[1], Errors.TallySlashingProposer__InvalidSlashAmounts(_slashAmounts));
603
+ if (config.slashAmountSmall > config.slashAmountMedium) {
604
+ errors.push(
605
+ `slashAmountSmall (${config.slashAmountSmall}) must be less than or equal to slashAmountMedium (${config.slashAmountMedium})`,
606
+ );
607
+ }
608
+
609
+ // From: require(_slashAmounts[1] <= _slashAmounts[2], Errors.TallySlashingProposer__InvalidSlashAmounts(_slashAmounts));
610
+ if (config.slashAmountMedium > config.slashAmountLarge) {
611
+ errors.push(
612
+ `slashAmountMedium (${config.slashAmountMedium}) must be less than or equal to slashAmountLarge (${config.slashAmountLarge})`,
613
+ );
614
+ }
615
+
616
+ // From: require(LIFETIME_IN_ROUNDS < ROUNDABOUT_SIZE, Errors.TallySlashingProposer__LifetimeMustBeLessThanRoundabout(...));
617
+ const ROUNDABOUT_SIZE = 128; // Constant from TallySlashingProposer
618
+ if (config.slashingLifetimeInRounds >= ROUNDABOUT_SIZE) {
619
+ errors.push(`slashingLifetimeInRounds (${config.slashingLifetimeInRounds}) must be less than ${ROUNDABOUT_SIZE}`);
620
+ }
621
+
622
+ // From: require(ROUND_SIZE_IN_EPOCHS > 0, Errors.TallySlashingProposer__RoundSizeInEpochsMustBeGreaterThanZero(...));
623
+ if (config.slashingRoundSizeInEpochs <= 0) {
624
+ errors.push(`slashingRoundSizeInEpochs (${config.slashingRoundSizeInEpochs}) must be greater than 0`);
625
+ }
626
+
627
+ // From: require(ROUND_SIZE < MAX_ROUND_SIZE, Errors.TallySlashingProposer__RoundSizeTooLarge(ROUND_SIZE, MAX_ROUND_SIZE));
628
+ const MAX_ROUND_SIZE = 1024; // Constant from TallySlashingProposer
629
+ if (roundSizeInSlots >= MAX_ROUND_SIZE) {
630
+ errors.push(`slashing round size in slots (${roundSizeInSlots}) must be less than ${MAX_ROUND_SIZE}`);
631
+ }
632
+
633
+ // From: require(COMMITTEE_SIZE > 0, Errors.TallySlashingProposer__CommitteeSizeMustBeGreaterThanZero(COMMITTEE_SIZE));
634
+ if (config.aztecTargetCommitteeSize <= 0) {
635
+ errors.push(`aztecTargetCommitteeSize (${config.aztecTargetCommitteeSize}) must be greater than 0`);
636
+ }
637
+
638
+ // From: require(voteSize <= 128, Errors.TallySlashingProposer__VoteSizeTooBig(voteSize, 128));
639
+ // voteSize = COMMITTEE_SIZE * ROUND_SIZE_IN_EPOCHS / 4
640
+ const voteSize = (config.aztecTargetCommitteeSize * config.slashingRoundSizeInEpochs) / 4;
641
+ if (voteSize > 128) {
642
+ errors.push(`vote size (${voteSize}) must be <= 128 (committee size * round size in epochs / 4)`);
643
+ }
644
+
645
+ // From: require(COMMITTEE_SIZE * ROUND_SIZE_IN_EPOCHS % 4 == 0, Errors.TallySlashingProposer__InvalidCommitteeAndRoundSize(...));
646
+ if ((config.aztecTargetCommitteeSize * config.slashingRoundSizeInEpochs) % 4 !== 0) {
647
+ errors.push(
648
+ `aztecTargetCommitteeSize * slashingRoundSizeInEpochs (${config.aztecTargetCommitteeSize * config.slashingRoundSizeInEpochs}) must be divisible by 4`,
649
+ );
650
+ }
651
+
652
+ // Slashing offset validation: should be positive to allow proper slashing timing
653
+ if (config.slashingOffsetInRounds < 0) {
654
+ errors.push('slashingOffsetInRounds cannot be negative');
655
+ }
656
+ }