@aztec/end-to-end 0.0.1-commit.e558bd1c → 0.0.1-commit.e57c76e

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 (222) hide show
  1. package/README.md +27 -0
  2. package/dest/bench/client_flows/benchmark.d.ts +15 -1
  3. package/dest/bench/client_flows/benchmark.d.ts.map +1 -1
  4. package/dest/bench/client_flows/benchmark.js +17 -0
  5. package/dest/bench/client_flows/client_flows_benchmark.d.ts +3 -3
  6. package/dest/bench/client_flows/client_flows_benchmark.d.ts.map +1 -1
  7. package/dest/bench/client_flows/client_flows_benchmark.js +36 -39
  8. package/dest/bench/client_flows/config.d.ts +2 -2
  9. package/dest/bench/client_flows/config.d.ts.map +1 -1
  10. package/dest/bench/client_flows/config.js +18 -0
  11. package/dest/bench/utils.d.ts +1 -1
  12. package/dest/bench/utils.d.ts.map +1 -1
  13. package/dest/bench/utils.js +8 -3
  14. package/dest/e2e_blacklist_token_contract/blacklist_token_contract_test.d.ts +8 -5
  15. package/dest/e2e_blacklist_token_contract/blacklist_token_contract_test.d.ts.map +1 -1
  16. package/dest/e2e_blacklist_token_contract/blacklist_token_contract_test.js +36 -17
  17. package/dest/e2e_cross_chain_messaging/cross_chain_messaging_test.d.ts +16 -5
  18. package/dest/e2e_cross_chain_messaging/cross_chain_messaging_test.d.ts.map +1 -1
  19. package/dest/e2e_cross_chain_messaging/cross_chain_messaging_test.js +39 -7
  20. package/dest/e2e_deploy_contract/deploy_test.d.ts +4 -4
  21. package/dest/e2e_deploy_contract/deploy_test.d.ts.map +1 -1
  22. package/dest/e2e_deploy_contract/deploy_test.js +2 -1
  23. package/dest/e2e_epochs/epochs_test.d.ts +33 -8
  24. package/dest/e2e_epochs/epochs_test.d.ts.map +1 -1
  25. package/dest/e2e_epochs/epochs_test.js +143 -44
  26. package/dest/e2e_fees/fees_test.d.ts +6 -3
  27. package/dest/e2e_fees/fees_test.d.ts.map +1 -1
  28. package/dest/e2e_fees/fees_test.js +50 -17
  29. package/dest/e2e_nested_contract/nested_contract_test.d.ts +3 -3
  30. package/dest/e2e_nested_contract/nested_contract_test.d.ts.map +1 -1
  31. package/dest/e2e_nested_contract/nested_contract_test.js +6 -7
  32. package/dest/e2e_p2p/inactivity_slash_test.d.ts +1 -1
  33. package/dest/e2e_p2p/inactivity_slash_test.d.ts.map +1 -1
  34. package/dest/e2e_p2p/inactivity_slash_test.js +4 -3
  35. package/dest/e2e_p2p/p2p_network.d.ts +14 -12
  36. package/dest/e2e_p2p/p2p_network.d.ts.map +1 -1
  37. package/dest/e2e_p2p/p2p_network.js +70 -34
  38. package/dest/e2e_p2p/reqresp/utils.d.ts +3 -3
  39. package/dest/e2e_p2p/reqresp/utils.d.ts.map +1 -1
  40. package/dest/e2e_p2p/reqresp/utils.js +67 -14
  41. package/dest/e2e_p2p/shared.d.ts +37 -8
  42. package/dest/e2e_p2p/shared.d.ts.map +1 -1
  43. package/dest/e2e_p2p/shared.js +88 -51
  44. package/dest/e2e_storage_proof/fixtures/storage_proof_fetcher.d.ts +2 -0
  45. package/dest/e2e_storage_proof/fixtures/storage_proof_fetcher.d.ts.map +1 -0
  46. package/dest/e2e_storage_proof/fixtures/storage_proof_fetcher.js +184 -0
  47. package/dest/e2e_storage_proof/fixtures/storage_proof_fixture.d.ts +18 -0
  48. package/dest/e2e_storage_proof/fixtures/storage_proof_fixture.d.ts.map +1 -0
  49. package/dest/e2e_storage_proof/fixtures/storage_proof_fixture.js +120 -0
  50. package/dest/e2e_token_contract/token_contract_test.d.ts +6 -4
  51. package/dest/e2e_token_contract/token_contract_test.d.ts.map +1 -1
  52. package/dest/e2e_token_contract/token_contract_test.js +23 -11
  53. package/dest/fixtures/authwit_proxy.d.ts +15 -0
  54. package/dest/fixtures/authwit_proxy.d.ts.map +1 -0
  55. package/dest/fixtures/authwit_proxy.js +34 -0
  56. package/dest/fixtures/e2e_prover_test.d.ts +9 -8
  57. package/dest/fixtures/e2e_prover_test.d.ts.map +1 -1
  58. package/dest/fixtures/e2e_prover_test.js +39 -50
  59. package/dest/fixtures/elu_monitor.d.ts +21 -0
  60. package/dest/fixtures/elu_monitor.d.ts.map +1 -0
  61. package/dest/fixtures/elu_monitor.js +102 -0
  62. package/dest/fixtures/fixtures.d.ts +74 -1
  63. package/dest/fixtures/fixtures.d.ts.map +1 -1
  64. package/dest/fixtures/fixtures.js +71 -0
  65. package/dest/fixtures/get_bb_config.d.ts +1 -1
  66. package/dest/fixtures/get_bb_config.d.ts.map +1 -1
  67. package/dest/fixtures/get_bb_config.js +5 -5
  68. package/dest/fixtures/ha_setup.d.ts +2 -2
  69. package/dest/fixtures/ha_setup.d.ts.map +1 -1
  70. package/dest/fixtures/ha_setup.js +4 -2
  71. package/dest/fixtures/schnorr_hardcoded_account_contract.d.ts +25 -0
  72. package/dest/fixtures/schnorr_hardcoded_account_contract.d.ts.map +1 -0
  73. package/dest/fixtures/schnorr_hardcoded_account_contract.js +37 -0
  74. package/dest/fixtures/setup.d.ts +86 -32
  75. package/dest/fixtures/setup.d.ts.map +1 -1
  76. package/dest/fixtures/setup.js +209 -169
  77. package/dest/fixtures/setup_p2p_test.d.ts +14 -7
  78. package/dest/fixtures/setup_p2p_test.d.ts.map +1 -1
  79. package/dest/fixtures/setup_p2p_test.js +18 -15
  80. package/dest/fixtures/token_utils.d.ts +2 -2
  81. package/dest/fixtures/token_utils.d.ts.map +1 -1
  82. package/dest/fixtures/token_utils.js +5 -7
  83. package/dest/fixtures/utils.d.ts +2 -2
  84. package/dest/fixtures/utils.d.ts.map +1 -1
  85. package/dest/fixtures/utils.js +1 -1
  86. package/dest/forward-compatibility/wallet_rpc_client.d.ts +7 -0
  87. package/dest/forward-compatibility/wallet_rpc_client.d.ts.map +1 -0
  88. package/dest/forward-compatibility/wallet_rpc_client.js +15 -0
  89. package/dest/forward-compatibility/wallet_service.d.ts +3 -0
  90. package/dest/forward-compatibility/wallet_service.d.ts.map +1 -0
  91. package/dest/forward-compatibility/wallet_service.js +109 -0
  92. package/dest/install_legacy_contracts.d.cts +10 -0
  93. package/dest/install_legacy_contracts.d.cts.map +1 -0
  94. package/dest/legacy-jest-resolver.d.cts +3 -0
  95. package/dest/legacy-jest-resolver.d.cts.map +1 -0
  96. package/dest/shared/cross_chain_test_harness.d.ts +4 -2
  97. package/dest/shared/cross_chain_test_harness.d.ts.map +1 -1
  98. package/dest/shared/cross_chain_test_harness.js +22 -18
  99. package/dest/shared/gas_portal_test_harness.d.ts +8 -5
  100. package/dest/shared/gas_portal_test_harness.d.ts.map +1 -1
  101. package/dest/shared/gas_portal_test_harness.js +19 -10
  102. package/dest/shared/index.d.ts +2 -1
  103. package/dest/shared/index.d.ts.map +1 -1
  104. package/dest/shared/index.js +1 -0
  105. package/dest/shared/jest_setup.js +41 -1
  106. package/dest/shared/mock_state_view.d.ts +86 -0
  107. package/dest/shared/mock_state_view.d.ts.map +1 -0
  108. package/dest/shared/mock_state_view.js +186 -0
  109. package/dest/shared/submit-transactions.d.ts +2 -2
  110. package/dest/shared/submit-transactions.d.ts.map +1 -1
  111. package/dest/shared/submit-transactions.js +1 -1
  112. package/dest/shared/uniswap_l1_l2.d.ts +1 -1
  113. package/dest/shared/uniswap_l1_l2.d.ts.map +1 -1
  114. package/dest/shared/uniswap_l1_l2.js +56 -41
  115. package/dest/shared/wait_for_l1_to_l2_message.d.ts +13 -0
  116. package/dest/shared/wait_for_l1_to_l2_message.d.ts.map +1 -0
  117. package/dest/shared/wait_for_l1_to_l2_message.js +10 -0
  118. package/dest/simulators/lending_simulator.d.ts +10 -3
  119. package/dest/simulators/lending_simulator.d.ts.map +1 -1
  120. package/dest/simulators/lending_simulator.js +26 -14
  121. package/dest/simulators/token_simulator.d.ts +1 -1
  122. package/dest/simulators/token_simulator.d.ts.map +1 -1
  123. package/dest/simulators/token_simulator.js +3 -24
  124. package/dest/spartan/setup_test_wallets.d.ts +12 -3
  125. package/dest/spartan/setup_test_wallets.d.ts.map +1 -1
  126. package/dest/spartan/setup_test_wallets.js +108 -41
  127. package/dest/spartan/tx_metrics.d.ts +18 -4
  128. package/dest/spartan/tx_metrics.d.ts.map +1 -1
  129. package/dest/spartan/tx_metrics.js +74 -21
  130. package/dest/spartan/utils/bot.d.ts +3 -2
  131. package/dest/spartan/utils/bot.d.ts.map +1 -1
  132. package/dest/spartan/utils/bot.js +2 -1
  133. package/dest/spartan/utils/config.d.ts +10 -30
  134. package/dest/spartan/utils/config.d.ts.map +1 -1
  135. package/dest/spartan/utils/config.js +3 -1
  136. package/dest/spartan/utils/index.d.ts +4 -2
  137. package/dest/spartan/utils/index.d.ts.map +1 -1
  138. package/dest/spartan/utils/index.js +5 -1
  139. package/dest/spartan/utils/k8s.d.ts +3 -1
  140. package/dest/spartan/utils/k8s.d.ts.map +1 -1
  141. package/dest/spartan/utils/k8s.js +6 -0
  142. package/dest/spartan/utils/nodes.d.ts +4 -5
  143. package/dest/spartan/utils/nodes.d.ts.map +1 -1
  144. package/dest/spartan/utils/nodes.js +11 -11
  145. package/dest/spartan/utils/pod_logs.d.ts +25 -0
  146. package/dest/spartan/utils/pod_logs.d.ts.map +1 -0
  147. package/dest/spartan/utils/pod_logs.js +74 -0
  148. package/dest/test-wallet/test_wallet.d.ts +85 -0
  149. package/dest/test-wallet/test_wallet.d.ts.map +1 -0
  150. package/dest/test-wallet/test_wallet.js +273 -0
  151. package/dest/test-wallet/utils.d.ts +41 -0
  152. package/dest/test-wallet/utils.d.ts.map +1 -0
  153. package/dest/test-wallet/utils.js +66 -0
  154. package/dest/test-wallet/wallet_worker_script.d.ts +2 -0
  155. package/dest/test-wallet/wallet_worker_script.d.ts.map +1 -0
  156. package/dest/test-wallet/wallet_worker_script.js +53 -0
  157. package/dest/test-wallet/worker_wallet.d.ts +53 -0
  158. package/dest/test-wallet/worker_wallet.d.ts.map +1 -0
  159. package/dest/test-wallet/worker_wallet.js +155 -0
  160. package/dest/test-wallet/worker_wallet_schema.d.ts +160 -0
  161. package/dest/test-wallet/worker_wallet_schema.d.ts.map +1 -0
  162. package/dest/test-wallet/worker_wallet_schema.js +22 -0
  163. package/package.json +49 -45
  164. package/src/bench/client_flows/benchmark.ts +19 -0
  165. package/src/bench/client_flows/client_flows_benchmark.ts +64 -49
  166. package/src/bench/client_flows/config.ts +9 -1
  167. package/src/bench/utils.ts +10 -4
  168. package/src/e2e_blacklist_token_contract/blacklist_token_contract_test.ts +52 -25
  169. package/src/e2e_cross_chain_messaging/cross_chain_messaging_test.ts +53 -15
  170. package/src/e2e_deploy_contract/deploy_test.ts +6 -5
  171. package/src/e2e_epochs/epochs_test.ts +166 -68
  172. package/src/e2e_fees/bridging_race.notest.ts +1 -1
  173. package/src/e2e_fees/fees_test.ts +57 -32
  174. package/src/e2e_nested_contract/nested_contract_test.ts +10 -6
  175. package/src/e2e_p2p/inactivity_slash_test.ts +8 -7
  176. package/src/e2e_p2p/p2p_network.ts +93 -49
  177. package/src/e2e_p2p/reqresp/utils.ts +84 -17
  178. package/src/e2e_p2p/shared.ts +108 -65
  179. package/src/e2e_storage_proof/fixtures/storage_proof.json +915 -0
  180. package/src/e2e_storage_proof/fixtures/storage_proof_fetcher.ts +190 -0
  181. package/src/e2e_storage_proof/fixtures/storage_proof_fixture.ts +173 -0
  182. package/src/e2e_token_contract/token_contract_test.ts +38 -11
  183. package/src/fixtures/authwit_proxy.ts +54 -0
  184. package/src/fixtures/dumps/epoch_proof_result.json +1 -1
  185. package/src/fixtures/e2e_prover_test.ts +49 -56
  186. package/src/fixtures/elu_monitor.ts +126 -0
  187. package/src/fixtures/fixtures.ts +93 -0
  188. package/src/fixtures/get_bb_config.ts +7 -6
  189. package/src/fixtures/ha_setup.ts +7 -3
  190. package/src/fixtures/schnorr_hardcoded_account_contract.ts +49 -0
  191. package/src/fixtures/setup.ts +272 -233
  192. package/src/fixtures/setup_p2p_test.ts +21 -25
  193. package/src/fixtures/token_utils.ts +3 -3
  194. package/src/fixtures/utils.ts +2 -0
  195. package/src/forward-compatibility/wallet_rpc_client.ts +14 -0
  196. package/src/forward-compatibility/wallet_service.ts +104 -0
  197. package/src/guides/up_quick_start.sh +3 -5
  198. package/src/install_legacy_contracts.cjs +75 -0
  199. package/src/legacy-jest-resolver.cjs +112 -0
  200. package/src/shared/cross_chain_test_harness.ts +27 -13
  201. package/src/shared/gas_portal_test_harness.ts +21 -11
  202. package/src/shared/index.ts +1 -0
  203. package/src/shared/jest_setup.ts +51 -1
  204. package/src/shared/mock_state_view.ts +188 -0
  205. package/src/shared/submit-transactions.ts +3 -2
  206. package/src/shared/uniswap_l1_l2.ts +104 -53
  207. package/src/shared/wait_for_l1_to_l2_message.ts +23 -0
  208. package/src/simulators/lending_simulator.ts +32 -14
  209. package/src/simulators/token_simulator.ts +6 -30
  210. package/src/spartan/setup_test_wallets.ts +146 -35
  211. package/src/spartan/tx_metrics.ts +82 -24
  212. package/src/spartan/utils/bot.ts +4 -1
  213. package/src/spartan/utils/config.ts +2 -0
  214. package/src/spartan/utils/index.ts +7 -0
  215. package/src/spartan/utils/k8s.ts +8 -0
  216. package/src/spartan/utils/nodes.ts +17 -12
  217. package/src/spartan/utils/pod_logs.ts +99 -0
  218. package/src/test-wallet/test_wallet.ts +376 -0
  219. package/src/test-wallet/utils.ts +108 -0
  220. package/src/test-wallet/wallet_worker_script.ts +63 -0
  221. package/src/test-wallet/worker_wallet.ts +218 -0
  222. package/src/test-wallet/worker_wallet_schema.ts +13 -0
@@ -2,25 +2,26 @@ import type { AztecNodeService } from '@aztec/aztec-node';
2
2
  import { createLogger } from '@aztec/aztec.js/log';
3
3
  import { waitForTx } from '@aztec/aztec.js/node';
4
4
  import { Tx } from '@aztec/aztec.js/tx';
5
+ import { PROPOSER_PIPELINING_SLOT_OFFSET } from '@aztec/epoch-cache';
5
6
  import { RollupContract } from '@aztec/ethereum/contracts';
6
- import { SlotNumber } from '@aztec/foundation/branded-types';
7
+ import { CheckpointNumber, SlotNumber } from '@aztec/foundation/branded-types';
7
8
  import { timesAsync } from '@aztec/foundation/collection';
8
9
  import { retryUntil } from '@aztec/foundation/retry';
9
10
 
10
- import { jest } from '@jest/globals';
11
+ import { expect, jest } from '@jest/globals';
11
12
  import fs from 'fs';
12
13
  import os from 'os';
13
14
  import path from 'path';
14
15
 
15
- import { shouldCollectMetrics } from '../../fixtures/fixtures.js';
16
+ import { getBootNodeUdpPort, shouldCollectMetrics } from '../../fixtures/fixtures.js';
16
17
  import { createNodes } from '../../fixtures/setup_p2p_test.js';
17
- import { P2PNetworkTest, SHORTENED_BLOCK_TIME_CONFIG_NO_PRUNES, WAIT_FOR_TX_TIMEOUT } from '../p2p_network.js';
18
+ import { P2PNetworkTest } from '../p2p_network.js';
18
19
  import { prepareTransactions } from '../shared.js';
19
20
 
20
21
  // Don't set this to a higher value than 9 because each node will use a different L1 publisher account and anvil seeds
21
22
  export const NUM_VALIDATORS = 6;
22
- export const NUM_TXS_PER_NODE = 2;
23
- export const BOOT_NODE_UDP_PORT = 4500;
23
+ export const NUM_TXS_PER_NODE = 4;
24
+ export const BOOT_NODE_UDP_PORT = getBootNodeUdpPort();
24
25
 
25
26
  export const createReqrespDataDir = () => fs.mkdtempSync(path.join(os.tmpdir(), 'reqresp-'));
26
27
 
@@ -38,11 +39,20 @@ export async function createReqrespTest(options: ReqrespOptions = {}): Promise<P
38
39
  // To collect metrics - run in aztec-packages `docker compose --profile metrics up`
39
40
  metricsPort: shouldCollectMetrics(),
40
41
  initialConfig: {
41
- ...SHORTENED_BLOCK_TIME_CONFIG_NO_PRUNES,
42
- aztecSlotDuration: 24,
42
+ ethereumSlotDuration: 8,
43
+ aztecSlotDuration: 36,
44
+ blockDurationMs: 6000,
45
+ l1PublishingTime: 8,
46
+ minTxsPerBlock: 1,
47
+ maxTxsPerBlock: 2,
48
+ enforceTimeTable: true,
49
+ aztecProofSubmissionEpochs: 1024, // effectively do not reorg
43
50
  ...(disableStatusHandshake ? { p2pDisableStatusHandshake: true } : {}),
44
51
  listenAddress: '127.0.0.1',
45
52
  aztecEpochDuration: 64, // stable committee
53
+ // Pipelining: target-slot is one ahead of build-slot; inboxLag sources L1->L2
54
+ // messages from the previous checkpoint to avoid L1ToL2MessagesNotReadyError.
55
+ inboxLag: 2,
46
56
  },
47
57
  });
48
58
  await t.setup();
@@ -81,11 +91,11 @@ export async function runReqrespTxTest(params: {
81
91
 
82
92
  const nodes = await createNodes(
83
93
  aztecNodeConfig,
84
- t.ctx.dateProvider!,
94
+ t.ctx.dateProvider,
85
95
  t.bootstrapNodeEnr,
86
96
  NUM_VALIDATORS,
87
97
  BOOT_NODE_UDP_PORT,
88
- t.prefilledPublicData,
98
+ t.genesis,
89
99
  dataDir,
90
100
  shouldCollectMetrics(),
91
101
  );
@@ -95,7 +105,7 @@ export async function runReqrespTxTest(params: {
95
105
 
96
106
  await t.setupAccount();
97
107
 
98
- const targetBlockNumber = await t.ctx.aztecNodeService!.getBlockNumber();
108
+ const targetBlockNumber = await t.ctx.aztecNodeService.getBlockNumber();
99
109
  await retryUntil(
100
110
  async () => {
101
111
  const blockNumbers = await Promise.all(nodes.map(node => node.getBlockNumber()));
@@ -108,7 +118,7 @@ export async function runReqrespTxTest(params: {
108
118
 
109
119
  t.logger.info('Preparing transactions to send');
110
120
  const txBatches = await timesAsync(2, () =>
111
- prepareTransactions(t.logger, t.ctx.aztecNodeService!, NUM_TXS_PER_NODE, t.fundedAccount),
121
+ prepareTransactions(t.logger, t.ctx.aztecNodeService, NUM_TXS_PER_NODE, t.fundedAccount),
112
122
  );
113
123
 
114
124
  t.logger.info('Removing initial node');
@@ -116,10 +126,18 @@ export async function runReqrespTxTest(params: {
116
126
 
117
127
  t.logger.info('Starting fresh slot');
118
128
  const [timestamp] = await t.ctx.cheatCodes.rollup.advanceToNextSlot();
119
- t.ctx.dateProvider!.setTime(Number(timestamp) * 1000);
129
+ t.ctx.dateProvider.setTime(Number(timestamp) * 1000);
120
130
  const startSlotTimestamp = BigInt(timestamp);
121
131
 
122
- const { proposerIndexes, nodesToTurnOffTxGossip } = await getProposerIndexes(t, startSlotTimestamp);
132
+ // Under pipelining the active builder during wallclock slot S targets slot S+1, so
133
+ // we must address the proposer of S+1 (not S) for batch 0. Shift the proposer lookup
134
+ // window by the pipelining offset so we always send to the currently-building proposer.
135
+ const proposerSlotOffset = PROPOSER_PIPELINING_SLOT_OFFSET;
136
+ const { proposerIndexes, nodesToTurnOffTxGossip } = await getProposerIndexes(
137
+ t,
138
+ startSlotTimestamp,
139
+ proposerSlotOffset,
140
+ );
123
141
  t.logger.info(`Turning off tx gossip for nodes: ${nodesToTurnOffTxGossip.map(getNodePort)}`);
124
142
  t.logger.info(`Sending txs to proposer nodes: ${proposerIndexes.map(getNodePort)}`);
125
143
 
@@ -143,6 +161,13 @@ export async function runReqrespTxTest(params: {
143
161
  const submittedTxs = await Promise.all(
144
162
  txBatches.map(async (batch, batchIndex) => {
145
163
  const proposerNode = nodes[proposerIndexes[batchIndex]];
164
+ for (const tx of batch) {
165
+ t.logger.info(`Tx ${tx.getTxHash().toString()} base64: ${tx.toBuffer().toString('base64')}`);
166
+ }
167
+ const txHashes = batch.map(tx => tx.getTxHash().toString());
168
+ t.logger.info(
169
+ `Sending batch ${batchIndex} to proposer ${getNodePort(proposerIndexes[batchIndex])}: ${txHashes.join(', ')}`,
170
+ );
146
171
  await Promise.all(
147
172
  batch.map(async tx => {
148
173
  try {
@@ -157,12 +182,23 @@ export async function runReqrespTxTest(params: {
157
182
  }),
158
183
  );
159
184
 
185
+ // Log pool state per node after sending
186
+ for (let i = 0; i < NUM_VALIDATORS; i++) {
187
+ const count = await nodes[i].getPendingTxCount();
188
+ t.logger.info(`Node ${getNodePort(i)} pool has ${count} pending txs`);
189
+ }
190
+
191
+ // Use the test's own aztecSlotDuration (not the env default that p2p_network's
192
+ // WAIT_FOR_TX_TIMEOUT is derived from) so the timeout scales with this test's 36s slot.
193
+ // Under pipelining the round-trip is roughly build-slot + target-slot + L1 publish, so
194
+ // budget for >= 3 slots.
195
+ const waitForTxTimeout = t.ctx.aztecNodeConfig.aztecSlotDuration * 4.5;
160
196
  t.logger.info('Waiting for all transactions to be mined');
161
197
  await Promise.all(
162
198
  submittedTxs.flatMap((batch, batchIndex) =>
163
199
  batch.map(async (submittedTx, txIndex) => {
164
200
  t.logger.info(`Waiting for tx ${batchIndex}-${txIndex} ${submittedTx.txHash.toString()} to be mined`);
165
- await waitForTx(submittedTx.node, submittedTx.txHash, { timeout: WAIT_FOR_TX_TIMEOUT * 1.5 });
201
+ await waitForTx(submittedTx.node, submittedTx.txHash, { timeout: waitForTxTimeout });
166
202
  t.logger.info(`Tx ${batchIndex}-${txIndex} ${submittedTx.txHash.toString()} has been mined`);
167
203
  }),
168
204
  ),
@@ -170,10 +206,41 @@ export async function runReqrespTxTest(params: {
170
206
 
171
207
  t.logger.info('All transactions mined');
172
208
 
209
+ // Assert that multiple blocks were built for at least one slot
210
+ t.logger.info('Verifying multiple blocks for at least one checkpoint');
211
+ // Wait for L1 checkpoint sync, which may lag behind P2P block propagation.
212
+ const checkpoints = await retryUntil(
213
+ async () => {
214
+ const cps = await nodes[0].getCheckpoints(CheckpointNumber(1), 50, { includeBlocks: true });
215
+ return cps.length > 0 && cps.some(cp => (cp.blocks?.length ?? 0) >= 2) ? cps : undefined;
216
+ },
217
+ 'waiting for multi-block checkpoint to sync from L1',
218
+ 30,
219
+ 1,
220
+ );
221
+
222
+ let mbpsFound = false;
223
+ let expectedBlockNumber = checkpoints[0].blocks![0].number;
224
+
225
+ for (const published of checkpoints) {
226
+ const blocks = published.blocks!;
227
+ const blockCount = blocks.length;
228
+ mbpsFound = mbpsFound || blockCount >= 2;
229
+
230
+ for (let i = 0; i < blockCount; i++) {
231
+ const block = blocks[i];
232
+ expect(block.indexWithinCheckpoint).toBe(i);
233
+ expect(block.checkpointNumber).toBe(published.number);
234
+ expect(block.number).toBe(expectedBlockNumber);
235
+ expectedBlockNumber++;
236
+ }
237
+ }
238
+
239
+ expect(mbpsFound).toBe(true);
173
240
  return nodes;
174
241
  }
175
242
 
176
- async function getProposerIndexes(t: P2PNetworkTest, startSlotTimestamp: bigint) {
243
+ async function getProposerIndexes(t: P2PNetworkTest, startSlotTimestamp: bigint, slotOffset = 0) {
177
244
  // Get the nodes for the next set of slots
178
245
  const rollupContract = new RollupContract(
179
246
  t.ctx.deployL1ContractsValues.l1Client,
@@ -185,7 +252,7 @@ async function getProposerIndexes(t: P2PNetworkTest, startSlotTimestamp: bigint)
185
252
 
186
253
  const proposers = await Promise.all(
187
254
  Array.from({ length: 3 }, async (_, i) => {
188
- const slot = SlotNumber(startSlot + i);
255
+ const slot = SlotNumber(startSlot + slotOffset + i);
189
256
  const slotTimestamp = await rollupContract.getTimestampForSlot(slot);
190
257
  return await rollupContract.getProposerAt(slotTimestamp);
191
258
  }),
@@ -6,12 +6,9 @@ import { Fr } from '@aztec/aztec.js/fields';
6
6
  import type { Logger } from '@aztec/aztec.js/log';
7
7
  import { TxHash } from '@aztec/aztec.js/tx';
8
8
  import type { RollupCheatCodes } from '@aztec/aztec/testing';
9
- import type {
10
- EmpireSlashingProposerContract,
11
- RollupContract,
12
- TallySlashingProposerContract,
13
- } from '@aztec/ethereum/contracts';
14
- import { EpochNumber } from '@aztec/foundation/branded-types';
9
+ import type { EpochCacheInterface } from '@aztec/epoch-cache';
10
+ import type { RollupContract, SlashingProposerContract } from '@aztec/ethereum/contracts';
11
+ import { EpochNumber, SlotNumber } from '@aztec/foundation/branded-types';
15
12
  import { timesAsync, unique } from '@aztec/foundation/collection';
16
13
  import { EthAddress } from '@aztec/foundation/eth-address';
17
14
  import { retryUntil } from '@aztec/foundation/retry';
@@ -21,10 +18,11 @@ import { TestContract, TestContractArtifact } from '@aztec/noir-test-contracts.j
21
18
  import { getPXEConfig, getPXEConfig as getRpcConfig } from '@aztec/pxe/server';
22
19
  import { getRoundForOffense } from '@aztec/slasher';
23
20
  import type { AztecNodeAdmin } from '@aztec/stdlib/interfaces/client';
24
- import type { SlashFactoryContract } from '@aztec/stdlib/l1-contracts';
25
- import { type ProvenTx, TestWallet, proveInteraction } from '@aztec/test-wallet/server';
26
21
 
22
+ import { SchnorrHardcodedKeyAccountContract } from '../fixtures/schnorr_hardcoded_account_contract.js';
27
23
  import { submitTxsTo } from '../shared/submit-transactions.js';
24
+ import { TestWallet } from '../test-wallet/test_wallet.js';
25
+ import { type ProvenTx, proveInteraction } from '../test-wallet/utils.js';
28
26
 
29
27
  // submits a set of transactions to the provided Private eXecution Environment (PXE)
30
28
  export const submitComplexTxsTo = async (
@@ -40,7 +38,7 @@ export const submitComplexTxsTo = async (
40
38
  const spamCount = 15;
41
39
  for (let i = 0; i < numTxs; i++) {
42
40
  const method = spamContract.methods.spam(seed + BigInt(i * spamCount), spamCount, !!opts.callPublic);
43
- const txHash = await method.send({ from, wait: NO_WAIT });
41
+ const { txHash } = await method.send({ from, wait: NO_WAIT });
44
42
  logger.info(`Tx sent with hash ${txHash.toString()}`);
45
43
  txs.push(txHash);
46
44
  }
@@ -58,10 +56,17 @@ export const submitTransactions = async (
58
56
  rpcConfig.proverEnabled = false;
59
57
  const wallet = await TestWallet.create(
60
58
  node,
61
- { ...getPXEConfig(), proverEnabled: false },
59
+ // Use checkpointed chain tip to avoid anchoring on provisional blocks that the archiver can prune
60
+ // when their slot ends without a checkpoint landing on L1.
61
+ { ...getPXEConfig(), proverEnabled: false, syncChainTip: 'checkpointed' },
62
62
  { loggerActorLabel: 'pxe-tx' },
63
63
  );
64
- const fundedAccountManager = await wallet.createSchnorrAccount(fundedAccount.secret, fundedAccount.salt);
64
+ const contract = new SchnorrHardcodedKeyAccountContract();
65
+ const fundedAccountManager = await wallet.createAccount({
66
+ secret: fundedAccount.secret,
67
+ salt: fundedAccount.salt,
68
+ contract,
69
+ });
65
70
  return submitTxsTo(wallet, fundedAccountManager.address, numTxs, logger);
66
71
  };
67
72
 
@@ -76,10 +81,15 @@ export async function prepareTransactions(
76
81
 
77
82
  const wallet = await TestWallet.create(
78
83
  node,
79
- { ...getPXEConfig(), proverEnabled: false },
84
+ { ...getPXEConfig(), proverEnabled: false, syncChainTip: 'checkpointed' },
80
85
  { loggerActorLabel: 'pxe-tx' },
81
86
  );
82
- const fundedAccountManager = await wallet.createSchnorrAccount(fundedAccount.secret, fundedAccount.salt);
87
+ const accountContract = new SchnorrHardcodedKeyAccountContract();
88
+ const fundedAccountManager = await wallet.createAccount({
89
+ secret: fundedAccount.secret,
90
+ salt: fundedAccount.salt,
91
+ contract: accountContract,
92
+ });
83
93
 
84
94
  const testContractInstance = await getContractInstanceFromInstantiationParams(TestContractArtifact, {
85
95
  salt: Fr.random(),
@@ -97,7 +107,7 @@ export async function prepareTransactions(
97
107
  }
98
108
 
99
109
  export function awaitProposalExecution(
100
- slashingProposer: EmpireSlashingProposerContract | TallySlashingProposerContract,
110
+ slashingProposer: SlashingProposerContract,
101
111
  timeoutSeconds: number,
102
112
  logger: Logger,
103
113
  ): Promise<bigint> {
@@ -107,24 +117,12 @@ export function awaitProposalExecution(
107
117
  reject(new Error(`Timeout waiting for proposal execution after ${timeoutSeconds}s`));
108
118
  }, timeoutSeconds * 1000);
109
119
 
110
- if (slashingProposer.type === 'empire') {
111
- const unwatch = slashingProposer.listenToPayloadSubmitted(args => {
112
- logger.warn(`Proposal ${args.payload} from round ${args.round} executed`);
113
- clearTimeout(timeout);
114
- unwatch();
115
- resolve(args.round);
116
- });
117
- } else if (slashingProposer.type === 'tally') {
118
- const unwatch = slashingProposer.listenToRoundExecuted(args => {
119
- logger.warn(`Slash from round ${args.round} executed`);
120
- clearTimeout(timeout);
121
- unwatch();
122
- resolve(args.round);
123
- });
124
- } else {
120
+ const unwatch = slashingProposer.listenToRoundExecuted(args => {
121
+ logger.warn(`Slash from round ${args.round} executed`);
125
122
  clearTimeout(timeout);
126
- reject(new Error(`Unknown slashing proposer type: ${(slashingProposer as any).type}`));
127
- }
123
+ unwatch();
124
+ resolve(args.round);
125
+ });
128
126
  });
129
127
  }
130
128
 
@@ -149,6 +147,72 @@ export async function awaitCommitteeExists({
149
147
  return committee!.map(c => c.toString() as `0x${string}`);
150
148
  }
151
149
 
150
+ /**
151
+ * Advance epochs until we find one where the target proposer is selected for a slot at least
152
+ * `warmupSlots` into the epoch, then stop one epoch before it. This leaves time for the caller to
153
+ * start sequencers before warping to the target epoch, avoiding the race where the target epoch
154
+ * passes before sequencers are ready.
155
+ *
156
+ * The first `warmupSlots` slots of the epoch are skipped on purpose. Callers warp to one slot
157
+ * before the target epoch and, under proposer pipelining, the proposer begins building one slot
158
+ * before its proposal slot. If the proposer were in the first slot of the epoch, that build would
159
+ * begin at the exact instant of the warp, leaving the freshly-started sequencer no warm-up margin;
160
+ * it then serializes its (often AVM-heavy) proposal past the slot boundary and honest receivers
161
+ * reject it as late. Picking a slot at least `warmupSlots` into the epoch guarantees that many full
162
+ * slots of wall-clock between the warp and the start of the proposer's build.
163
+ *
164
+ * Returns the target epoch and the concrete target slot so the caller can warp to it after starting
165
+ * sequencers.
166
+ */
167
+ export async function advanceToEpochBeforeProposer({
168
+ epochCache,
169
+ cheatCodes,
170
+ targetProposer,
171
+ logger,
172
+ maxAttempts = 20,
173
+ warmupSlots = 1,
174
+ }: {
175
+ epochCache: EpochCacheInterface;
176
+ cheatCodes: RollupCheatCodes;
177
+ targetProposer: EthAddress;
178
+ logger: Logger;
179
+ maxAttempts?: number;
180
+ warmupSlots?: number;
181
+ }): Promise<{ targetEpoch: EpochNumber; targetSlot: SlotNumber }> {
182
+ const { epochDuration } = await cheatCodes.getConfig();
183
+
184
+ for (let attempt = 0; attempt < maxAttempts; attempt++) {
185
+ const currentEpoch = await cheatCodes.getEpoch();
186
+ // Check the NEXT epoch's slots so we stay one epoch before the target,
187
+ // giving the caller time to start sequencers before the target epoch arrives.
188
+ const nextEpoch = Number(currentEpoch) + 1;
189
+ const epochStartSlot = nextEpoch * Number(epochDuration);
190
+ // Skip the first `warmupSlots` slots so the caller keeps a warm-up margin after warping to one
191
+ // slot before the epoch (see the doc comment above).
192
+ const startSlot = epochStartSlot + warmupSlots;
193
+ const endSlot = epochStartSlot + Number(epochDuration);
194
+
195
+ logger.info(
196
+ `Checking next epoch ${nextEpoch} (slots ${startSlot}-${endSlot - 1}) for proposer ${targetProposer} (current epoch: ${currentEpoch})`,
197
+ );
198
+
199
+ for (let s = startSlot; s < endSlot; s++) {
200
+ const proposer = await epochCache.getProposerAttesterAddressInSlot(SlotNumber(s));
201
+ if (proposer && proposer.equals(targetProposer)) {
202
+ logger.warn(
203
+ `Found target proposer ${targetProposer} in slot ${s} of epoch ${nextEpoch}. Staying at epoch ${currentEpoch} to allow sequencer startup.`,
204
+ );
205
+ return { targetEpoch: EpochNumber(nextEpoch), targetSlot: SlotNumber(s) };
206
+ }
207
+ }
208
+
209
+ logger.info(`Target proposer not found in epoch ${nextEpoch}, advancing to next epoch`);
210
+ await cheatCodes.advanceToNextEpoch();
211
+ }
212
+
213
+ throw new Error(`Target proposer ${targetProposer} not found in any slot after ${maxAttempts} epoch attempts`);
214
+ }
215
+
152
216
  export async function awaitOffenseDetected({
153
217
  logger,
154
218
  nodeAdmin,
@@ -191,7 +255,6 @@ export async function awaitCommitteeKicked({
191
255
  rollup,
192
256
  cheatCodes,
193
257
  committee,
194
- slashFactory,
195
258
  slashingProposer,
196
259
  slashingRoundSize,
197
260
  aztecSlotDuration,
@@ -202,8 +265,7 @@ export async function awaitCommitteeKicked({
202
265
  rollup: RollupContract;
203
266
  cheatCodes: RollupCheatCodes;
204
267
  committee: readonly `0x${string}`[];
205
- slashFactory: SlashFactoryContract;
206
- slashingProposer: EmpireSlashingProposerContract | TallySlashingProposerContract | undefined;
268
+ slashingProposer: SlashingProposerContract | undefined;
207
269
  slashingRoundSize: number;
208
270
  aztecSlotDuration: number;
209
271
  aztecEpochDuration: number;
@@ -216,36 +278,14 @@ export async function awaitCommitteeKicked({
216
278
 
217
279
  await cheatCodes.debugRollup();
218
280
 
219
- if (slashingProposer.type === 'empire') {
220
- // Await for the slash payload to be created if empire (no payload is created on tally until execution time)
221
- const targetEpoch = EpochNumber((await cheatCodes.getEpoch()) + (await rollup.getLagInEpochsForValidatorSet()) + 1);
222
- logger.info(`Advancing to epoch ${targetEpoch} so we start slashing`);
223
- await cheatCodes.advanceToEpoch(targetEpoch);
224
-
225
- const slashPayloadEvents = await retryUntil(
226
- async () => {
227
- const events = await slashFactory.getSlashPayloadCreatedEvents();
228
- return events.length > 0 ? events : undefined;
229
- },
230
- 'slash payload created',
231
- 120,
232
- 1,
233
- );
234
- expect(slashPayloadEvents.length).toBe(1);
235
- // The uniqueness check is needed since a validator may be slashed more than once on the same round (eg because they let two epochs be pruned)
236
- expect(unique(slashPayloadEvents[0].slashes.map(slash => slash.validator.toString()))).toHaveLength(
237
- committee.length,
238
- );
239
- } else {
240
- // Use the slash offset to ensure we are in the right epoch for tally
241
- const slashOffsetInRounds = await slashingProposer.getSlashOffsetInRounds();
242
- const slashingRoundSizeInEpochs = slashingRoundSize / aztecEpochDuration;
243
- const slashingOffsetInEpochs = Number(slashOffsetInRounds) * slashingRoundSizeInEpochs;
244
- const firstEpochInOffenseRound = offenseEpoch - (offenseEpoch % slashingRoundSizeInEpochs);
245
- const targetEpoch = firstEpochInOffenseRound + slashingOffsetInEpochs;
246
- logger.info(`Advancing to epoch ${targetEpoch} so we start slashing`);
247
- await cheatCodes.advanceToEpoch(EpochNumber(targetEpoch), { offset: -aztecSlotDuration / 2 });
248
- }
281
+ // Use the slash offset to ensure we are in the right epoch for tally
282
+ const slashOffsetInRounds = await slashingProposer.getSlashOffsetInRounds();
283
+ const slashingRoundSizeInEpochs = slashingRoundSize / aztecEpochDuration;
284
+ const slashingOffsetInEpochs = Number(slashOffsetInRounds) * slashingRoundSizeInEpochs;
285
+ const firstEpochInOffenseRound = offenseEpoch - (offenseEpoch % slashingRoundSizeInEpochs);
286
+ const targetEpoch = firstEpochInOffenseRound + slashingOffsetInEpochs;
287
+ logger.info(`Advancing to epoch ${targetEpoch} so we start slashing`);
288
+ await cheatCodes.advanceToEpoch(EpochNumber(targetEpoch), { offset: -aztecSlotDuration / 2 });
249
289
 
250
290
  const attestersPre = await rollup.getAttesters();
251
291
  expect(attestersPre.length).toBe(committee.length);
@@ -255,7 +295,10 @@ export async function awaitCommitteeKicked({
255
295
  expect(attesterInfo.status).toEqual(1); // Validating
256
296
  }
257
297
 
258
- const timeout = slashingRoundSize * 2 * aztecSlotDuration + 30;
298
+ // Allow up to four round-lengths so that under proposer pipelining, where individual rounds
299
+ // sometimes fail to gather quorum because parts of the committee miss votes due to chain-state
300
+ // races, we still see a later round execute the slash.
301
+ const timeout = slashingRoundSize * 4 * aztecSlotDuration + 30;
259
302
  logger.info(`Waiting for slash to be executed (timeout ${timeout}s)`);
260
303
  await awaitProposalExecution(slashingProposer, timeout, logger);
261
304