@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.
- package/README.md +27 -0
- package/dest/bench/client_flows/benchmark.d.ts +15 -1
- package/dest/bench/client_flows/benchmark.d.ts.map +1 -1
- package/dest/bench/client_flows/benchmark.js +17 -0
- package/dest/bench/client_flows/client_flows_benchmark.d.ts +3 -3
- package/dest/bench/client_flows/client_flows_benchmark.d.ts.map +1 -1
- package/dest/bench/client_flows/client_flows_benchmark.js +36 -39
- package/dest/bench/client_flows/config.d.ts +2 -2
- package/dest/bench/client_flows/config.d.ts.map +1 -1
- package/dest/bench/client_flows/config.js +18 -0
- package/dest/bench/utils.d.ts +1 -1
- package/dest/bench/utils.d.ts.map +1 -1
- package/dest/bench/utils.js +8 -3
- package/dest/e2e_blacklist_token_contract/blacklist_token_contract_test.d.ts +8 -5
- package/dest/e2e_blacklist_token_contract/blacklist_token_contract_test.d.ts.map +1 -1
- package/dest/e2e_blacklist_token_contract/blacklist_token_contract_test.js +36 -17
- package/dest/e2e_cross_chain_messaging/cross_chain_messaging_test.d.ts +16 -5
- package/dest/e2e_cross_chain_messaging/cross_chain_messaging_test.d.ts.map +1 -1
- package/dest/e2e_cross_chain_messaging/cross_chain_messaging_test.js +39 -7
- package/dest/e2e_deploy_contract/deploy_test.d.ts +4 -4
- package/dest/e2e_deploy_contract/deploy_test.d.ts.map +1 -1
- package/dest/e2e_deploy_contract/deploy_test.js +2 -1
- package/dest/e2e_epochs/epochs_test.d.ts +33 -8
- package/dest/e2e_epochs/epochs_test.d.ts.map +1 -1
- package/dest/e2e_epochs/epochs_test.js +143 -44
- package/dest/e2e_fees/fees_test.d.ts +6 -3
- package/dest/e2e_fees/fees_test.d.ts.map +1 -1
- package/dest/e2e_fees/fees_test.js +50 -17
- package/dest/e2e_nested_contract/nested_contract_test.d.ts +3 -3
- package/dest/e2e_nested_contract/nested_contract_test.d.ts.map +1 -1
- package/dest/e2e_nested_contract/nested_contract_test.js +6 -7
- package/dest/e2e_p2p/inactivity_slash_test.d.ts +1 -1
- package/dest/e2e_p2p/inactivity_slash_test.d.ts.map +1 -1
- package/dest/e2e_p2p/inactivity_slash_test.js +4 -3
- package/dest/e2e_p2p/p2p_network.d.ts +14 -12
- package/dest/e2e_p2p/p2p_network.d.ts.map +1 -1
- package/dest/e2e_p2p/p2p_network.js +70 -34
- package/dest/e2e_p2p/reqresp/utils.d.ts +3 -3
- package/dest/e2e_p2p/reqresp/utils.d.ts.map +1 -1
- package/dest/e2e_p2p/reqresp/utils.js +67 -14
- package/dest/e2e_p2p/shared.d.ts +37 -8
- package/dest/e2e_p2p/shared.d.ts.map +1 -1
- package/dest/e2e_p2p/shared.js +88 -51
- package/dest/e2e_storage_proof/fixtures/storage_proof_fetcher.d.ts +2 -0
- package/dest/e2e_storage_proof/fixtures/storage_proof_fetcher.d.ts.map +1 -0
- package/dest/e2e_storage_proof/fixtures/storage_proof_fetcher.js +184 -0
- package/dest/e2e_storage_proof/fixtures/storage_proof_fixture.d.ts +18 -0
- package/dest/e2e_storage_proof/fixtures/storage_proof_fixture.d.ts.map +1 -0
- package/dest/e2e_storage_proof/fixtures/storage_proof_fixture.js +120 -0
- package/dest/e2e_token_contract/token_contract_test.d.ts +6 -4
- package/dest/e2e_token_contract/token_contract_test.d.ts.map +1 -1
- package/dest/e2e_token_contract/token_contract_test.js +23 -11
- package/dest/fixtures/authwit_proxy.d.ts +15 -0
- package/dest/fixtures/authwit_proxy.d.ts.map +1 -0
- package/dest/fixtures/authwit_proxy.js +34 -0
- package/dest/fixtures/e2e_prover_test.d.ts +9 -8
- package/dest/fixtures/e2e_prover_test.d.ts.map +1 -1
- package/dest/fixtures/e2e_prover_test.js +39 -50
- package/dest/fixtures/elu_monitor.d.ts +21 -0
- package/dest/fixtures/elu_monitor.d.ts.map +1 -0
- package/dest/fixtures/elu_monitor.js +102 -0
- package/dest/fixtures/fixtures.d.ts +74 -1
- package/dest/fixtures/fixtures.d.ts.map +1 -1
- package/dest/fixtures/fixtures.js +71 -0
- package/dest/fixtures/get_bb_config.d.ts +1 -1
- package/dest/fixtures/get_bb_config.d.ts.map +1 -1
- package/dest/fixtures/get_bb_config.js +5 -5
- package/dest/fixtures/ha_setup.d.ts +2 -2
- package/dest/fixtures/ha_setup.d.ts.map +1 -1
- package/dest/fixtures/ha_setup.js +4 -2
- package/dest/fixtures/schnorr_hardcoded_account_contract.d.ts +25 -0
- package/dest/fixtures/schnorr_hardcoded_account_contract.d.ts.map +1 -0
- package/dest/fixtures/schnorr_hardcoded_account_contract.js +37 -0
- package/dest/fixtures/setup.d.ts +86 -32
- package/dest/fixtures/setup.d.ts.map +1 -1
- package/dest/fixtures/setup.js +209 -169
- package/dest/fixtures/setup_p2p_test.d.ts +14 -7
- package/dest/fixtures/setup_p2p_test.d.ts.map +1 -1
- package/dest/fixtures/setup_p2p_test.js +18 -15
- package/dest/fixtures/token_utils.d.ts +2 -2
- package/dest/fixtures/token_utils.d.ts.map +1 -1
- package/dest/fixtures/token_utils.js +5 -7
- package/dest/fixtures/utils.d.ts +2 -2
- package/dest/fixtures/utils.d.ts.map +1 -1
- package/dest/fixtures/utils.js +1 -1
- package/dest/forward-compatibility/wallet_rpc_client.d.ts +7 -0
- package/dest/forward-compatibility/wallet_rpc_client.d.ts.map +1 -0
- package/dest/forward-compatibility/wallet_rpc_client.js +15 -0
- package/dest/forward-compatibility/wallet_service.d.ts +3 -0
- package/dest/forward-compatibility/wallet_service.d.ts.map +1 -0
- package/dest/forward-compatibility/wallet_service.js +109 -0
- package/dest/install_legacy_contracts.d.cts +10 -0
- package/dest/install_legacy_contracts.d.cts.map +1 -0
- package/dest/legacy-jest-resolver.d.cts +3 -0
- package/dest/legacy-jest-resolver.d.cts.map +1 -0
- package/dest/shared/cross_chain_test_harness.d.ts +4 -2
- package/dest/shared/cross_chain_test_harness.d.ts.map +1 -1
- package/dest/shared/cross_chain_test_harness.js +22 -18
- package/dest/shared/gas_portal_test_harness.d.ts +8 -5
- package/dest/shared/gas_portal_test_harness.d.ts.map +1 -1
- package/dest/shared/gas_portal_test_harness.js +19 -10
- package/dest/shared/index.d.ts +2 -1
- package/dest/shared/index.d.ts.map +1 -1
- package/dest/shared/index.js +1 -0
- package/dest/shared/jest_setup.js +41 -1
- package/dest/shared/mock_state_view.d.ts +86 -0
- package/dest/shared/mock_state_view.d.ts.map +1 -0
- package/dest/shared/mock_state_view.js +186 -0
- package/dest/shared/submit-transactions.d.ts +2 -2
- package/dest/shared/submit-transactions.d.ts.map +1 -1
- package/dest/shared/submit-transactions.js +1 -1
- package/dest/shared/uniswap_l1_l2.d.ts +1 -1
- package/dest/shared/uniswap_l1_l2.d.ts.map +1 -1
- package/dest/shared/uniswap_l1_l2.js +56 -41
- package/dest/shared/wait_for_l1_to_l2_message.d.ts +13 -0
- package/dest/shared/wait_for_l1_to_l2_message.d.ts.map +1 -0
- package/dest/shared/wait_for_l1_to_l2_message.js +10 -0
- package/dest/simulators/lending_simulator.d.ts +10 -3
- package/dest/simulators/lending_simulator.d.ts.map +1 -1
- package/dest/simulators/lending_simulator.js +26 -14
- package/dest/simulators/token_simulator.d.ts +1 -1
- package/dest/simulators/token_simulator.d.ts.map +1 -1
- package/dest/simulators/token_simulator.js +3 -24
- package/dest/spartan/setup_test_wallets.d.ts +12 -3
- package/dest/spartan/setup_test_wallets.d.ts.map +1 -1
- package/dest/spartan/setup_test_wallets.js +108 -41
- package/dest/spartan/tx_metrics.d.ts +18 -4
- package/dest/spartan/tx_metrics.d.ts.map +1 -1
- package/dest/spartan/tx_metrics.js +74 -21
- package/dest/spartan/utils/bot.d.ts +3 -2
- package/dest/spartan/utils/bot.d.ts.map +1 -1
- package/dest/spartan/utils/bot.js +2 -1
- package/dest/spartan/utils/config.d.ts +10 -30
- package/dest/spartan/utils/config.d.ts.map +1 -1
- package/dest/spartan/utils/config.js +3 -1
- package/dest/spartan/utils/index.d.ts +4 -2
- package/dest/spartan/utils/index.d.ts.map +1 -1
- package/dest/spartan/utils/index.js +5 -1
- package/dest/spartan/utils/k8s.d.ts +3 -1
- package/dest/spartan/utils/k8s.d.ts.map +1 -1
- package/dest/spartan/utils/k8s.js +6 -0
- package/dest/spartan/utils/nodes.d.ts +4 -5
- package/dest/spartan/utils/nodes.d.ts.map +1 -1
- package/dest/spartan/utils/nodes.js +11 -11
- package/dest/spartan/utils/pod_logs.d.ts +25 -0
- package/dest/spartan/utils/pod_logs.d.ts.map +1 -0
- package/dest/spartan/utils/pod_logs.js +74 -0
- package/dest/test-wallet/test_wallet.d.ts +85 -0
- package/dest/test-wallet/test_wallet.d.ts.map +1 -0
- package/dest/test-wallet/test_wallet.js +273 -0
- package/dest/test-wallet/utils.d.ts +41 -0
- package/dest/test-wallet/utils.d.ts.map +1 -0
- package/dest/test-wallet/utils.js +66 -0
- package/dest/test-wallet/wallet_worker_script.d.ts +2 -0
- package/dest/test-wallet/wallet_worker_script.d.ts.map +1 -0
- package/dest/test-wallet/wallet_worker_script.js +53 -0
- package/dest/test-wallet/worker_wallet.d.ts +53 -0
- package/dest/test-wallet/worker_wallet.d.ts.map +1 -0
- package/dest/test-wallet/worker_wallet.js +155 -0
- package/dest/test-wallet/worker_wallet_schema.d.ts +160 -0
- package/dest/test-wallet/worker_wallet_schema.d.ts.map +1 -0
- package/dest/test-wallet/worker_wallet_schema.js +22 -0
- package/package.json +49 -45
- package/src/bench/client_flows/benchmark.ts +19 -0
- package/src/bench/client_flows/client_flows_benchmark.ts +64 -49
- package/src/bench/client_flows/config.ts +9 -1
- package/src/bench/utils.ts +10 -4
- package/src/e2e_blacklist_token_contract/blacklist_token_contract_test.ts +52 -25
- package/src/e2e_cross_chain_messaging/cross_chain_messaging_test.ts +53 -15
- package/src/e2e_deploy_contract/deploy_test.ts +6 -5
- package/src/e2e_epochs/epochs_test.ts +166 -68
- package/src/e2e_fees/bridging_race.notest.ts +1 -1
- package/src/e2e_fees/fees_test.ts +57 -32
- package/src/e2e_nested_contract/nested_contract_test.ts +10 -6
- package/src/e2e_p2p/inactivity_slash_test.ts +8 -7
- package/src/e2e_p2p/p2p_network.ts +93 -49
- package/src/e2e_p2p/reqresp/utils.ts +84 -17
- package/src/e2e_p2p/shared.ts +108 -65
- package/src/e2e_storage_proof/fixtures/storage_proof.json +915 -0
- package/src/e2e_storage_proof/fixtures/storage_proof_fetcher.ts +190 -0
- package/src/e2e_storage_proof/fixtures/storage_proof_fixture.ts +173 -0
- package/src/e2e_token_contract/token_contract_test.ts +38 -11
- package/src/fixtures/authwit_proxy.ts +54 -0
- package/src/fixtures/dumps/epoch_proof_result.json +1 -1
- package/src/fixtures/e2e_prover_test.ts +49 -56
- package/src/fixtures/elu_monitor.ts +126 -0
- package/src/fixtures/fixtures.ts +93 -0
- package/src/fixtures/get_bb_config.ts +7 -6
- package/src/fixtures/ha_setup.ts +7 -3
- package/src/fixtures/schnorr_hardcoded_account_contract.ts +49 -0
- package/src/fixtures/setup.ts +272 -233
- package/src/fixtures/setup_p2p_test.ts +21 -25
- package/src/fixtures/token_utils.ts +3 -3
- package/src/fixtures/utils.ts +2 -0
- package/src/forward-compatibility/wallet_rpc_client.ts +14 -0
- package/src/forward-compatibility/wallet_service.ts +104 -0
- package/src/guides/up_quick_start.sh +3 -5
- package/src/install_legacy_contracts.cjs +75 -0
- package/src/legacy-jest-resolver.cjs +112 -0
- package/src/shared/cross_chain_test_harness.ts +27 -13
- package/src/shared/gas_portal_test_harness.ts +21 -11
- package/src/shared/index.ts +1 -0
- package/src/shared/jest_setup.ts +51 -1
- package/src/shared/mock_state_view.ts +188 -0
- package/src/shared/submit-transactions.ts +3 -2
- package/src/shared/uniswap_l1_l2.ts +104 -53
- package/src/shared/wait_for_l1_to_l2_message.ts +23 -0
- package/src/simulators/lending_simulator.ts +32 -14
- package/src/simulators/token_simulator.ts +6 -30
- package/src/spartan/setup_test_wallets.ts +146 -35
- package/src/spartan/tx_metrics.ts +82 -24
- package/src/spartan/utils/bot.ts +4 -1
- package/src/spartan/utils/config.ts +2 -0
- package/src/spartan/utils/index.ts +7 -0
- package/src/spartan/utils/k8s.ts +8 -0
- package/src/spartan/utils/nodes.ts +17 -12
- package/src/spartan/utils/pod_logs.ts +99 -0
- package/src/test-wallet/test_wallet.ts +376 -0
- package/src/test-wallet/utils.ts +108 -0
- package/src/test-wallet/wallet_worker_script.ts +63 -0
- package/src/test-wallet/worker_wallet.ts +218 -0
- 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
|
|
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 =
|
|
23
|
-
export const BOOT_NODE_UDP_PORT =
|
|
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
|
-
|
|
42
|
-
aztecSlotDuration:
|
|
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.
|
|
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
|
|
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
|
|
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
|
|
129
|
+
t.ctx.dateProvider.setTime(Number(timestamp) * 1000);
|
|
120
130
|
const startSlotTimestamp = BigInt(timestamp);
|
|
121
131
|
|
|
122
|
-
|
|
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:
|
|
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
|
}),
|
package/src/e2e_p2p/shared.ts
CHANGED
|
@@ -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
|
-
|
|
11
|
-
|
|
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
|
-
|
|
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
|
|
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
|
|
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:
|
|
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
|
-
|
|
111
|
-
|
|
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
|
-
|
|
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
|
-
|
|
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
|
-
|
|
220
|
-
|
|
221
|
-
|
|
222
|
-
|
|
223
|
-
|
|
224
|
-
|
|
225
|
-
|
|
226
|
-
|
|
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
|
-
|
|
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
|
|