@aztec/archiver 0.0.1-commit.96bb3f7 → 0.0.1-commit.993d240
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 +164 -22
- package/dest/archiver.d.ts +158 -0
- package/dest/archiver.d.ts.map +1 -0
- package/dest/archiver.js +881 -0
- package/dest/config.d.ts +33 -0
- package/dest/config.d.ts.map +1 -0
- package/dest/{archiver/config.js → config.js} +31 -14
- package/dest/errors.d.ts +87 -0
- package/dest/errors.d.ts.map +1 -0
- package/dest/errors.js +129 -0
- package/dest/factory.d.ts +16 -10
- package/dest/factory.d.ts.map +1 -1
- package/dest/factory.js +112 -20
- package/dest/index.d.ts +19 -4
- package/dest/index.d.ts.map +1 -1
- package/dest/index.js +17 -3
- package/dest/interfaces.d.ts +9 -0
- package/dest/interfaces.d.ts.map +1 -0
- package/dest/interfaces.js +3 -0
- package/dest/{archiver/l1 → l1}/bin/retrieve-calldata.d.ts +1 -1
- package/dest/l1/bin/retrieve-calldata.d.ts.map +1 -0
- package/dest/{archiver/l1 → l1}/bin/retrieve-calldata.js +35 -32
- package/dest/l1/calldata_retriever.d.ts +136 -0
- package/dest/l1/calldata_retriever.d.ts.map +1 -0
- package/dest/l1/calldata_retriever.js +412 -0
- package/dest/l1/data_retrieval.d.ts +97 -0
- package/dest/l1/data_retrieval.d.ts.map +1 -0
- package/dest/{archiver/l1 → l1}/data_retrieval.js +65 -89
- package/dest/{archiver/l1 → l1}/debug_tx.d.ts +1 -1
- package/dest/l1/debug_tx.d.ts.map +1 -0
- package/dest/{archiver/l1 → l1}/spire_proposer.d.ts +5 -5
- package/dest/l1/spire_proposer.d.ts.map +1 -0
- package/dest/{archiver/l1 → l1}/spire_proposer.js +9 -17
- package/dest/l1/trace_tx.d.ts +43 -0
- package/dest/l1/trace_tx.d.ts.map +1 -0
- package/dest/l1/types.d.ts +12 -0
- package/dest/l1/types.d.ts.map +1 -0
- package/dest/l1/validate_historical_logs.d.ts +23 -0
- package/dest/l1/validate_historical_logs.d.ts.map +1 -0
- package/dest/l1/validate_historical_logs.js +108 -0
- package/dest/{archiver/l1 → l1}/validate_trace.d.ts +6 -3
- package/dest/l1/validate_trace.d.ts.map +1 -0
- package/dest/{archiver/l1 → l1}/validate_trace.js +13 -9
- package/dest/modules/contract_data_source_adapter.d.ts +25 -0
- package/dest/modules/contract_data_source_adapter.d.ts.map +1 -0
- package/dest/modules/contract_data_source_adapter.js +40 -0
- package/dest/modules/data_source_base.d.ts +113 -0
- package/dest/modules/data_source_base.d.ts.map +1 -0
- package/dest/modules/data_source_base.js +351 -0
- package/dest/modules/data_store_updater.d.ts +105 -0
- package/dest/modules/data_store_updater.d.ts.map +1 -0
- package/dest/modules/data_store_updater.js +392 -0
- package/dest/modules/instrumentation.d.ts +55 -0
- package/dest/modules/instrumentation.d.ts.map +1 -0
- package/dest/{archiver → modules}/instrumentation.js +61 -19
- package/dest/modules/l1_synchronizer.d.ts +77 -0
- package/dest/modules/l1_synchronizer.d.ts.map +1 -0
- package/dest/modules/l1_synchronizer.js +1344 -0
- package/dest/modules/validation.d.ts +18 -0
- package/dest/modules/validation.d.ts.map +1 -0
- package/dest/{archiver → modules}/validation.js +12 -6
- package/dest/store/block_store.d.ts +300 -0
- package/dest/store/block_store.d.ts.map +1 -0
- package/dest/store/block_store.js +1219 -0
- package/dest/store/contract_class_store.d.ts +31 -0
- package/dest/store/contract_class_store.d.ts.map +1 -0
- package/dest/store/contract_class_store.js +80 -0
- package/dest/store/contract_instance_store.d.ts +51 -0
- package/dest/store/contract_instance_store.d.ts.map +1 -0
- package/dest/{archiver/kv_archiver_store → store}/contract_instance_store.js +38 -3
- package/dest/store/data_stores.d.ts +68 -0
- package/dest/store/data_stores.d.ts.map +1 -0
- package/dest/store/data_stores.js +54 -0
- package/dest/store/function_names_cache.d.ts +17 -0
- package/dest/store/function_names_cache.d.ts.map +1 -0
- package/dest/store/function_names_cache.js +30 -0
- package/dest/store/l2_tips_cache.d.ts +25 -0
- package/dest/store/l2_tips_cache.d.ts.map +1 -0
- package/dest/store/l2_tips_cache.js +26 -0
- package/dest/store/log_store.d.ts +59 -0
- package/dest/store/log_store.d.ts.map +1 -0
- package/dest/store/log_store.js +310 -0
- package/dest/store/log_store_codec.d.ts +70 -0
- package/dest/store/log_store_codec.d.ts.map +1 -0
- package/dest/store/log_store_codec.js +101 -0
- package/dest/store/message_store.d.ts +50 -0
- package/dest/store/message_store.d.ts.map +1 -0
- package/dest/{archiver/kv_archiver_store → store}/message_store.js +51 -9
- package/dest/{archiver/structs → structs}/data_retrieval.d.ts +1 -1
- package/dest/structs/data_retrieval.d.ts.map +1 -0
- package/dest/structs/inbox_message.d.ts +15 -0
- package/dest/structs/inbox_message.d.ts.map +1 -0
- package/dest/{archiver/structs → structs}/published.d.ts +1 -1
- package/dest/structs/published.d.ts.map +1 -0
- package/dest/test/fake_l1_state.d.ts +214 -0
- package/dest/test/fake_l1_state.d.ts.map +1 -0
- package/dest/test/fake_l1_state.js +517 -0
- package/dest/test/index.d.ts +2 -1
- package/dest/test/index.d.ts.map +1 -1
- package/dest/test/index.js +4 -1
- package/dest/test/mock_archiver.d.ts +2 -2
- package/dest/test/mock_archiver.d.ts.map +1 -1
- package/dest/test/mock_archiver.js +3 -3
- package/dest/test/mock_l1_to_l2_message_source.d.ts +1 -1
- package/dest/test/mock_l1_to_l2_message_source.d.ts.map +1 -1
- package/dest/test/mock_l1_to_l2_message_source.js +2 -1
- package/dest/test/mock_l2_block_source.d.ts +65 -41
- package/dest/test/mock_l2_block_source.d.ts.map +1 -1
- package/dest/test/mock_l2_block_source.js +330 -151
- package/dest/test/mock_structs.d.ts +81 -3
- package/dest/test/mock_structs.d.ts.map +1 -1
- package/dest/test/mock_structs.js +152 -7
- package/dest/test/noop_l1_archiver.d.ts +29 -0
- package/dest/test/noop_l1_archiver.d.ts.map +1 -0
- package/dest/test/noop_l1_archiver.js +85 -0
- package/package.json +17 -18
- package/src/archiver.ts +681 -0
- package/src/{archiver/config.ts → config.ts} +43 -12
- package/src/errors.ts +203 -0
- package/src/factory.ts +175 -22
- package/src/index.ts +27 -3
- package/src/interfaces.ts +9 -0
- package/src/l1/README.md +55 -0
- package/src/{archiver/l1 → l1}/bin/retrieve-calldata.ts +45 -33
- package/src/l1/calldata_retriever.ts +522 -0
- package/src/{archiver/l1 → l1}/data_retrieval.ts +106 -134
- package/src/{archiver/l1 → l1}/spire_proposer.ts +7 -15
- package/src/l1/validate_historical_logs.ts +140 -0
- package/src/{archiver/l1 → l1}/validate_trace.ts +24 -6
- package/src/modules/contract_data_source_adapter.ts +55 -0
- package/src/modules/data_source_base.ts +493 -0
- package/src/modules/data_store_updater.ts +518 -0
- package/src/{archiver → modules}/instrumentation.ts +72 -20
- package/src/modules/l1_synchronizer.ts +1257 -0
- package/src/{archiver → modules}/validation.ts +15 -9
- package/src/store/block_store.ts +1590 -0
- package/src/store/contract_class_store.ts +108 -0
- package/src/{archiver/kv_archiver_store → store}/contract_instance_store.ts +52 -6
- package/src/store/data_stores.ts +104 -0
- package/src/store/function_names_cache.ts +37 -0
- package/src/store/l2_tips_cache.ts +35 -0
- package/src/store/log_store.ts +379 -0
- package/src/store/log_store_codec.ts +132 -0
- package/src/{archiver/kv_archiver_store → store}/message_store.ts +60 -10
- package/src/{archiver/structs → structs}/inbox_message.ts +1 -1
- package/src/test/fake_l1_state.ts +770 -0
- package/src/test/index.ts +4 -0
- package/src/test/mock_archiver.ts +4 -3
- package/src/test/mock_l1_to_l2_message_source.ts +1 -0
- package/src/test/mock_l2_block_source.ts +403 -171
- package/src/test/mock_structs.ts +283 -8
- package/src/test/noop_l1_archiver.ts +139 -0
- package/dest/archiver/archiver.d.ts +0 -307
- package/dest/archiver/archiver.d.ts.map +0 -1
- package/dest/archiver/archiver.js +0 -2102
- package/dest/archiver/archiver_store.d.ts +0 -315
- package/dest/archiver/archiver_store.d.ts.map +0 -1
- package/dest/archiver/archiver_store.js +0 -4
- package/dest/archiver/archiver_store_test_suite.d.ts +0 -8
- package/dest/archiver/archiver_store_test_suite.d.ts.map +0 -1
- package/dest/archiver/archiver_store_test_suite.js +0 -2770
- package/dest/archiver/config.d.ts +0 -22
- package/dest/archiver/config.d.ts.map +0 -1
- package/dest/archiver/errors.d.ts +0 -36
- package/dest/archiver/errors.d.ts.map +0 -1
- package/dest/archiver/errors.js +0 -54
- package/dest/archiver/index.d.ts +0 -7
- package/dest/archiver/index.d.ts.map +0 -1
- package/dest/archiver/index.js +0 -4
- package/dest/archiver/instrumentation.d.ts +0 -37
- package/dest/archiver/instrumentation.d.ts.map +0 -1
- package/dest/archiver/kv_archiver_store/block_store.d.ts +0 -164
- package/dest/archiver/kv_archiver_store/block_store.d.ts.map +0 -1
- package/dest/archiver/kv_archiver_store/block_store.js +0 -626
- package/dest/archiver/kv_archiver_store/contract_class_store.d.ts +0 -18
- package/dest/archiver/kv_archiver_store/contract_class_store.d.ts.map +0 -1
- package/dest/archiver/kv_archiver_store/contract_class_store.js +0 -120
- package/dest/archiver/kv_archiver_store/contract_instance_store.d.ts +0 -24
- package/dest/archiver/kv_archiver_store/contract_instance_store.d.ts.map +0 -1
- package/dest/archiver/kv_archiver_store/kv_archiver_store.d.ts +0 -159
- package/dest/archiver/kv_archiver_store/kv_archiver_store.d.ts.map +0 -1
- package/dest/archiver/kv_archiver_store/kv_archiver_store.js +0 -316
- package/dest/archiver/kv_archiver_store/log_store.d.ts +0 -45
- package/dest/archiver/kv_archiver_store/log_store.d.ts.map +0 -1
- package/dest/archiver/kv_archiver_store/log_store.js +0 -401
- package/dest/archiver/kv_archiver_store/message_store.d.ts +0 -40
- package/dest/archiver/kv_archiver_store/message_store.d.ts.map +0 -1
- package/dest/archiver/l1/bin/retrieve-calldata.d.ts.map +0 -1
- package/dest/archiver/l1/calldata_retriever.d.ts +0 -112
- package/dest/archiver/l1/calldata_retriever.d.ts.map +0 -1
- package/dest/archiver/l1/calldata_retriever.js +0 -471
- package/dest/archiver/l1/data_retrieval.d.ts +0 -90
- package/dest/archiver/l1/data_retrieval.d.ts.map +0 -1
- package/dest/archiver/l1/debug_tx.d.ts.map +0 -1
- package/dest/archiver/l1/spire_proposer.d.ts.map +0 -1
- package/dest/archiver/l1/trace_tx.d.ts +0 -97
- package/dest/archiver/l1/trace_tx.d.ts.map +0 -1
- package/dest/archiver/l1/types.d.ts +0 -12
- package/dest/archiver/l1/types.d.ts.map +0 -1
- package/dest/archiver/l1/validate_trace.d.ts.map +0 -1
- package/dest/archiver/structs/data_retrieval.d.ts.map +0 -1
- package/dest/archiver/structs/inbox_message.d.ts +0 -15
- package/dest/archiver/structs/inbox_message.d.ts.map +0 -1
- package/dest/archiver/structs/published.d.ts.map +0 -1
- package/dest/archiver/validation.d.ts +0 -17
- package/dest/archiver/validation.d.ts.map +0 -1
- package/dest/rpc/index.d.ts +0 -9
- package/dest/rpc/index.d.ts.map +0 -1
- package/dest/rpc/index.js +0 -15
- package/src/archiver/archiver.ts +0 -2265
- package/src/archiver/archiver_store.ts +0 -380
- package/src/archiver/archiver_store_test_suite.ts +0 -2842
- package/src/archiver/errors.ts +0 -90
- package/src/archiver/index.ts +0 -6
- package/src/archiver/kv_archiver_store/block_store.ts +0 -850
- package/src/archiver/kv_archiver_store/contract_class_store.ts +0 -176
- package/src/archiver/kv_archiver_store/kv_archiver_store.ts +0 -442
- package/src/archiver/kv_archiver_store/log_store.ts +0 -516
- package/src/archiver/l1/README.md +0 -98
- package/src/archiver/l1/calldata_retriever.ts +0 -641
- package/src/rpc/index.ts +0 -16
- /package/dest/{archiver/l1 → l1}/debug_tx.js +0 -0
- /package/dest/{archiver/l1 → l1}/trace_tx.js +0 -0
- /package/dest/{archiver/l1 → l1}/types.js +0 -0
- /package/dest/{archiver/structs → structs}/data_retrieval.js +0 -0
- /package/dest/{archiver/structs → structs}/inbox_message.js +0 -0
- /package/dest/{archiver/structs → structs}/published.js +0 -0
- /package/src/{archiver/l1 → l1}/debug_tx.ts +0 -0
- /package/src/{archiver/l1 → l1}/trace_tx.ts +0 -0
- /package/src/{archiver/l1 → l1}/types.ts +0 -0
- /package/src/{archiver/structs → structs}/data_retrieval.ts +0 -0
- /package/src/{archiver/structs → structs}/published.ts +0 -0
|
@@ -1,6 +1,12 @@
|
|
|
1
1
|
import { GENESIS_ARCHIVE_ROOT } from '@aztec/constants';
|
|
2
2
|
import { DefaultL1ContractsConfig } from '@aztec/ethereum/config';
|
|
3
|
-
import {
|
|
3
|
+
import {
|
|
4
|
+
BlockNumber,
|
|
5
|
+
CheckpointNumber,
|
|
6
|
+
EpochNumber,
|
|
7
|
+
IndexWithinCheckpoint,
|
|
8
|
+
SlotNumber,
|
|
9
|
+
} from '@aztec/foundation/branded-types';
|
|
4
10
|
import { Buffer32 } from '@aztec/foundation/buffer';
|
|
5
11
|
import { Fr } from '@aztec/foundation/curves/bn254';
|
|
6
12
|
import { EthAddress } from '@aztec/foundation/eth-address';
|
|
@@ -8,19 +14,40 @@ import { createLogger } from '@aztec/foundation/log';
|
|
|
8
14
|
import type { FunctionSelector } from '@aztec/stdlib/abi';
|
|
9
15
|
import type { AztecAddress } from '@aztec/stdlib/aztec-address';
|
|
10
16
|
import {
|
|
11
|
-
|
|
17
|
+
type BlockData,
|
|
18
|
+
type BlockHash,
|
|
19
|
+
type BlockQuery,
|
|
20
|
+
type BlockTag,
|
|
21
|
+
type BlocksQuery,
|
|
22
|
+
Body,
|
|
23
|
+
type CheckpointQuery,
|
|
24
|
+
type CheckpointsQuery,
|
|
25
|
+
GENESIS_BLOCK_HEADER_HASH,
|
|
26
|
+
GENESIS_CHECKPOINT_HEADER_HASH,
|
|
12
27
|
L2Block,
|
|
13
|
-
L2BlockHash,
|
|
14
|
-
L2BlockNew,
|
|
15
28
|
type L2BlockSource,
|
|
16
29
|
type L2Tips,
|
|
17
|
-
|
|
30
|
+
type ProposedCheckpointQuery,
|
|
18
31
|
type ValidateCheckpointResult,
|
|
19
32
|
} from '@aztec/stdlib/block';
|
|
20
|
-
import {
|
|
33
|
+
import {
|
|
34
|
+
Checkpoint,
|
|
35
|
+
type CheckpointData,
|
|
36
|
+
L1PublishedData,
|
|
37
|
+
type ProposedCheckpointData,
|
|
38
|
+
PublishedCheckpoint,
|
|
39
|
+
} from '@aztec/stdlib/checkpoint';
|
|
21
40
|
import type { ContractClassPublic, ContractDataSource, ContractInstanceWithAddress } from '@aztec/stdlib/contract';
|
|
22
|
-
import {
|
|
23
|
-
|
|
41
|
+
import {
|
|
42
|
+
EmptyL1RollupConstants,
|
|
43
|
+
type L1RollupConstants,
|
|
44
|
+
getEpochAtSlot,
|
|
45
|
+
getSlotRangeForEpoch,
|
|
46
|
+
} from '@aztec/stdlib/epoch-helpers';
|
|
47
|
+
import { computeCheckpointOutHash } from '@aztec/stdlib/messaging';
|
|
48
|
+
import { CheckpointHeader } from '@aztec/stdlib/rollup';
|
|
49
|
+
import { AppendOnlyTreeSnapshot } from '@aztec/stdlib/trees';
|
|
50
|
+
import { BlockHeader, TxExecutionResult, TxHash, TxReceipt, TxStatus } from '@aztec/stdlib/tx';
|
|
24
51
|
import type { UInt64 } from '@aztec/stdlib/types';
|
|
25
52
|
|
|
26
53
|
/**
|
|
@@ -28,30 +55,123 @@ import type { UInt64 } from '@aztec/stdlib/types';
|
|
|
28
55
|
*/
|
|
29
56
|
export class MockL2BlockSource implements L2BlockSource, ContractDataSource {
|
|
30
57
|
protected l2Blocks: L2Block[] = [];
|
|
58
|
+
protected checkpointList: Checkpoint[] = [];
|
|
31
59
|
|
|
32
60
|
private provenBlockNumber: number = 0;
|
|
33
61
|
private finalizedBlockNumber: number = 0;
|
|
34
62
|
private checkpointedBlockNumber: number = 0;
|
|
63
|
+
private proposedCheckpointBlockNumber: number = 0;
|
|
64
|
+
|
|
65
|
+
private initialHeader: BlockHeader = BlockHeader.empty();
|
|
66
|
+
private initialHeaderHash: BlockHash = GENESIS_BLOCK_HEADER_HASH;
|
|
67
|
+
private genesisArchiveRoot?: Fr;
|
|
68
|
+
private genesisBlock?: L2Block;
|
|
35
69
|
|
|
36
70
|
private log = createLogger('archiver:mock_l2_block_source');
|
|
37
71
|
|
|
72
|
+
/** Returns the initial header used to synthesize block 0. */
|
|
73
|
+
public getInitialHeader(): BlockHeader {
|
|
74
|
+
return this.initialHeader;
|
|
75
|
+
}
|
|
76
|
+
|
|
77
|
+
/**
|
|
78
|
+
* Sets the initial header used to synthesize block 0. Tests that wire up a real
|
|
79
|
+
* world-state should call this with `worldState.getInitialHeader()` so the L2BlockStream
|
|
80
|
+
* agrees on the genesis hash on both sides. Precomputes and caches the header hash so
|
|
81
|
+
* `getGenesisBlockHash()` can return synchronously.
|
|
82
|
+
*/
|
|
83
|
+
public async setInitialHeader(header: BlockHeader): Promise<void> {
|
|
84
|
+
this.initialHeader = header;
|
|
85
|
+
this.initialHeaderHash = await header.hash();
|
|
86
|
+
this.genesisBlock = undefined;
|
|
87
|
+
}
|
|
88
|
+
|
|
89
|
+
/**
|
|
90
|
+
* Returns the precomputed hash of the genesis block header. Defaults to the static
|
|
91
|
+
* {@link GENESIS_BLOCK_HEADER_HASH} unless {@link setInitialHeader} has been called with a
|
|
92
|
+
* custom header.
|
|
93
|
+
*/
|
|
94
|
+
public getGenesisBlockHash(): BlockHash {
|
|
95
|
+
return this.initialHeaderHash;
|
|
96
|
+
}
|
|
97
|
+
|
|
98
|
+
/**
|
|
99
|
+
* Sets the post-genesis archive root used to synthesize block 0. Mirrors the real archiver,
|
|
100
|
+
* whose synthetic block 0 carries `new AppendOnlyTreeSnapshot(genesisArchiveRoot, 1)` rather
|
|
101
|
+
* than `AppendOnlyTreeSnapshot.empty()`. Tests wiring up a real world-state should set this so
|
|
102
|
+
* archive-based block lookups against the mock match production semantics.
|
|
103
|
+
*/
|
|
104
|
+
public setGenesisArchiveRoot(root: Fr): void {
|
|
105
|
+
this.genesisArchiveRoot = root;
|
|
106
|
+
this.genesisBlock = undefined;
|
|
107
|
+
}
|
|
108
|
+
|
|
109
|
+
private getGenesisBlock(): L2Block {
|
|
110
|
+
if (this.genesisBlock) {
|
|
111
|
+
return this.genesisBlock;
|
|
112
|
+
}
|
|
113
|
+
const archive = this.genesisArchiveRoot
|
|
114
|
+
? new AppendOnlyTreeSnapshot(this.genesisArchiveRoot, 1)
|
|
115
|
+
: AppendOnlyTreeSnapshot.empty();
|
|
116
|
+
return (this.genesisBlock = new L2Block(
|
|
117
|
+
archive,
|
|
118
|
+
this.initialHeader,
|
|
119
|
+
Body.empty(),
|
|
120
|
+
CheckpointNumber.ZERO,
|
|
121
|
+
IndexWithinCheckpoint(0),
|
|
122
|
+
));
|
|
123
|
+
}
|
|
124
|
+
|
|
125
|
+
/** Creates blocks grouped into single-block checkpoints. */
|
|
38
126
|
public async createBlocks(numBlocks: number) {
|
|
39
|
-
|
|
40
|
-
|
|
41
|
-
|
|
42
|
-
|
|
127
|
+
await this.createCheckpoints(numBlocks, 1);
|
|
128
|
+
}
|
|
129
|
+
|
|
130
|
+
public getCheckpointNumber(): Promise<CheckpointNumber> {
|
|
131
|
+
return Promise.resolve(
|
|
132
|
+
this.checkpointList.length === 0 ? CheckpointNumber.ZERO : CheckpointNumber(this.checkpointList.length),
|
|
133
|
+
);
|
|
134
|
+
}
|
|
135
|
+
|
|
136
|
+
/** Creates checkpoints, each containing `blocksPerCheckpoint` blocks. */
|
|
137
|
+
public async createCheckpoints(numCheckpoints: number, blocksPerCheckpoint: number = 1) {
|
|
138
|
+
for (let c = 0; c < numCheckpoints; c++) {
|
|
139
|
+
const checkpointNum = CheckpointNumber(this.checkpointList.length + 1);
|
|
140
|
+
const startBlockNum = this.l2Blocks.length + 1;
|
|
141
|
+
const slotNumber = SlotNumber(Number(checkpointNum));
|
|
142
|
+
const checkpoint = await Checkpoint.random(checkpointNum, {
|
|
143
|
+
numBlocks: blocksPerCheckpoint,
|
|
144
|
+
startBlockNumber: startBlockNum,
|
|
145
|
+
slotNumber,
|
|
146
|
+
checkpointNumber: checkpointNum,
|
|
147
|
+
});
|
|
148
|
+
this.checkpointList.push(checkpoint);
|
|
149
|
+
this.l2Blocks.push(...checkpoint.blocks);
|
|
43
150
|
}
|
|
44
151
|
|
|
45
|
-
this.log.verbose(
|
|
152
|
+
this.log.verbose(
|
|
153
|
+
`Created ${numCheckpoints} checkpoints with ${blocksPerCheckpoint} blocks each in the mock L2 block source`,
|
|
154
|
+
);
|
|
46
155
|
}
|
|
47
156
|
|
|
48
|
-
public
|
|
157
|
+
public addProposedBlocks(blocks: L2Block[]) {
|
|
49
158
|
this.l2Blocks.push(...blocks);
|
|
50
|
-
this.log.verbose(`Added ${blocks.length} blocks to the mock L2 block source`);
|
|
159
|
+
this.log.verbose(`Added ${blocks.length} proposed blocks to the mock L2 block source`);
|
|
51
160
|
}
|
|
52
161
|
|
|
53
162
|
public removeBlocks(numBlocks: number) {
|
|
54
163
|
this.l2Blocks = this.l2Blocks.slice(0, -numBlocks);
|
|
164
|
+
const maxBlockNum = this.l2Blocks.length;
|
|
165
|
+
// Remove any checkpoint whose last block is beyond the remaining blocks.
|
|
166
|
+
this.checkpointList = this.checkpointList.filter(c => {
|
|
167
|
+
const lastBlockNum = c.blocks[0].number + c.blocks.length - 1;
|
|
168
|
+
return lastBlockNum <= maxBlockNum;
|
|
169
|
+
});
|
|
170
|
+
// Keep tip numbers consistent with remaining blocks.
|
|
171
|
+
this.checkpointedBlockNumber = Math.min(this.checkpointedBlockNumber, maxBlockNum);
|
|
172
|
+
this.proposedCheckpointBlockNumber = Math.min(this.proposedCheckpointBlockNumber, maxBlockNum);
|
|
173
|
+
this.provenBlockNumber = Math.min(this.provenBlockNumber, maxBlockNum);
|
|
174
|
+
this.finalizedBlockNumber = Math.min(this.finalizedBlockNumber, maxBlockNum);
|
|
55
175
|
this.log.verbose(`Removed ${numBlocks} blocks from the mock L2 block source`);
|
|
56
176
|
}
|
|
57
177
|
|
|
@@ -66,8 +186,42 @@ export class MockL2BlockSource implements L2BlockSource, ContractDataSource {
|
|
|
66
186
|
this.finalizedBlockNumber = finalizedBlockNumber;
|
|
67
187
|
}
|
|
68
188
|
|
|
189
|
+
public setProposedCheckpointBlockNumber(blockNumber: number) {
|
|
190
|
+
this.proposedCheckpointBlockNumber = blockNumber;
|
|
191
|
+
}
|
|
192
|
+
|
|
69
193
|
public setCheckpointedBlockNumber(checkpointedBlockNumber: number) {
|
|
194
|
+
const prevCheckpointed = this.checkpointedBlockNumber;
|
|
70
195
|
this.checkpointedBlockNumber = checkpointedBlockNumber;
|
|
196
|
+
// Proposed checkpoint is always at least as advanced as checkpointed
|
|
197
|
+
if (this.proposedCheckpointBlockNumber < checkpointedBlockNumber) {
|
|
198
|
+
this.proposedCheckpointBlockNumber = checkpointedBlockNumber;
|
|
199
|
+
}
|
|
200
|
+
// Auto-create single-block checkpoints for newly checkpointed blocks that don't have one yet.
|
|
201
|
+
// This handles blocks added via addProposedBlocks that are now being marked as checkpointed.
|
|
202
|
+
const newCheckpoints: Checkpoint[] = [];
|
|
203
|
+
for (let blockNum = prevCheckpointed + 1; blockNum <= checkpointedBlockNumber; blockNum++) {
|
|
204
|
+
const block = this.l2Blocks[blockNum - 1];
|
|
205
|
+
if (!block) {
|
|
206
|
+
continue;
|
|
207
|
+
}
|
|
208
|
+
if (this.checkpointList.some(c => c.blocks.some(b => b.number === block.number))) {
|
|
209
|
+
continue;
|
|
210
|
+
}
|
|
211
|
+
const checkpointNum = CheckpointNumber(this.checkpointList.length + newCheckpoints.length + 1);
|
|
212
|
+
const checkpoint = new Checkpoint(
|
|
213
|
+
block.archive,
|
|
214
|
+
CheckpointHeader.random({ slotNumber: block.header.globalVariables.slotNumber }),
|
|
215
|
+
[block],
|
|
216
|
+
checkpointNum,
|
|
217
|
+
);
|
|
218
|
+
newCheckpoints.push(checkpoint);
|
|
219
|
+
}
|
|
220
|
+
// Insert new checkpoints in order by number.
|
|
221
|
+
if (newCheckpoints.length > 0) {
|
|
222
|
+
this.checkpointList.push(...newCheckpoints);
|
|
223
|
+
this.checkpointList.sort((a, b) => a.number - b.number);
|
|
224
|
+
}
|
|
71
225
|
}
|
|
72
226
|
|
|
73
227
|
/**
|
|
@@ -90,176 +244,112 @@ export class MockL2BlockSource implements L2BlockSource, ContractDataSource {
|
|
|
90
244
|
* Gets the number of the latest L2 block processed by the block source implementation.
|
|
91
245
|
* @returns In this mock instance, returns the number of L2 blocks that we've mocked.
|
|
92
246
|
*/
|
|
93
|
-
public getBlockNumber()
|
|
94
|
-
|
|
95
|
-
|
|
96
|
-
|
|
97
|
-
|
|
98
|
-
return Promise.resolve(BlockNumber(this.provenBlockNumber));
|
|
99
|
-
}
|
|
100
|
-
|
|
101
|
-
public getCheckpointedBlock(number: BlockNumber): Promise<CheckpointedL2Block | undefined> {
|
|
102
|
-
if (number > this.checkpointedBlockNumber) {
|
|
103
|
-
return Promise.resolve(undefined);
|
|
247
|
+
public getBlockNumber(): Promise<BlockNumber>;
|
|
248
|
+
public getBlockNumber(query: BlockQuery): Promise<BlockNumber | undefined>;
|
|
249
|
+
public async getBlockNumber(query?: BlockQuery): Promise<BlockNumber | undefined> {
|
|
250
|
+
if (!query) {
|
|
251
|
+
return BlockNumber(this.l2Blocks.length);
|
|
104
252
|
}
|
|
105
|
-
|
|
106
|
-
|
|
107
|
-
return Promise.resolve(undefined);
|
|
253
|
+
if ('number' in query) {
|
|
254
|
+
return query.number;
|
|
108
255
|
}
|
|
109
|
-
|
|
110
|
-
|
|
111
|
-
block.toL2Block(),
|
|
112
|
-
new L1PublishedData(BigInt(number), BigInt(number), `0x${number.toString(16).padStart(64, '0')}`),
|
|
113
|
-
[],
|
|
114
|
-
);
|
|
115
|
-
return Promise.resolve(checkpointedBlock);
|
|
116
|
-
}
|
|
117
|
-
|
|
118
|
-
public async getCheckpointedBlocks(
|
|
119
|
-
from: BlockNumber,
|
|
120
|
-
limit: number,
|
|
121
|
-
_proven?: boolean,
|
|
122
|
-
): Promise<CheckpointedL2Block[]> {
|
|
123
|
-
const result: CheckpointedL2Block[] = [];
|
|
124
|
-
for (let i = 0; i < limit; i++) {
|
|
125
|
-
const blockNum = from + i;
|
|
126
|
-
if (blockNum > this.checkpointedBlockNumber) {
|
|
127
|
-
break;
|
|
128
|
-
}
|
|
129
|
-
const block = await this.getCheckpointedBlock(BlockNumber(blockNum));
|
|
130
|
-
if (block) {
|
|
131
|
-
result.push(block);
|
|
132
|
-
}
|
|
256
|
+
if ('tag' in query) {
|
|
257
|
+
return BlockNumber(this.resolveBlockTag(query.tag));
|
|
133
258
|
}
|
|
134
|
-
|
|
259
|
+
const block = await this.getBlock(query);
|
|
260
|
+
return block ? block.header.globalVariables.blockNumber : undefined;
|
|
135
261
|
}
|
|
136
262
|
|
|
137
|
-
|
|
138
|
-
|
|
139
|
-
* @param number - The block number to return (inclusive).
|
|
140
|
-
* @returns The requested L2 block.
|
|
141
|
-
*/
|
|
142
|
-
public getBlock(number: number) {
|
|
143
|
-
return Promise.resolve(this.l2Blocks[number - 1]);
|
|
263
|
+
public getProposedCheckpointL2BlockNumber() {
|
|
264
|
+
return Promise.resolve(BlockNumber(this.proposedCheckpointBlockNumber));
|
|
144
265
|
}
|
|
145
266
|
|
|
146
|
-
|
|
147
|
-
|
|
148
|
-
|
|
149
|
-
|
|
150
|
-
|
|
151
|
-
|
|
152
|
-
const block = this.l2Blocks[number - 1];
|
|
153
|
-
return Promise.resolve(block?.toL2Block());
|
|
267
|
+
public getCheckpoint(query: CheckpointQuery): Promise<PublishedCheckpoint | undefined> {
|
|
268
|
+
const checkpoint = this.resolveCheckpointQuery(query);
|
|
269
|
+
if (!checkpoint) {
|
|
270
|
+
return Promise.resolve(undefined);
|
|
271
|
+
}
|
|
272
|
+
return Promise.resolve(new PublishedCheckpoint(checkpoint, this.mockL1DataForCheckpoint(checkpoint), []));
|
|
154
273
|
}
|
|
155
274
|
|
|
156
|
-
|
|
157
|
-
|
|
158
|
-
* @param from - Number of the first block to return (inclusive).
|
|
159
|
-
* @param limit - The maximum number of blocks to return.
|
|
160
|
-
* @returns The requested mocked L2 blocks.
|
|
161
|
-
*/
|
|
162
|
-
public getBlocks(from: number, limit: number, proven?: boolean) {
|
|
275
|
+
public getCheckpoints(query: CheckpointsQuery): Promise<PublishedCheckpoint[]> {
|
|
276
|
+
const checkpoints = this.resolveCheckpointsQuery(query);
|
|
163
277
|
return Promise.resolve(
|
|
164
|
-
this.
|
|
165
|
-
.slice(from - 1, from - 1 + limit)
|
|
166
|
-
.filter(b => !proven || this.provenBlockNumber === undefined || b.number <= this.provenBlockNumber),
|
|
278
|
+
checkpoints.map(checkpoint => new PublishedCheckpoint(checkpoint, this.mockL1DataForCheckpoint(checkpoint), [])),
|
|
167
279
|
);
|
|
168
280
|
}
|
|
169
281
|
|
|
170
|
-
public
|
|
171
|
-
|
|
172
|
-
return
|
|
282
|
+
public getCheckpointByArchive(archive: Fr): Promise<Checkpoint | undefined> {
|
|
283
|
+
const checkpoint = this.checkpointList.find(c => c.archive.root.equals(archive));
|
|
284
|
+
return Promise.resolve(checkpoint);
|
|
173
285
|
}
|
|
174
286
|
|
|
175
|
-
public
|
|
176
|
-
|
|
177
|
-
|
|
287
|
+
public getCheckpointData(query: CheckpointQuery): Promise<CheckpointData | undefined> {
|
|
288
|
+
const checkpoint = this.resolveCheckpointQuery(query);
|
|
289
|
+
if (!checkpoint) {
|
|
290
|
+
return Promise.resolve(undefined);
|
|
291
|
+
}
|
|
292
|
+
return Promise.resolve(this.checkpointToData(checkpoint));
|
|
178
293
|
}
|
|
179
294
|
|
|
180
|
-
public
|
|
181
|
-
const
|
|
182
|
-
return
|
|
183
|
-
PublishedL2Block.fromFields({
|
|
184
|
-
block,
|
|
185
|
-
l1: new L1PublishedData(BigInt(block.number), BigInt(block.number), Buffer32.random().toString()),
|
|
186
|
-
attestations: [],
|
|
187
|
-
}),
|
|
188
|
-
);
|
|
295
|
+
public getCheckpointsData(query: CheckpointsQuery): Promise<CheckpointData[]> {
|
|
296
|
+
const checkpoints = this.resolveCheckpointsQuery(query);
|
|
297
|
+
return Promise.resolve(checkpoints.map(c => this.checkpointToData(c)));
|
|
189
298
|
}
|
|
190
299
|
|
|
191
|
-
|
|
192
|
-
|
|
193
|
-
|
|
300
|
+
private checkpointToData(checkpoint: Checkpoint): CheckpointData {
|
|
301
|
+
return {
|
|
302
|
+
checkpointNumber: checkpoint.number,
|
|
303
|
+
header: checkpoint.header,
|
|
304
|
+
archive: checkpoint.archive,
|
|
305
|
+
checkpointOutHash: computeCheckpointOutHash(
|
|
306
|
+
checkpoint.blocks.map(b => b.body.txEffects.map(tx => tx.l2ToL1Msgs)),
|
|
307
|
+
),
|
|
308
|
+
startBlock: checkpoint.blocks[0].number,
|
|
309
|
+
blockCount: checkpoint.blocks.length,
|
|
310
|
+
feeAssetPriceModifier: checkpoint.feeAssetPriceModifier,
|
|
311
|
+
attestations: [],
|
|
312
|
+
l1: this.mockL1DataForCheckpoint(checkpoint),
|
|
313
|
+
};
|
|
194
314
|
}
|
|
195
315
|
|
|
196
|
-
|
|
197
|
-
|
|
198
|
-
|
|
199
|
-
if (hash.equals(blockHash)) {
|
|
200
|
-
return PublishedL2Block.fromFields({
|
|
201
|
-
block,
|
|
202
|
-
l1: new L1PublishedData(BigInt(block.number), BigInt(block.number), Buffer32.random().toString()),
|
|
203
|
-
attestations: [],
|
|
204
|
-
});
|
|
205
|
-
}
|
|
316
|
+
private resolveCheckpointQuery(query: CheckpointQuery): Checkpoint | undefined {
|
|
317
|
+
if ('number' in query) {
|
|
318
|
+
return this.checkpointList[query.number - 1];
|
|
206
319
|
}
|
|
207
|
-
|
|
208
|
-
|
|
209
|
-
|
|
210
|
-
public getPublishedBlockByArchive(archive: Fr): Promise<PublishedL2Block | undefined> {
|
|
211
|
-
const block = this.l2Blocks.find(b => b.archive.root.equals(archive));
|
|
212
|
-
if (!block) {
|
|
213
|
-
return Promise.resolve(undefined);
|
|
320
|
+
if ('slot' in query) {
|
|
321
|
+
return this.checkpointList.find(c => c.header.slotNumber === query.slot);
|
|
214
322
|
}
|
|
215
|
-
|
|
216
|
-
|
|
217
|
-
|
|
218
|
-
|
|
219
|
-
|
|
220
|
-
|
|
221
|
-
|
|
222
|
-
|
|
223
|
-
|
|
224
|
-
|
|
225
|
-
|
|
226
|
-
|
|
227
|
-
|
|
228
|
-
return
|
|
323
|
+
switch (query.tag) {
|
|
324
|
+
case 'checkpointed':
|
|
325
|
+
return this.checkpointList[this.checkpointList.length - 1];
|
|
326
|
+
case 'proven': {
|
|
327
|
+
const provenCheckpoint = this.checkpointList.filter(c =>
|
|
328
|
+
c.blocks.some(b => b.number <= this.provenBlockNumber),
|
|
329
|
+
);
|
|
330
|
+
return provenCheckpoint.at(-1);
|
|
331
|
+
}
|
|
332
|
+
case 'finalized': {
|
|
333
|
+
const finalizedCheckpoint = this.checkpointList.filter(c =>
|
|
334
|
+
c.blocks.some(b => b.number <= this.finalizedBlockNumber),
|
|
335
|
+
);
|
|
336
|
+
return finalizedCheckpoint.at(-1);
|
|
229
337
|
}
|
|
230
338
|
}
|
|
231
|
-
return undefined;
|
|
232
|
-
}
|
|
233
|
-
|
|
234
|
-
public getBlockHeaderByArchive(archive: Fr): Promise<BlockHeader | undefined> {
|
|
235
|
-
const block = this.l2Blocks.find(b => b.archive.root.equals(archive));
|
|
236
|
-
return Promise.resolve(block?.getBlockHeader());
|
|
237
|
-
}
|
|
238
|
-
|
|
239
|
-
getBlockHeader(number: number | 'latest'): Promise<BlockHeader | undefined> {
|
|
240
|
-
return Promise.resolve(this.l2Blocks.at(typeof number === 'number' ? number - 1 : -1)?.getBlockHeader());
|
|
241
339
|
}
|
|
242
340
|
|
|
243
|
-
|
|
244
|
-
|
|
245
|
-
|
|
341
|
+
private resolveCheckpointsQuery(query: CheckpointsQuery): Checkpoint[] {
|
|
342
|
+
if ('from' in query) {
|
|
343
|
+
return this.checkpointList.slice(query.from - 1, query.from - 1 + query.limit);
|
|
344
|
+
}
|
|
345
|
+
return this.getCheckpointsInEpoch(query.epoch);
|
|
246
346
|
}
|
|
247
347
|
|
|
248
|
-
|
|
249
|
-
const
|
|
250
|
-
const [start, end] = getSlotRangeForEpoch(epochNumber, { epochDuration });
|
|
251
|
-
const blocks = this.l2Blocks.filter(b => {
|
|
252
|
-
const slot = b.header.globalVariables.slotNumber;
|
|
253
|
-
return slot >= start && slot <= end;
|
|
254
|
-
});
|
|
348
|
+
getBlocksForSlot(slotNumber: SlotNumber): Promise<L2Block[]> {
|
|
349
|
+
const blocks = this.l2Blocks.filter(b => b.header.globalVariables.slotNumber === slotNumber);
|
|
255
350
|
return Promise.resolve(blocks);
|
|
256
351
|
}
|
|
257
352
|
|
|
258
|
-
async getBlockHeadersForEpoch(epochNumber: EpochNumber): Promise<BlockHeader[]> {
|
|
259
|
-
const blocks = await this.getBlocksForEpoch(epochNumber);
|
|
260
|
-
return blocks.map(b => b.getBlockHeader());
|
|
261
|
-
}
|
|
262
|
-
|
|
263
353
|
/**
|
|
264
354
|
* Gets a tx effect.
|
|
265
355
|
* @param txHash - The hash of the tx corresponding to the tx effect.
|
|
@@ -276,8 +366,8 @@ export class MockL2BlockSource implements L2BlockSource, ContractDataSource {
|
|
|
276
366
|
return {
|
|
277
367
|
data: txEffect,
|
|
278
368
|
l2BlockNumber: block.number,
|
|
279
|
-
l2BlockHash:
|
|
280
|
-
txIndexInBlock: block.body.txEffects.
|
|
369
|
+
l2BlockHash: await block.hash(),
|
|
370
|
+
txIndexInBlock: block.body.txEffects.findIndex(t => t.txHash.equals(txHash)),
|
|
281
371
|
};
|
|
282
372
|
}
|
|
283
373
|
|
|
@@ -290,13 +380,16 @@ export class MockL2BlockSource implements L2BlockSource, ContractDataSource {
|
|
|
290
380
|
for (const block of this.l2Blocks) {
|
|
291
381
|
for (const txEffect of block.body.txEffects) {
|
|
292
382
|
if (txEffect.txHash.equals(txHash)) {
|
|
383
|
+
// In mock, assume all txs are checkpointed with successful execution
|
|
293
384
|
return new TxReceipt(
|
|
294
385
|
txHash,
|
|
295
|
-
TxStatus.
|
|
296
|
-
|
|
386
|
+
TxStatus.CHECKPOINTED,
|
|
387
|
+
TxExecutionResult.SUCCESS,
|
|
388
|
+
undefined,
|
|
297
389
|
txEffect.transactionFee.toBigInt(),
|
|
298
|
-
|
|
390
|
+
await block.hash(),
|
|
299
391
|
block.number,
|
|
392
|
+
getEpochAtSlot(block.slot, EmptyL1RollupConstants),
|
|
300
393
|
);
|
|
301
394
|
}
|
|
302
395
|
}
|
|
@@ -305,53 +398,77 @@ export class MockL2BlockSource implements L2BlockSource, ContractDataSource {
|
|
|
305
398
|
}
|
|
306
399
|
|
|
307
400
|
async getL2Tips(): Promise<L2Tips> {
|
|
308
|
-
const [latest, proven, finalized, checkpointed] = [
|
|
401
|
+
const [latest, proven, finalized, checkpointed, proposedCheckpoint] = [
|
|
309
402
|
await this.getBlockNumber(),
|
|
310
|
-
|
|
403
|
+
this.provenBlockNumber,
|
|
311
404
|
this.finalizedBlockNumber,
|
|
312
405
|
this.checkpointedBlockNumber,
|
|
406
|
+
await this.getProposedCheckpointL2BlockNumber(),
|
|
313
407
|
] as const;
|
|
314
408
|
|
|
315
409
|
const latestBlock = this.l2Blocks[latest - 1];
|
|
316
410
|
const provenBlock = this.l2Blocks[proven - 1];
|
|
317
411
|
const finalizedBlock = this.l2Blocks[finalized - 1];
|
|
318
412
|
const checkpointedBlock = this.l2Blocks[checkpointed - 1];
|
|
413
|
+
const proposedCheckpointBlock = this.l2Blocks[proposedCheckpoint - 1];
|
|
414
|
+
|
|
415
|
+
// For genesis tips (block number 0) report the dynamic initial header hash so consumers
|
|
416
|
+
// running L2BlockStream against this mock agree at block 0 with their local tip store.
|
|
417
|
+
const genesisHash = (await this.initialHeader.hash()).toString();
|
|
418
|
+
const tipHash = async (block: L2Block | undefined, number: number): Promise<string> => {
|
|
419
|
+
if (block) {
|
|
420
|
+
return (await block.hash()).toString();
|
|
421
|
+
}
|
|
422
|
+
return number === 0 ? genesisHash : '';
|
|
423
|
+
};
|
|
319
424
|
|
|
320
425
|
const latestBlockId = {
|
|
321
426
|
number: BlockNumber(latest),
|
|
322
|
-
hash:
|
|
427
|
+
hash: await tipHash(latestBlock, latest),
|
|
323
428
|
};
|
|
324
429
|
const provenBlockId = {
|
|
325
430
|
number: BlockNumber(proven),
|
|
326
|
-
hash:
|
|
431
|
+
hash: await tipHash(provenBlock, proven),
|
|
327
432
|
};
|
|
328
433
|
const finalizedBlockId = {
|
|
329
434
|
number: BlockNumber(finalized),
|
|
330
|
-
hash:
|
|
435
|
+
hash: await tipHash(finalizedBlock, finalized),
|
|
331
436
|
};
|
|
332
437
|
const checkpointedBlockId = {
|
|
333
438
|
number: BlockNumber(checkpointed),
|
|
334
|
-
hash:
|
|
439
|
+
hash: await tipHash(checkpointedBlock, checkpointed),
|
|
440
|
+
};
|
|
441
|
+
const proposedCheckpointBlockId = {
|
|
442
|
+
number: BlockNumber(proposedCheckpoint),
|
|
443
|
+
hash: await tipHash(proposedCheckpointBlock, proposedCheckpoint),
|
|
335
444
|
};
|
|
336
445
|
|
|
337
|
-
const makeTipId = (blockId: typeof latestBlockId) =>
|
|
338
|
-
|
|
339
|
-
|
|
340
|
-
|
|
446
|
+
const makeTipId = (blockId: typeof latestBlockId) => {
|
|
447
|
+
const checkpointNumber = this.findCheckpointNumberForBlock(blockId.number) ?? CheckpointNumber(0);
|
|
448
|
+
// Match production semantics: checkpoint 0 is fully synthetic (no real checkpoint header
|
|
449
|
+
// exists at 0), so its hash stays at the protocol constant `GENESIS_CHECKPOINT_HEADER_HASH`
|
|
450
|
+
// even though the block-0 hash is dynamic. See L2TipsCache for the production path.
|
|
451
|
+
const hash = checkpointNumber === 0 ? GENESIS_CHECKPOINT_HEADER_HASH.toString() : blockId.hash;
|
|
452
|
+
return {
|
|
453
|
+
block: blockId,
|
|
454
|
+
checkpoint: { number: checkpointNumber, hash },
|
|
455
|
+
};
|
|
456
|
+
};
|
|
341
457
|
|
|
342
458
|
return {
|
|
343
459
|
proposed: latestBlockId,
|
|
344
460
|
checkpointed: makeTipId(checkpointedBlockId),
|
|
345
461
|
proven: makeTipId(provenBlockId),
|
|
346
462
|
finalized: makeTipId(finalizedBlockId),
|
|
463
|
+
proposedCheckpoint: makeTipId(proposedCheckpointBlockId),
|
|
347
464
|
};
|
|
348
465
|
}
|
|
349
466
|
|
|
350
|
-
|
|
467
|
+
getSyncedL2EpochNumber(): Promise<EpochNumber> {
|
|
351
468
|
throw new Error('Method not implemented.');
|
|
352
469
|
}
|
|
353
470
|
|
|
354
|
-
|
|
471
|
+
getSyncedL2SlotNumber(): Promise<SlotNumber> {
|
|
355
472
|
throw new Error('Method not implemented.');
|
|
356
473
|
}
|
|
357
474
|
|
|
@@ -363,8 +480,12 @@ export class MockL2BlockSource implements L2BlockSource, ContractDataSource {
|
|
|
363
480
|
return Promise.resolve(EmptyL1RollupConstants);
|
|
364
481
|
}
|
|
365
482
|
|
|
483
|
+
isPruneDueAtSlot(_slot: SlotNumber): Promise<boolean> {
|
|
484
|
+
return Promise.resolve(false);
|
|
485
|
+
}
|
|
486
|
+
|
|
366
487
|
getGenesisValues(): Promise<{ genesisArchiveRoot: Fr }> {
|
|
367
|
-
return Promise.resolve({ genesisArchiveRoot: new Fr(GENESIS_ARCHIVE_ROOT) });
|
|
488
|
+
return Promise.resolve({ genesisArchiveRoot: this.genesisArchiveRoot ?? new Fr(GENESIS_ARCHIVE_ROOT) });
|
|
368
489
|
}
|
|
369
490
|
|
|
370
491
|
getL1Timestamp(): Promise<bigint> {
|
|
@@ -417,6 +538,95 @@ export class MockL2BlockSource implements L2BlockSource, ContractDataSource {
|
|
|
417
538
|
return Promise.resolve();
|
|
418
539
|
}
|
|
419
540
|
|
|
541
|
+
async getBlock(query: BlockQuery): Promise<L2Block | undefined> {
|
|
542
|
+
if ('number' in query) {
|
|
543
|
+
if (query.number === 0) {
|
|
544
|
+
return this.getGenesisBlock();
|
|
545
|
+
}
|
|
546
|
+
return this.l2Blocks[query.number - 1];
|
|
547
|
+
}
|
|
548
|
+
if ('hash' in query) {
|
|
549
|
+
const genesis = this.getGenesisBlock();
|
|
550
|
+
if ((await genesis.hash()).equals(query.hash)) {
|
|
551
|
+
return genesis;
|
|
552
|
+
}
|
|
553
|
+
for (const b of this.l2Blocks) {
|
|
554
|
+
const hash = await b.hash();
|
|
555
|
+
if (hash.equals(query.hash)) {
|
|
556
|
+
return b;
|
|
557
|
+
}
|
|
558
|
+
}
|
|
559
|
+
return undefined;
|
|
560
|
+
}
|
|
561
|
+
if ('archive' in query) {
|
|
562
|
+
const genesis = this.getGenesisBlock();
|
|
563
|
+
if (genesis.archive.root.equals(query.archive)) {
|
|
564
|
+
return genesis;
|
|
565
|
+
}
|
|
566
|
+
return this.l2Blocks.find(b => b.archive.root.equals(query.archive));
|
|
567
|
+
}
|
|
568
|
+
const number = this.resolveBlockTag(query.tag);
|
|
569
|
+
if (number === 0) {
|
|
570
|
+
return this.getGenesisBlock();
|
|
571
|
+
}
|
|
572
|
+
return this.l2Blocks[number - 1];
|
|
573
|
+
}
|
|
574
|
+
|
|
575
|
+
private resolveBlockTag(tag: BlockTag): number {
|
|
576
|
+
switch (tag) {
|
|
577
|
+
case 'latest':
|
|
578
|
+
case 'proposed':
|
|
579
|
+
return this.l2Blocks.length;
|
|
580
|
+
case 'checkpointed':
|
|
581
|
+
return this.checkpointedBlockNumber;
|
|
582
|
+
case 'proven':
|
|
583
|
+
return this.provenBlockNumber;
|
|
584
|
+
case 'finalized':
|
|
585
|
+
return this.finalizedBlockNumber;
|
|
586
|
+
}
|
|
587
|
+
}
|
|
588
|
+
|
|
589
|
+
getBlocks(query: BlocksQuery): Promise<L2Block[]> {
|
|
590
|
+
let blocks: L2Block[];
|
|
591
|
+
if ('from' in query) {
|
|
592
|
+
blocks = this.l2Blocks.slice(query.from - 1, query.from - 1 + query.limit);
|
|
593
|
+
} else {
|
|
594
|
+
const epochCheckpoints = this.getCheckpointsInEpoch(query.epoch);
|
|
595
|
+
blocks = epochCheckpoints.flatMap(c => c.blocks);
|
|
596
|
+
}
|
|
597
|
+
if (query.onlyCheckpointed) {
|
|
598
|
+
blocks = blocks.filter(b => b.header.globalVariables.blockNumber <= this.checkpointedBlockNumber);
|
|
599
|
+
}
|
|
600
|
+
return Promise.resolve(blocks);
|
|
601
|
+
}
|
|
602
|
+
|
|
603
|
+
async getBlockData(query: BlockQuery): Promise<BlockData | undefined> {
|
|
604
|
+
const block = await this.getBlock(query);
|
|
605
|
+
if (!block) {
|
|
606
|
+
return undefined;
|
|
607
|
+
}
|
|
608
|
+
return {
|
|
609
|
+
header: block.header,
|
|
610
|
+
archive: block.archive,
|
|
611
|
+
blockHash: await block.hash(),
|
|
612
|
+
checkpointNumber: block.checkpointNumber,
|
|
613
|
+
indexWithinCheckpoint: block.indexWithinCheckpoint,
|
|
614
|
+
};
|
|
615
|
+
}
|
|
616
|
+
|
|
617
|
+
async getBlocksData(query: BlocksQuery): Promise<BlockData[]> {
|
|
618
|
+
const blocks = await this.getBlocks(query);
|
|
619
|
+
return Promise.all(
|
|
620
|
+
blocks.map(async block => ({
|
|
621
|
+
header: block.header,
|
|
622
|
+
archive: block.archive,
|
|
623
|
+
blockHash: await block.hash(),
|
|
624
|
+
checkpointNumber: block.checkpointNumber,
|
|
625
|
+
indexWithinCheckpoint: block.indexWithinCheckpoint,
|
|
626
|
+
})),
|
|
627
|
+
);
|
|
628
|
+
}
|
|
629
|
+
|
|
420
630
|
isPendingChainInvalid(): Promise<boolean> {
|
|
421
631
|
return Promise.resolve(false);
|
|
422
632
|
}
|
|
@@ -424,4 +634,26 @@ export class MockL2BlockSource implements L2BlockSource, ContractDataSource {
|
|
|
424
634
|
getPendingChainValidationStatus(): Promise<ValidateCheckpointResult> {
|
|
425
635
|
return Promise.resolve({ valid: true });
|
|
426
636
|
}
|
|
637
|
+
|
|
638
|
+
getProposedCheckpointData(_query?: ProposedCheckpointQuery): Promise<ProposedCheckpointData | undefined> {
|
|
639
|
+
return Promise.resolve(undefined);
|
|
640
|
+
}
|
|
641
|
+
|
|
642
|
+
/** Returns checkpoints whose slot falls within the given epoch. */
|
|
643
|
+
private getCheckpointsInEpoch(epochNumber: EpochNumber): Checkpoint[] {
|
|
644
|
+
const epochDuration = DefaultL1ContractsConfig.aztecEpochDuration;
|
|
645
|
+
const [start, end] = getSlotRangeForEpoch(epochNumber, { epochDuration });
|
|
646
|
+
return this.checkpointList.filter(c => c.header.slotNumber >= start && c.header.slotNumber <= end);
|
|
647
|
+
}
|
|
648
|
+
|
|
649
|
+
/** Creates a mock L1PublishedData for a checkpoint. */
|
|
650
|
+
private mockL1DataForCheckpoint(checkpoint: Checkpoint): L1PublishedData {
|
|
651
|
+
return new L1PublishedData(BigInt(checkpoint.number), BigInt(checkpoint.number), Buffer32.random().toString());
|
|
652
|
+
}
|
|
653
|
+
|
|
654
|
+
/** Finds the checkpoint number for a block, or undefined if the block is not in any checkpoint. */
|
|
655
|
+
private findCheckpointNumberForBlock(blockNumber: BlockNumber): CheckpointNumber | undefined {
|
|
656
|
+
const checkpoint = this.checkpointList.find(c => c.blocks.some(b => b.number === blockNumber));
|
|
657
|
+
return checkpoint?.number;
|
|
658
|
+
}
|
|
427
659
|
}
|