@aztec/end-to-end 0.0.1-commit.db765a8 → 0.0.1-commit.ddcf04837
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/client_flows_benchmark.d.ts +1 -1
- package/dest/bench/client_flows/client_flows_benchmark.d.ts.map +1 -1
- package/dest/bench/client_flows/client_flows_benchmark.js +21 -29
- package/dest/e2e_blacklist_token_contract/blacklist_token_contract_test.d.ts +3 -2
- 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 +1 -1
- package/dest/e2e_epochs/epochs_test.d.ts +3 -1
- package/dest/e2e_epochs/epochs_test.d.ts.map +1 -1
- package/dest/e2e_epochs/epochs_test.js +8 -5
- package/dest/e2e_fees/fees_test.d.ts +1 -1
- package/dest/e2e_fees/fees_test.d.ts.map +1 -1
- package/dest/e2e_fees/fees_test.js +9 -2
- package/dest/e2e_p2p/inactivity_slash_test.js +3 -3
- package/dest/e2e_p2p/p2p_network.d.ts +6 -8
- package/dest/e2e_p2p/p2p_network.d.ts.map +1 -1
- package/dest/e2e_p2p/p2p_network.js +15 -14
- package/dest/e2e_p2p/reqresp/utils.d.ts +1 -1
- package/dest/e2e_p2p/reqresp/utils.d.ts.map +1 -1
- package/dest/e2e_p2p/reqresp/utils.js +16 -3
- package/dest/e2e_p2p/shared.d.ts +16 -12
- package/dest/e2e_p2p/shared.d.ts.map +1 -1
- package/dest/e2e_p2p/shared.js +33 -51
- package/dest/fixtures/authwit_proxy.d.ts +1 -1
- package/dest/fixtures/authwit_proxy.d.ts.map +1 -1
- package/dest/fixtures/authwit_proxy.js +4 -0
- package/dest/fixtures/e2e_prover_test.d.ts +4 -3
- package/dest/fixtures/e2e_prover_test.d.ts.map +1 -1
- package/dest/fixtures/e2e_prover_test.js +7 -12
- 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/setup.d.ts +16 -8
- package/dest/fixtures/setup.d.ts.map +1 -1
- package/dest/fixtures/setup.js +24 -18
- package/dest/fixtures/setup_p2p_test.d.ts +6 -6
- package/dest/fixtures/setup_p2p_test.d.ts.map +1 -1
- package/dest/fixtures/setup_p2p_test.js +8 -8
- package/dest/fixtures/token_utils.d.ts +1 -1
- package/dest/fixtures/token_utils.d.ts.map +1 -1
- package/dest/fixtures/token_utils.js +2 -5
- package/dest/legacy-jest-resolver.d.cts +3 -0
- package/dest/legacy-jest-resolver.d.cts.map +1 -0
- 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 +9 -12
- package/dest/simulators/lending_simulator.d.ts +1 -1
- package/dest/simulators/lending_simulator.d.ts.map +1 -1
- package/dest/simulators/lending_simulator.js +2 -2
- package/dest/simulators/token_simulator.js +1 -1
- package/dest/spartan/setup_test_wallets.d.ts +4 -2
- package/dest/spartan/setup_test_wallets.d.ts.map +1 -1
- package/dest/spartan/setup_test_wallets.js +70 -40
- package/dest/spartan/utils/config.d.ts +4 -1
- package/dest/spartan/utils/config.d.ts.map +1 -1
- package/dest/spartan/utils/config.js +1 -0
- package/dest/spartan/utils/index.d.ts +2 -1
- package/dest/spartan/utils/index.d.ts.map +1 -1
- package/dest/spartan/utils/index.js +2 -0
- 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 +24 -23
- package/dest/test-wallet/test_wallet.d.ts.map +1 -1
- package/dest/test-wallet/test_wallet.js +115 -80
- package/dest/test-wallet/worker_wallet.d.ts +4 -4
- package/dest/test-wallet/worker_wallet.d.ts.map +1 -1
- package/dest/test-wallet/worker_wallet_schema.d.ts +9 -4
- package/dest/test-wallet/worker_wallet_schema.d.ts.map +1 -1
- package/package.json +40 -40
- package/src/bench/client_flows/client_flows_benchmark.ts +35 -23
- package/src/e2e_blacklist_token_contract/blacklist_token_contract_test.ts +3 -6
- package/src/e2e_epochs/epochs_test.ts +17 -5
- package/src/e2e_fees/fees_test.ts +5 -2
- package/src/e2e_p2p/inactivity_slash_test.ts +3 -3
- package/src/e2e_p2p/p2p_network.ts +23 -28
- package/src/e2e_p2p/reqresp/utils.ts +24 -3
- package/src/e2e_p2p/shared.ts +36 -67
- package/src/fixtures/authwit_proxy.ts +4 -0
- package/src/fixtures/e2e_prover_test.ts +12 -17
- package/src/fixtures/get_bb_config.ts +7 -6
- package/src/fixtures/setup.ts +38 -25
- package/src/fixtures/setup_p2p_test.ts +9 -9
- package/src/fixtures/token_utils.ts +1 -4
- package/src/legacy-jest-resolver.cjs +135 -0
- package/src/shared/uniswap_l1_l2.ts +29 -24
- package/src/simulators/lending_simulator.ts +4 -2
- package/src/simulators/token_simulator.ts +1 -1
- package/src/spartan/setup_test_wallets.ts +97 -35
- package/src/spartan/utils/config.ts +1 -0
- package/src/spartan/utils/index.ts +3 -0
- package/src/spartan/utils/pod_logs.ts +99 -0
- package/src/test-wallet/test_wallet.ts +144 -99
- package/src/test-wallet/worker_wallet.ts +3 -2
|
@@ -110,7 +110,7 @@ export class TokenSimulator {
|
|
|
110
110
|
chunk(calls, 5).map(batch => new BatchCall(this.defaultWallet, batch).simulate({ from: this.defaultAddress })),
|
|
111
111
|
)
|
|
112
112
|
)
|
|
113
|
-
.
|
|
113
|
+
.flatMap(r => r.result)
|
|
114
114
|
.map(r => r.result);
|
|
115
115
|
expect(results[0]).toEqual(this.totalSupply);
|
|
116
116
|
|
|
@@ -1,4 +1,5 @@
|
|
|
1
1
|
import { generateSchnorrAccounts } from '@aztec/accounts/testing';
|
|
2
|
+
import { NO_FROM } from '@aztec/aztec.js/account';
|
|
2
3
|
import { AztecAddress } from '@aztec/aztec.js/addresses';
|
|
3
4
|
import { NO_WAIT } from '@aztec/aztec.js/contracts';
|
|
4
5
|
import { L1FeeJuicePortalManager } from '@aztec/aztec.js/ethereum';
|
|
@@ -10,7 +11,7 @@ import type { Wallet } from '@aztec/aztec.js/wallet';
|
|
|
10
11
|
import { createEthereumChain } from '@aztec/ethereum/chain';
|
|
11
12
|
import { createExtendedL1Client } from '@aztec/ethereum/client';
|
|
12
13
|
import type { Logger } from '@aztec/foundation/log';
|
|
13
|
-
import { retryUntil } from '@aztec/foundation/retry';
|
|
14
|
+
import { makeBackoff, retry, retryUntil } from '@aztec/foundation/retry';
|
|
14
15
|
import { TokenContract } from '@aztec/noir-contracts.js/Token';
|
|
15
16
|
import type { AztecNodeAdmin } from '@aztec/stdlib/interfaces/client';
|
|
16
17
|
import { registerInitialLocalNetworkAccountsInWallet } from '@aztec/wallets/testing';
|
|
@@ -89,7 +90,7 @@ export async function deploySponsoredTestAccountsWithTokens(
|
|
|
89
90
|
const paymentMethod = new SponsoredFeePaymentMethod(await getSponsoredFPCAddress());
|
|
90
91
|
const recipientDeployMethod = await recipientAccount.getDeployMethod();
|
|
91
92
|
await recipientDeployMethod.send({
|
|
92
|
-
from:
|
|
93
|
+
from: NO_FROM,
|
|
93
94
|
fee: { paymentMethod },
|
|
94
95
|
wait: { timeout: 2400 },
|
|
95
96
|
});
|
|
@@ -97,7 +98,7 @@ export async function deploySponsoredTestAccountsWithTokens(
|
|
|
97
98
|
fundedAccounts.map(async a => {
|
|
98
99
|
const deployMethod = await a.getDeployMethod();
|
|
99
100
|
await deployMethod.send({
|
|
100
|
-
from:
|
|
101
|
+
from: NO_FROM,
|
|
101
102
|
fee: { paymentMethod },
|
|
102
103
|
wait: { timeout: 2400 },
|
|
103
104
|
}); // increase timeout on purpose in order to account for two empty epochs
|
|
@@ -129,50 +130,91 @@ export async function deploySponsoredTestAccountsWithTokens(
|
|
|
129
130
|
}
|
|
130
131
|
|
|
131
132
|
async function deployAccountWithDiagnostics(
|
|
132
|
-
account: { getDeployMethod: () => Promise<{ send: (opts: any) => any }>; address: any },
|
|
133
|
+
account: { getDeployMethod: () => Promise<{ simulate: (opts: any) => any; send: (opts: any) => any }>; address: any },
|
|
133
134
|
paymentMethod: SponsoredFeePaymentMethod,
|
|
134
135
|
aztecNode: AztecNode,
|
|
135
136
|
logger: Logger,
|
|
136
137
|
accountLabel: string,
|
|
138
|
+
estimateGas?: boolean,
|
|
137
139
|
): Promise<void> {
|
|
138
140
|
const deployMethod = await account.getDeployMethod();
|
|
139
|
-
let
|
|
140
|
-
|
|
141
|
-
const
|
|
142
|
-
|
|
143
|
-
|
|
144
|
-
logger.info(`${accountLabel} deployed at ${account.address}`);
|
|
145
|
-
} catch (error) {
|
|
146
|
-
const blockNumber = await aztecNode.getBlockNumber();
|
|
147
|
-
let receipt;
|
|
148
|
-
try {
|
|
149
|
-
receipt = await aztecNode.getTxReceipt(txHash);
|
|
150
|
-
} catch {
|
|
151
|
-
receipt = 'unavailable';
|
|
152
|
-
}
|
|
153
|
-
logger.error(`${accountLabel} deployment failed`, {
|
|
154
|
-
txHash: txHash.toString(),
|
|
155
|
-
receipt: JSON.stringify(receipt),
|
|
156
|
-
currentBlockNumber: blockNumber,
|
|
157
|
-
error: String(error),
|
|
158
|
-
});
|
|
159
|
-
throw error;
|
|
141
|
+
let gasSettings: any;
|
|
142
|
+
if (estimateGas) {
|
|
143
|
+
const sim = await deployMethod.simulate({ from: NO_FROM, fee: { paymentMethod } });
|
|
144
|
+
gasSettings = sim.estimatedGas;
|
|
145
|
+
logger.info(`${accountLabel} estimated gas: DA=${gasSettings.gasLimits.daGas} L2=${gasSettings.gasLimits.l2Gas}`);
|
|
160
146
|
}
|
|
147
|
+
|
|
148
|
+
// Track the tx hash across retries so we don't re-send when the previous tx is still pending.
|
|
149
|
+
let sentTxHash: { txHash: any } | undefined;
|
|
150
|
+
|
|
151
|
+
await retry(
|
|
152
|
+
async () => {
|
|
153
|
+
// Check if already deployed (handles case where previous attempt succeeded but waitForTx timed out)
|
|
154
|
+
const existing = await aztecNode.getContract(account.address);
|
|
155
|
+
if (existing) {
|
|
156
|
+
logger.info(`${accountLabel} already deployed at ${account.address}, skipping`);
|
|
157
|
+
return;
|
|
158
|
+
}
|
|
159
|
+
|
|
160
|
+
// If we already sent a tx, check if it was dropped before deciding to re-send.
|
|
161
|
+
if (sentTxHash) {
|
|
162
|
+
const prevReceipt = await aztecNode.getTxReceipt(sentTxHash.txHash);
|
|
163
|
+
if (prevReceipt.isDropped()) {
|
|
164
|
+
logger.info(`${accountLabel} previous tx ${sentTxHash.txHash} was dropped, re-sending`);
|
|
165
|
+
sentTxHash = undefined;
|
|
166
|
+
} else {
|
|
167
|
+
logger.info(`${accountLabel} previous tx ${sentTxHash.txHash} still pending, waiting again...`);
|
|
168
|
+
}
|
|
169
|
+
}
|
|
170
|
+
|
|
171
|
+
if (!sentTxHash) {
|
|
172
|
+
const deployResult = await deployMethod.send({
|
|
173
|
+
from: NO_FROM,
|
|
174
|
+
fee: { paymentMethod, gasSettings },
|
|
175
|
+
wait: NO_WAIT,
|
|
176
|
+
});
|
|
177
|
+
sentTxHash = { txHash: deployResult.txHash };
|
|
178
|
+
logger.info(`${accountLabel} tx sent`, { txHash: sentTxHash.txHash.toString() });
|
|
179
|
+
}
|
|
180
|
+
|
|
181
|
+
const receipt = await waitForTx(aztecNode, sentTxHash.txHash, { timeout: 600 });
|
|
182
|
+
if (receipt.isDropped()) {
|
|
183
|
+
sentTxHash = undefined;
|
|
184
|
+
throw new Error(`${accountLabel} tx was dropped, retrying...`);
|
|
185
|
+
}
|
|
186
|
+
logger.info(`${accountLabel} deployed at ${account.address}`);
|
|
187
|
+
},
|
|
188
|
+
`deploy ${accountLabel}`,
|
|
189
|
+
makeBackoff([1, 2, 4, 8, 16]),
|
|
190
|
+
logger,
|
|
191
|
+
);
|
|
161
192
|
}
|
|
162
193
|
|
|
163
194
|
async function deployAccountsInBatches(
|
|
164
|
-
accounts: {
|
|
195
|
+
accounts: {
|
|
196
|
+
getDeployMethod: () => Promise<{ simulate: (opts: any) => any; send: (opts: any) => any }>;
|
|
197
|
+
address: any;
|
|
198
|
+
}[],
|
|
165
199
|
paymentMethod: SponsoredFeePaymentMethod,
|
|
166
200
|
aztecNode: AztecNode,
|
|
167
201
|
logger: Logger,
|
|
168
202
|
labelPrefix: string,
|
|
169
203
|
batchSize = 2,
|
|
204
|
+
estimateGas?: boolean,
|
|
170
205
|
): Promise<void> {
|
|
171
206
|
for (let i = 0; i < accounts.length; i += batchSize) {
|
|
172
207
|
const batch = accounts.slice(i, i + batchSize);
|
|
173
208
|
await Promise.all(
|
|
174
209
|
batch.map((account, idx) =>
|
|
175
|
-
deployAccountWithDiagnostics(
|
|
210
|
+
deployAccountWithDiagnostics(
|
|
211
|
+
account,
|
|
212
|
+
paymentMethod,
|
|
213
|
+
aztecNode,
|
|
214
|
+
logger,
|
|
215
|
+
`${labelPrefix}${i + idx + 1}`,
|
|
216
|
+
estimateGas,
|
|
217
|
+
),
|
|
176
218
|
),
|
|
177
219
|
);
|
|
178
220
|
}
|
|
@@ -183,6 +225,7 @@ export async function deploySponsoredTestAccounts(
|
|
|
183
225
|
aztecNode: AztecNode,
|
|
184
226
|
logger: Logger,
|
|
185
227
|
numberOfFundedWallets = 1,
|
|
228
|
+
opts?: { estimateGas?: boolean },
|
|
186
229
|
): Promise<TestAccountsWithoutTokens> {
|
|
187
230
|
const [recipient, ...funded] = await generateSchnorrAccounts(numberOfFundedWallets + 1);
|
|
188
231
|
const recipientAccount = await wallet.createSchnorrAccount(recipient.secret, recipient.salt);
|
|
@@ -192,8 +235,23 @@ export async function deploySponsoredTestAccounts(
|
|
|
192
235
|
|
|
193
236
|
const paymentMethod = new SponsoredFeePaymentMethod(await getSponsoredFPCAddress());
|
|
194
237
|
|
|
195
|
-
await deployAccountWithDiagnostics(
|
|
196
|
-
|
|
238
|
+
await deployAccountWithDiagnostics(
|
|
239
|
+
recipientAccount,
|
|
240
|
+
paymentMethod,
|
|
241
|
+
aztecNode,
|
|
242
|
+
logger,
|
|
243
|
+
'Recipient account',
|
|
244
|
+
opts?.estimateGas,
|
|
245
|
+
);
|
|
246
|
+
await deployAccountsInBatches(
|
|
247
|
+
fundedAccounts,
|
|
248
|
+
paymentMethod,
|
|
249
|
+
aztecNode,
|
|
250
|
+
logger,
|
|
251
|
+
'Funded account ',
|
|
252
|
+
2,
|
|
253
|
+
opts?.estimateGas,
|
|
254
|
+
);
|
|
197
255
|
|
|
198
256
|
return {
|
|
199
257
|
aztecNode,
|
|
@@ -231,7 +289,7 @@ export async function deployTestAccountsWithTokens(
|
|
|
231
289
|
fundedAccounts.map(async (a, i) => {
|
|
232
290
|
const paymentMethod = new FeeJuicePaymentMethodWithClaim(a.address, claims[i]);
|
|
233
291
|
const deployMethod = await a.getDeployMethod();
|
|
234
|
-
await deployMethod.send({ from:
|
|
292
|
+
await deployMethod.send({ from: NO_FROM, fee: { paymentMethod } });
|
|
235
293
|
logger.info(`Account deployed at ${a.address}`);
|
|
236
294
|
}),
|
|
237
295
|
);
|
|
@@ -275,7 +333,7 @@ async function bridgeL1FeeJuice(
|
|
|
275
333
|
const claim = await portal.bridgeTokensPublic(recipient, amount, true /* mint */);
|
|
276
334
|
|
|
277
335
|
const isSynced = async () =>
|
|
278
|
-
(await aztecNode.
|
|
336
|
+
(await aztecNode.getL1ToL2MessageCheckpoint(Fr.fromHexString(claim.messageHash))) !== undefined;
|
|
279
337
|
await retryUntil(isSynced, `message ${claim.messageHash} sync`, 24, 0.5);
|
|
280
338
|
|
|
281
339
|
log.info(`Created a claim for ${amount} L1 fee juice to ${recipient}.`, claim);
|
|
@@ -307,14 +365,18 @@ async function deployTokenAndMint(
|
|
|
307
365
|
logger: Logger,
|
|
308
366
|
) {
|
|
309
367
|
logger.verbose(`Deploying TokenContract...`);
|
|
310
|
-
const {
|
|
311
|
-
|
|
312
|
-
|
|
368
|
+
const { contract: tokenContract } = await TokenContract.deploy(
|
|
369
|
+
wallet,
|
|
370
|
+
admin,
|
|
371
|
+
TOKEN_NAME,
|
|
372
|
+
TOKEN_SYMBOL,
|
|
373
|
+
TOKEN_DECIMALS,
|
|
374
|
+
).send({
|
|
313
375
|
from: admin,
|
|
314
376
|
fee: {
|
|
315
377
|
paymentMethod,
|
|
316
378
|
},
|
|
317
|
-
wait: { timeout: 600
|
|
379
|
+
wait: { timeout: 600 },
|
|
318
380
|
});
|
|
319
381
|
|
|
320
382
|
const tokenAddress = tokenContract.address;
|
|
@@ -8,6 +8,7 @@ const logger = createLogger('e2e:k8s-utils');
|
|
|
8
8
|
const testConfigSchema = z.object({
|
|
9
9
|
NAMESPACE: z.string().default('scenario'),
|
|
10
10
|
REAL_VERIFIER: schemas.Boolean.optional().default(true),
|
|
11
|
+
DEBUG_FORCE_TX_PROOF_VERIFICATION: schemas.Boolean.optional().default(true),
|
|
11
12
|
CREATE_ETH_DEVNET: schemas.Boolean.optional().default(false),
|
|
12
13
|
L1_RPC_URLS_JSON: z.string().optional(),
|
|
13
14
|
L1_ACCOUNT_MNEMONIC: z.string().optional(),
|
|
@@ -66,3 +66,6 @@ export { getPublicViemClient, getL1DeploymentAddresses, getNodeClient } from './
|
|
|
66
66
|
|
|
67
67
|
// Health checks
|
|
68
68
|
export { ChainHealth, type ChainHealthSnapshot } from './health.js';
|
|
69
|
+
|
|
70
|
+
// Pod log extraction
|
|
71
|
+
export { type BlockBuiltLogEntry, fetchBlockBuiltLogs } from './pod_logs.js';
|
|
@@ -0,0 +1,99 @@
|
|
|
1
|
+
import type { Logger } from '@aztec/foundation/log';
|
|
2
|
+
|
|
3
|
+
import { exec } from 'child_process';
|
|
4
|
+
import { promisify } from 'util';
|
|
5
|
+
|
|
6
|
+
import { getSequencers } from './nodes.js';
|
|
7
|
+
|
|
8
|
+
const execAsync = promisify(exec);
|
|
9
|
+
|
|
10
|
+
/** Parsed l2-block-built stats from a sequencer pod log line. */
|
|
11
|
+
export type BlockBuiltLogEntry = {
|
|
12
|
+
blockNumber: number;
|
|
13
|
+
txCount: number;
|
|
14
|
+
duration: number;
|
|
15
|
+
publicProcessDuration: number;
|
|
16
|
+
manaPerSec: number;
|
|
17
|
+
privateLogCount: number;
|
|
18
|
+
publicLogCount: number;
|
|
19
|
+
contractClassLogCount: number;
|
|
20
|
+
contractClassLogSize: number;
|
|
21
|
+
};
|
|
22
|
+
|
|
23
|
+
const FIELDS: (keyof BlockBuiltLogEntry)[] = [
|
|
24
|
+
'blockNumber',
|
|
25
|
+
'txCount',
|
|
26
|
+
'duration',
|
|
27
|
+
'publicProcessDuration',
|
|
28
|
+
'manaPerSec',
|
|
29
|
+
'privateLogCount',
|
|
30
|
+
'publicLogCount',
|
|
31
|
+
'contractClassLogCount',
|
|
32
|
+
'contractClassLogSize',
|
|
33
|
+
];
|
|
34
|
+
|
|
35
|
+
/**
|
|
36
|
+
* Fetches l2-block-built log entries from sequencer pods for given block numbers.
|
|
37
|
+
* Queries all validator pods (only the proposer will have the log for a given block).
|
|
38
|
+
*
|
|
39
|
+
* @param namespace - Kubernetes namespace
|
|
40
|
+
* @param sinceTime - ISO 8601 timestamp to limit log search (e.g., from before block building was re-enabled)
|
|
41
|
+
* @param blockNumbers - Set of block numbers to filter for
|
|
42
|
+
* @param logger - Logger instance
|
|
43
|
+
* @returns Array of parsed BlockBuiltLogEntry, de-duplicated by blockNumber, sorted ascending
|
|
44
|
+
*/
|
|
45
|
+
export async function fetchBlockBuiltLogs(
|
|
46
|
+
namespace: string,
|
|
47
|
+
sinceTime: string,
|
|
48
|
+
blockNumbers: Set<number>,
|
|
49
|
+
logger: Logger,
|
|
50
|
+
): Promise<BlockBuiltLogEntry[]> {
|
|
51
|
+
const pods = await getSequencers(namespace);
|
|
52
|
+
const entriesByBlock = new Map<number, BlockBuiltLogEntry>();
|
|
53
|
+
|
|
54
|
+
// Subtract 60s from sinceTime to account for clock skew between test runner and k8s pods.
|
|
55
|
+
// Block number filtering ensures we only match the right blocks, so extra lines are harmless.
|
|
56
|
+
const sinceDate = new Date(new Date(sinceTime).getTime() - 60_000);
|
|
57
|
+
const sinceFlag = sinceDate.toISOString();
|
|
58
|
+
|
|
59
|
+
for (const pod of pods) {
|
|
60
|
+
try {
|
|
61
|
+
const cmd = `kubectl logs ${pod} -n ${namespace} -c aztec --since-time=${sinceFlag}`;
|
|
62
|
+
logger.info(`Fetching logs: ${cmd}`);
|
|
63
|
+
const { stdout } = await execAsync(cmd, { maxBuffer: 10 * 1024 * 1024 });
|
|
64
|
+
|
|
65
|
+
const lines = stdout.split('\n');
|
|
66
|
+
const matchingLines = lines.filter(l => l.includes('l2-block-built'));
|
|
67
|
+
logger.info(`Pod ${pod}: ${lines.length} log lines, ${matchingLines.length} contain l2-block-built`);
|
|
68
|
+
|
|
69
|
+
for (const line of matchingLines) {
|
|
70
|
+
try {
|
|
71
|
+
const parsed = JSON.parse(line);
|
|
72
|
+
if (parsed.eventName !== 'l2-block-built' || !blockNumbers.has(parsed.blockNumber)) {
|
|
73
|
+
continue;
|
|
74
|
+
}
|
|
75
|
+
if (entriesByBlock.has(parsed.blockNumber)) {
|
|
76
|
+
continue;
|
|
77
|
+
}
|
|
78
|
+
const entry: BlockBuiltLogEntry = {} as BlockBuiltLogEntry;
|
|
79
|
+
for (const field of FIELDS) {
|
|
80
|
+
entry[field] = parsed[field] ?? 0;
|
|
81
|
+
}
|
|
82
|
+
entriesByBlock.set(entry.blockNumber, entry);
|
|
83
|
+
logger.verbose(`Parsed l2-block-built log for block ${entry.blockNumber}`, entry);
|
|
84
|
+
} catch {
|
|
85
|
+
// Not valid JSON, skip
|
|
86
|
+
}
|
|
87
|
+
}
|
|
88
|
+
} catch (err) {
|
|
89
|
+
logger.warn(`Failed to fetch logs from pod ${pod}: ${err}`);
|
|
90
|
+
}
|
|
91
|
+
}
|
|
92
|
+
|
|
93
|
+
if (entriesByBlock.size < blockNumbers.size) {
|
|
94
|
+
const missing = [...blockNumbers].filter(bn => !entriesByBlock.has(bn));
|
|
95
|
+
logger.warn(`Missing l2-block-built logs for block(s): ${missing.join(', ')}`);
|
|
96
|
+
}
|
|
97
|
+
|
|
98
|
+
return [...entriesByBlock.values()].sort((a, b) => a.blockNumber - b.blockNumber);
|
|
99
|
+
}
|