@aztec/txe 0.0.1-commit.b655e406 → 0.0.1-commit.b6e433891
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/dest/bin/index.d.ts +1 -1
- package/dest/constants.d.ts +3 -0
- package/dest/constants.d.ts.map +1 -0
- package/dest/constants.js +2 -0
- package/dest/index.d.ts +1 -1
- package/dest/index.d.ts.map +1 -1
- package/dest/index.js +91 -56
- package/dest/oracle/interfaces.d.ts +33 -29
- package/dest/oracle/interfaces.d.ts.map +1 -1
- package/dest/oracle/txe_oracle_public_context.d.ts +16 -16
- package/dest/oracle/txe_oracle_public_context.d.ts.map +1 -1
- package/dest/oracle/txe_oracle_public_context.js +20 -22
- package/dest/oracle/txe_oracle_top_level_context.d.ts +37 -27
- package/dest/oracle/txe_oracle_top_level_context.d.ts.map +1 -1
- package/dest/oracle/txe_oracle_top_level_context.js +228 -106
- package/dest/rpc_translator.d.ts +98 -78
- package/dest/rpc_translator.d.ts.map +1 -1
- package/dest/rpc_translator.js +367 -171
- package/dest/state_machine/archiver.d.ts +21 -51
- package/dest/state_machine/archiver.d.ts.map +1 -1
- package/dest/state_machine/archiver.js +66 -99
- package/dest/state_machine/dummy_p2p_client.d.ts +21 -14
- package/dest/state_machine/dummy_p2p_client.d.ts.map +1 -1
- package/dest/state_machine/dummy_p2p_client.js +42 -22
- package/dest/state_machine/global_variable_builder.d.ts +6 -4
- package/dest/state_machine/global_variable_builder.d.ts.map +1 -1
- package/dest/state_machine/global_variable_builder.js +13 -1
- package/dest/state_machine/index.d.ts +9 -7
- package/dest/state_machine/index.d.ts.map +1 -1
- package/dest/state_machine/index.js +44 -23
- package/dest/state_machine/mock_epoch_cache.d.ts +30 -12
- package/dest/state_machine/mock_epoch_cache.d.ts.map +1 -1
- package/dest/state_machine/mock_epoch_cache.js +53 -15
- package/dest/state_machine/synchronizer.d.ts +6 -5
- package/dest/state_machine/synchronizer.d.ts.map +1 -1
- package/dest/state_machine/synchronizer.js +8 -7
- package/dest/txe_session.d.ts +27 -15
- package/dest/txe_session.d.ts.map +1 -1
- package/dest/txe_session.js +173 -56
- package/dest/util/encoding.d.ts +615 -16
- package/dest/util/encoding.d.ts.map +1 -1
- package/dest/util/encoding.js +1 -1
- package/dest/util/expected_failure_error.d.ts +1 -1
- package/dest/util/expected_failure_error.d.ts.map +1 -1
- package/dest/util/txe_account_store.d.ts +10 -0
- package/dest/util/txe_account_store.d.ts.map +1 -0
- package/dest/util/{txe_account_data_provider.js → txe_account_store.js} +1 -1
- package/dest/util/txe_public_contract_data_source.d.ts +8 -8
- package/dest/util/txe_public_contract_data_source.d.ts.map +1 -1
- package/dest/util/txe_public_contract_data_source.js +12 -29
- package/dest/utils/block_creation.d.ts +21 -6
- package/dest/utils/block_creation.d.ts.map +1 -1
- package/dest/utils/block_creation.js +38 -4
- package/dest/utils/tx_effect_creation.d.ts +3 -3
- package/dest/utils/tx_effect_creation.d.ts.map +1 -1
- package/dest/utils/tx_effect_creation.js +4 -7
- package/package.json +18 -17
- package/src/constants.ts +3 -0
- package/src/index.ts +103 -63
- package/src/oracle/interfaces.ts +36 -32
- package/src/oracle/txe_oracle_public_context.ts +21 -28
- package/src/oracle/txe_oracle_top_level_context.ts +272 -145
- package/src/rpc_translator.ts +422 -194
- package/src/state_machine/archiver.ts +63 -117
- package/src/state_machine/dummy_p2p_client.ts +59 -29
- package/src/state_machine/global_variable_builder.ts +22 -4
- package/src/state_machine/index.ts +64 -21
- package/src/state_machine/mock_epoch_cache.ts +67 -23
- package/src/state_machine/synchronizer.ts +9 -8
- package/src/txe_session.ts +239 -105
- package/src/util/encoding.ts +1 -1
- package/src/util/{txe_account_data_provider.ts → txe_account_store.ts} +1 -1
- package/src/util/txe_public_contract_data_source.ts +20 -45
- package/src/utils/block_creation.ts +49 -14
- package/src/utils/tx_effect_creation.ts +5 -12
- package/dest/util/txe_account_data_provider.d.ts +0 -10
- package/dest/util/txe_account_data_provider.d.ts.map +0 -1
- package/dest/util/txe_contract_data_provider.d.ts +0 -12
- package/dest/util/txe_contract_data_provider.d.ts.map +0 -1
- package/dest/util/txe_contract_data_provider.js +0 -22
- package/src/util/txe_contract_data_provider.ts +0 -36
|
@@ -1,151 +1,97 @@
|
|
|
1
|
-
import {
|
|
1
|
+
import { ArchiverDataSourceBase, ArchiverDataStoreUpdater, KVArchiverDataStore } from '@aztec/archiver';
|
|
2
2
|
import { GENESIS_ARCHIVE_ROOT } from '@aztec/constants';
|
|
3
|
+
import { CheckpointNumber, type EpochNumber, type SlotNumber } from '@aztec/foundation/branded-types';
|
|
4
|
+
import { Fr } from '@aztec/foundation/curves/bn254';
|
|
3
5
|
import type { EthAddress } from '@aztec/foundation/eth-address';
|
|
4
|
-
import { Fr } from '@aztec/foundation/fields';
|
|
5
6
|
import type { AztecAsyncKVStore } from '@aztec/kv-store';
|
|
6
|
-
import type {
|
|
7
|
-
import type {
|
|
8
|
-
import type { ContractInstanceWithAddress } from '@aztec/stdlib/contract';
|
|
7
|
+
import type { CheckpointId, L2BlockId, L2TipId, L2Tips, ValidateCheckpointResult } from '@aztec/stdlib/block';
|
|
8
|
+
import type { PublishedCheckpoint } from '@aztec/stdlib/checkpoint';
|
|
9
9
|
import type { L1RollupConstants } from '@aztec/stdlib/epoch-helpers';
|
|
10
|
-
import type { BlockHeader } from '@aztec/stdlib/tx';
|
|
11
|
-
import type { UInt64 } from '@aztec/stdlib/types';
|
|
12
10
|
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
public override async addBlocks(blocks: PublishedL2Block[]): Promise<boolean> {
|
|
22
|
-
const opResults = await Promise.all([
|
|
23
|
-
this.store.addLogs(blocks.map(block => block.block)),
|
|
24
|
-
this.store.addBlocks(blocks),
|
|
25
|
-
]);
|
|
26
|
-
|
|
27
|
-
return opResults.every(Boolean);
|
|
28
|
-
}
|
|
29
|
-
|
|
30
|
-
/**
|
|
31
|
-
* Gets the number of the latest L2 block processed by the block source implementation.
|
|
32
|
-
* @returns The number of the latest L2 block processed by the block source implementation.
|
|
33
|
-
*/
|
|
34
|
-
public getBlockNumber(): Promise<number> {
|
|
35
|
-
return this.store.getSynchedL2BlockNumber();
|
|
36
|
-
}
|
|
37
|
-
|
|
38
|
-
/**
|
|
39
|
-
* Gets the number of the latest L2 block proven seen by the block source implementation.
|
|
40
|
-
* @returns The number of the latest L2 block proven seen by the block source implementation.
|
|
41
|
-
*/
|
|
42
|
-
public getProvenBlockNumber(): Promise<number> {
|
|
43
|
-
return this.store.getSynchedL2BlockNumber();
|
|
44
|
-
}
|
|
45
|
-
|
|
46
|
-
/**
|
|
47
|
-
* Gets a published l2 block. If a negative number is passed, the block returned is the most recent.
|
|
48
|
-
* @param number - The block number to return (inclusive).
|
|
49
|
-
* @returns The requested L2 block.
|
|
50
|
-
*/
|
|
51
|
-
public override async getPublishedBlock(number: number): Promise<PublishedL2Block | undefined> {
|
|
52
|
-
// If the number provided is -ve, then return the latest block.
|
|
53
|
-
if (number < 0) {
|
|
54
|
-
number = await this.store.getSynchedL2BlockNumber();
|
|
55
|
-
}
|
|
56
|
-
if (number == 0) {
|
|
57
|
-
return undefined;
|
|
58
|
-
}
|
|
59
|
-
const blocks = await this.store.getPublishedBlocks(number, 1);
|
|
60
|
-
return blocks.length === 0 ? undefined : blocks[0];
|
|
61
|
-
}
|
|
62
|
-
|
|
63
|
-
/**
|
|
64
|
-
* Gets an l2 block. If a negative number is passed, the block returned is the most recent.
|
|
65
|
-
* @param number - The block number to return (inclusive).
|
|
66
|
-
* @returns The requested L2 block.
|
|
67
|
-
*/
|
|
68
|
-
public getBlock(number: number | 'latest'): Promise<L2Block | undefined> {
|
|
69
|
-
return this.getPublishedBlock(number != 'latest' ? number : -1).then(block => block?.block);
|
|
70
|
-
}
|
|
71
|
-
|
|
72
|
-
/**
|
|
73
|
-
* Gets an l2 block header.
|
|
74
|
-
* @param number - The block number to return or 'latest' for the most recent one.
|
|
75
|
-
* @returns The requested L2 block header.
|
|
76
|
-
*/
|
|
77
|
-
public async getBlockHeader(number: number | 'latest'): Promise<BlockHeader | undefined> {
|
|
78
|
-
if (number === 'latest') {
|
|
79
|
-
number = await this.store.getSynchedL2BlockNumber();
|
|
80
|
-
}
|
|
81
|
-
if (number === 0) {
|
|
82
|
-
return undefined;
|
|
83
|
-
}
|
|
84
|
-
const headers = await this.store.getBlockHeaders(number, 1);
|
|
85
|
-
return headers.length === 0 ? undefined : headers[0];
|
|
86
|
-
}
|
|
87
|
-
|
|
88
|
-
public getBlocks(from: number, limit: number, _proven?: boolean): Promise<L2Block[]> {
|
|
89
|
-
return this.getPublishedBlocks(from, limit).then(blocks => blocks.map(b => b.block));
|
|
90
|
-
}
|
|
91
|
-
|
|
92
|
-
public getL2SlotNumber(): Promise<bigint> {
|
|
93
|
-
throw new Error('TXE Archiver does not implement "getL2SlotNumber"');
|
|
94
|
-
}
|
|
11
|
+
/**
|
|
12
|
+
* TXE Archiver implementation.
|
|
13
|
+
* Provides most of the endpoints needed by the node for reading from and writing to state,
|
|
14
|
+
* without needing any of the extra overhead that the Archiver itself requires (i.e. an L1 client).
|
|
15
|
+
*/
|
|
16
|
+
export class TXEArchiver extends ArchiverDataSourceBase {
|
|
17
|
+
private readonly updater = new ArchiverDataStoreUpdater(this.store);
|
|
95
18
|
|
|
96
|
-
|
|
97
|
-
|
|
98
|
-
|
|
99
|
-
|
|
100
|
-
public getBlocksForEpoch(_epochNumber: bigint): Promise<L2Block[]> {
|
|
101
|
-
throw new Error('TXE Archiver does not implement "getBlocksForEpoch"');
|
|
19
|
+
constructor(db: AztecAsyncKVStore) {
|
|
20
|
+
const store = new KVArchiverDataStore(db, 9999);
|
|
21
|
+
super(store);
|
|
102
22
|
}
|
|
103
23
|
|
|
104
|
-
public
|
|
105
|
-
|
|
24
|
+
public async addCheckpoints(checkpoints: PublishedCheckpoint[], result?: ValidateCheckpointResult): Promise<void> {
|
|
25
|
+
await this.updater.addCheckpoints(checkpoints, result);
|
|
106
26
|
}
|
|
107
27
|
|
|
108
|
-
public
|
|
109
|
-
throw new Error('TXE Archiver does not implement "
|
|
28
|
+
public getRollupAddress(): Promise<EthAddress> {
|
|
29
|
+
throw new Error('TXE Archiver does not implement "getRollupAddress"');
|
|
110
30
|
}
|
|
111
31
|
|
|
112
|
-
public
|
|
113
|
-
throw new Error('TXE Archiver does not implement "
|
|
32
|
+
public getRegistryAddress(): Promise<EthAddress> {
|
|
33
|
+
throw new Error('TXE Archiver does not implement "getRegistryAddress"');
|
|
114
34
|
}
|
|
115
35
|
|
|
116
36
|
public getL1Constants(): Promise<L1RollupConstants> {
|
|
117
|
-
throw new Error('TXE Archiver does not implement "
|
|
37
|
+
throw new Error('TXE Archiver does not implement "getL1Constants"');
|
|
118
38
|
}
|
|
119
39
|
|
|
120
40
|
public getGenesisValues(): Promise<{ genesisArchiveRoot: Fr }> {
|
|
121
41
|
return Promise.resolve({ genesisArchiveRoot: new Fr(GENESIS_ARCHIVE_ROOT) });
|
|
122
42
|
}
|
|
123
43
|
|
|
124
|
-
public
|
|
125
|
-
throw new Error('TXE Archiver does not implement "
|
|
44
|
+
public getL1Timestamp(): Promise<bigint | undefined> {
|
|
45
|
+
throw new Error('TXE Archiver does not implement "getL1Timestamp"');
|
|
126
46
|
}
|
|
127
47
|
|
|
128
|
-
public
|
|
129
|
-
|
|
130
|
-
|
|
48
|
+
public async getL2Tips(): Promise<L2Tips> {
|
|
49
|
+
// In TXE there is no possibility of reorgs and no blocks are ever getting proven so we just set 'latest', 'proven'
|
|
50
|
+
// and 'finalized' to the latest block.
|
|
51
|
+
const blockHeader = await this.getBlockHeader('latest');
|
|
52
|
+
if (!blockHeader) {
|
|
53
|
+
throw new Error('L2Tips requested from TXE Archiver but no block header found');
|
|
54
|
+
}
|
|
131
55
|
|
|
132
|
-
|
|
133
|
-
|
|
56
|
+
const number = blockHeader.globalVariables.blockNumber;
|
|
57
|
+
const hash = (await blockHeader.hash()).toString();
|
|
58
|
+
const checkpointedBlock = await this.getCheckpointedBlock(number);
|
|
59
|
+
if (!checkpointedBlock) {
|
|
60
|
+
throw new Error(`L2Tips requested from TXE Archiver but no checkpointed block found for block number ${number}`);
|
|
61
|
+
}
|
|
62
|
+
// TXE uses 1-block-per-checkpoint for testing simplicity, so we can use block number as checkpoint number.
|
|
63
|
+
// This uses the deprecated fromBlockNumber method intentionally for the TXE testing environment.
|
|
64
|
+
const checkpoint = await this.store.getRangeOfCheckpoints(CheckpointNumber.fromBlockNumber(number), 1);
|
|
65
|
+
if (checkpoint.length === 0) {
|
|
66
|
+
throw new Error(`L2Tips requested from TXE Archiver but no checkpoint found for block number ${number}`);
|
|
67
|
+
}
|
|
68
|
+
const blockId: L2BlockId = { number, hash };
|
|
69
|
+
const checkpointId: CheckpointId = {
|
|
70
|
+
number: checkpoint[0].checkpointNumber,
|
|
71
|
+
hash: checkpoint[0].header.hash().toString(),
|
|
72
|
+
};
|
|
73
|
+
const tipId: L2TipId = { block: blockId, checkpoint: checkpointId };
|
|
74
|
+
return {
|
|
75
|
+
proposed: blockId,
|
|
76
|
+
proven: tipId,
|
|
77
|
+
finalized: tipId,
|
|
78
|
+
checkpointed: tipId,
|
|
79
|
+
};
|
|
134
80
|
}
|
|
135
81
|
|
|
136
|
-
public
|
|
137
|
-
throw new Error('TXE Archiver does not implement "
|
|
82
|
+
public getSyncedL2SlotNumber(): Promise<SlotNumber | undefined> {
|
|
83
|
+
throw new Error('TXE Archiver does not implement "getSyncedL2SlotNumber"');
|
|
138
84
|
}
|
|
139
85
|
|
|
140
|
-
public
|
|
141
|
-
throw new Error('TXE Archiver does not implement "
|
|
86
|
+
public getSyncedL2EpochNumber(): Promise<EpochNumber | undefined> {
|
|
87
|
+
throw new Error('TXE Archiver does not implement "getSyncedL2EpochNumber"');
|
|
142
88
|
}
|
|
143
89
|
|
|
144
|
-
public
|
|
145
|
-
|
|
90
|
+
public isEpochComplete(_epochNumber: EpochNumber): Promise<boolean> {
|
|
91
|
+
throw new Error('TXE Archiver does not implement "isEpochComplete"');
|
|
146
92
|
}
|
|
147
93
|
|
|
148
|
-
public
|
|
149
|
-
|
|
94
|
+
public syncImmediate(): Promise<void> {
|
|
95
|
+
throw new Error('TXE Archiver does not implement "syncImmediate"');
|
|
150
96
|
}
|
|
151
97
|
}
|
|
@@ -1,9 +1,13 @@
|
|
|
1
|
+
import type { SlotNumber } from '@aztec/foundation/branded-types';
|
|
1
2
|
import type {
|
|
2
3
|
AuthRequest,
|
|
3
4
|
ENR,
|
|
4
5
|
P2P,
|
|
5
6
|
P2PBlockReceivedCallback,
|
|
7
|
+
P2PCheckpointReceivedCallback,
|
|
6
8
|
P2PConfig,
|
|
9
|
+
P2PDuplicateAttestationCallback,
|
|
10
|
+
P2PDuplicateProposalCallback,
|
|
7
11
|
P2PSyncState,
|
|
8
12
|
PeerId,
|
|
9
13
|
ReqRespSubProtocol,
|
|
@@ -12,12 +16,12 @@ import type {
|
|
|
12
16
|
StatusMessage,
|
|
13
17
|
} from '@aztec/p2p';
|
|
14
18
|
import type { EthAddress, L2BlockStreamEvent, L2Tips } from '@aztec/stdlib/block';
|
|
15
|
-
import type { PeerInfo } from '@aztec/stdlib/interfaces/server';
|
|
16
|
-
import type {
|
|
17
|
-
import type { Tx, TxHash } from '@aztec/stdlib/tx';
|
|
19
|
+
import type { ITxProvider, PeerInfo } from '@aztec/stdlib/interfaces/server';
|
|
20
|
+
import type { BlockProposal, CheckpointAttestation, CheckpointProposal, TopicType } from '@aztec/stdlib/p2p';
|
|
21
|
+
import type { BlockHeader, Tx, TxHash } from '@aztec/stdlib/tx';
|
|
18
22
|
|
|
19
23
|
export class DummyP2P implements P2P {
|
|
20
|
-
public
|
|
24
|
+
public validateTxsReceivedInBlockProposal(_txs: Tx[]): Promise<void> {
|
|
21
25
|
return Promise.resolve();
|
|
22
26
|
}
|
|
23
27
|
|
|
@@ -37,14 +41,30 @@ export class DummyP2P implements P2P {
|
|
|
37
41
|
throw new Error('DummyP2P does not implement "getPeers"');
|
|
38
42
|
}
|
|
39
43
|
|
|
44
|
+
public getGossipMeshPeerCount(_topicType: TopicType): Promise<number> {
|
|
45
|
+
return Promise.resolve(0);
|
|
46
|
+
}
|
|
47
|
+
|
|
40
48
|
public broadcastProposal(_proposal: BlockProposal): Promise<void> {
|
|
41
49
|
throw new Error('DummyP2P does not implement "broadcastProposal"');
|
|
42
50
|
}
|
|
43
51
|
|
|
52
|
+
public broadcastCheckpointProposal(_proposal: CheckpointProposal): Promise<void> {
|
|
53
|
+
throw new Error('DummyP2P does not implement "broadcastCheckpointProposal"');
|
|
54
|
+
}
|
|
55
|
+
|
|
56
|
+
public broadcastCheckpointAttestations(_attestations: CheckpointAttestation[]): Promise<void> {
|
|
57
|
+
throw new Error('DummyP2P does not implement "broadcastCheckpointAttestations"');
|
|
58
|
+
}
|
|
59
|
+
|
|
44
60
|
public registerBlockProposalHandler(_handler: P2PBlockReceivedCallback): void {
|
|
45
61
|
throw new Error('DummyP2P does not implement "registerBlockProposalHandler"');
|
|
46
62
|
}
|
|
47
63
|
|
|
64
|
+
public registerCheckpointProposalHandler(_handler: P2PCheckpointReceivedCallback): void {
|
|
65
|
+
throw new Error('DummyP2P does not implement "registerCheckpointProposalHandler"');
|
|
66
|
+
}
|
|
67
|
+
|
|
48
68
|
public requestTxs(_txHashes: TxHash[]): Promise<(Tx | undefined)[]> {
|
|
49
69
|
throw new Error('DummyP2P does not implement "requestTxs"');
|
|
50
70
|
}
|
|
@@ -57,8 +77,8 @@ export class DummyP2P implements P2P {
|
|
|
57
77
|
throw new Error('DummyP2P does not implement "sendTx"');
|
|
58
78
|
}
|
|
59
79
|
|
|
60
|
-
public
|
|
61
|
-
throw new Error('DummyP2P does not implement "
|
|
80
|
+
public handleFailedExecution(_txHashes: TxHash[]): Promise<void> {
|
|
81
|
+
throw new Error('DummyP2P does not implement "handleFailedExecution"');
|
|
62
82
|
}
|
|
63
83
|
|
|
64
84
|
public getTxByHashFromPool(_txHash: TxHash): Promise<Tx | undefined> {
|
|
@@ -74,13 +94,19 @@ export class DummyP2P implements P2P {
|
|
|
74
94
|
}
|
|
75
95
|
|
|
76
96
|
public getTxStatus(_txHash: TxHash): Promise<'pending' | 'mined' | undefined> {
|
|
77
|
-
|
|
97
|
+
// In TXE there is no concept of transactions but we need to implement this because of tagging. We return 'mined'
|
|
98
|
+
// tx status for any tx hash.
|
|
99
|
+
return Promise.resolve('mined');
|
|
78
100
|
}
|
|
79
101
|
|
|
80
102
|
public iteratePendingTxs(): AsyncIterableIterator<Tx> {
|
|
81
103
|
throw new Error('DummyP2P does not implement "iteratePendingTxs"');
|
|
82
104
|
}
|
|
83
105
|
|
|
106
|
+
public iterateEligiblePendingTxs(): AsyncIterableIterator<Tx> {
|
|
107
|
+
throw new Error('DummyP2P does not implement "iterateEligiblePendingTxs"');
|
|
108
|
+
}
|
|
109
|
+
|
|
84
110
|
public getPendingTxCount(): Promise<number> {
|
|
85
111
|
throw new Error('DummyP2P does not implement "getPendingTxCount"');
|
|
86
112
|
}
|
|
@@ -109,20 +135,20 @@ export class DummyP2P implements P2P {
|
|
|
109
135
|
throw new Error('DummyP2P does not implement "isP2PClient"');
|
|
110
136
|
}
|
|
111
137
|
|
|
112
|
-
public
|
|
113
|
-
throw new Error('DummyP2P does not implement "
|
|
138
|
+
public getTxProvider(): ITxProvider {
|
|
139
|
+
throw new Error('DummyP2P does not implement "getTxProvider"');
|
|
114
140
|
}
|
|
115
141
|
|
|
116
|
-
public
|
|
117
|
-
throw new Error('DummyP2P does not implement "
|
|
142
|
+
public getTxsByHash(_txHashes: TxHash[]): Promise<Tx[]> {
|
|
143
|
+
throw new Error('DummyP2P does not implement "getTxsByHash"');
|
|
118
144
|
}
|
|
119
145
|
|
|
120
|
-
public
|
|
121
|
-
|
|
146
|
+
public getCheckpointAttestationsForSlot(_slot: SlotNumber, _proposalId?: string): Promise<CheckpointAttestation[]> {
|
|
147
|
+
throw new Error('DummyP2P does not implement "getCheckpointAttestationsForSlot"');
|
|
122
148
|
}
|
|
123
149
|
|
|
124
|
-
public
|
|
125
|
-
throw new Error('DummyP2P does not implement "
|
|
150
|
+
public addOwnCheckpointAttestations(_attestations: CheckpointAttestation[]): Promise<void> {
|
|
151
|
+
throw new Error('DummyP2P does not implement "addOwnCheckpointAttestations"');
|
|
126
152
|
}
|
|
127
153
|
|
|
128
154
|
public getL2BlockHash(_number: number): Promise<string | undefined> {
|
|
@@ -145,14 +171,6 @@ export class DummyP2P implements P2P {
|
|
|
145
171
|
throw new Error('DummyP2P does not implement "sync"');
|
|
146
172
|
}
|
|
147
173
|
|
|
148
|
-
public requestTxsByHash(_txHashes: TxHash[]): Promise<Tx[]> {
|
|
149
|
-
throw new Error('DummyP2P does not implement "requestTxsByHash"');
|
|
150
|
-
}
|
|
151
|
-
|
|
152
|
-
public getTxs(_filter: 'all' | 'pending' | 'mined'): Promise<Tx[]> {
|
|
153
|
-
throw new Error('DummyP2P does not implement "getTxs"');
|
|
154
|
-
}
|
|
155
|
-
|
|
156
174
|
public getTxsByHashFromPool(_txHashes: TxHash[]): Promise<(Tx | undefined)[]> {
|
|
157
175
|
throw new Error('DummyP2P does not implement "getTxsByHashFromPool"');
|
|
158
176
|
}
|
|
@@ -161,10 +179,6 @@ export class DummyP2P implements P2P {
|
|
|
161
179
|
throw new Error('DummyP2P does not implement "hasTxsInPool"');
|
|
162
180
|
}
|
|
163
181
|
|
|
164
|
-
public addTxsToPool(_txs: Tx[]): Promise<number> {
|
|
165
|
-
throw new Error('DummyP2P does not implement "addTxs"');
|
|
166
|
-
}
|
|
167
|
-
|
|
168
182
|
public getSyncedLatestBlockNum(): Promise<number> {
|
|
169
183
|
throw new Error('DummyP2P does not implement "getSyncedLatestBlockNum"');
|
|
170
184
|
}
|
|
@@ -177,8 +191,12 @@ export class DummyP2P implements P2P {
|
|
|
177
191
|
throw new Error('DummyP2P does not implement "getSyncedLatestSlot"');
|
|
178
192
|
}
|
|
179
193
|
|
|
180
|
-
|
|
181
|
-
throw new Error('DummyP2P does not implement "
|
|
194
|
+
protectTxs(_txHashes: TxHash[], _blockHeader: BlockHeader): Promise<TxHash[]> {
|
|
195
|
+
throw new Error('DummyP2P does not implement "protectTxs".');
|
|
196
|
+
}
|
|
197
|
+
|
|
198
|
+
prepareForSlot(_slotNumber: SlotNumber): Promise<void> {
|
|
199
|
+
return Promise.resolve();
|
|
182
200
|
}
|
|
183
201
|
|
|
184
202
|
addReqRespSubProtocol(
|
|
@@ -194,4 +212,16 @@ export class DummyP2P implements P2P {
|
|
|
194
212
|
|
|
195
213
|
//This is no-op
|
|
196
214
|
public registerThisValidatorAddresses(_address: EthAddress[]): void {}
|
|
215
|
+
|
|
216
|
+
public registerDuplicateProposalCallback(_callback: P2PDuplicateProposalCallback): void {
|
|
217
|
+
throw new Error('DummyP2P does not implement "registerDuplicateProposalCallback"');
|
|
218
|
+
}
|
|
219
|
+
|
|
220
|
+
public registerDuplicateAttestationCallback(_callback: P2PDuplicateAttestationCallback): void {
|
|
221
|
+
throw new Error('DummyP2P does not implement "registerDuplicateAttestationCallback"');
|
|
222
|
+
}
|
|
223
|
+
|
|
224
|
+
public hasBlockProposalsForSlot(_slot: SlotNumber): Promise<boolean> {
|
|
225
|
+
throw new Error('DummyP2P does not implement "hasBlockProposalsForSlot"');
|
|
226
|
+
}
|
|
197
227
|
}
|
|
@@ -1,20 +1,38 @@
|
|
|
1
|
+
import { BlockNumber, type SlotNumber } from '@aztec/foundation/branded-types';
|
|
1
2
|
import type { EthAddress } from '@aztec/foundation/eth-address';
|
|
2
3
|
import type { AztecAddress } from '@aztec/stdlib/aztec-address';
|
|
3
4
|
import { GasFees } from '@aztec/stdlib/gas';
|
|
4
5
|
import { makeGlobalVariables } from '@aztec/stdlib/testing';
|
|
5
|
-
import { type GlobalVariableBuilder, GlobalVariables } from '@aztec/stdlib/tx';
|
|
6
|
+
import { type CheckpointGlobalVariables, type GlobalVariableBuilder, GlobalVariables } from '@aztec/stdlib/tx';
|
|
6
7
|
|
|
7
8
|
export class TXEGlobalVariablesBuilder implements GlobalVariableBuilder {
|
|
8
|
-
public
|
|
9
|
+
public getCurrentMinFees(): Promise<GasFees> {
|
|
9
10
|
return Promise.resolve(new GasFees(0, 0));
|
|
10
11
|
}
|
|
11
12
|
|
|
12
13
|
public buildGlobalVariables(
|
|
13
|
-
_blockNumber:
|
|
14
|
+
_blockNumber: BlockNumber,
|
|
14
15
|
_coinbase: EthAddress,
|
|
15
16
|
_feeRecipient: AztecAddress,
|
|
16
|
-
_slotNumber?:
|
|
17
|
+
_slotNumber?: SlotNumber,
|
|
17
18
|
): Promise<GlobalVariables> {
|
|
18
19
|
return Promise.resolve(makeGlobalVariables());
|
|
19
20
|
}
|
|
21
|
+
|
|
22
|
+
public buildCheckpointGlobalVariables(
|
|
23
|
+
_coinbase: EthAddress,
|
|
24
|
+
_feeRecipient: AztecAddress,
|
|
25
|
+
_slotNumber: SlotNumber,
|
|
26
|
+
): Promise<CheckpointGlobalVariables> {
|
|
27
|
+
const vars = makeGlobalVariables();
|
|
28
|
+
return Promise.resolve({
|
|
29
|
+
chainId: vars.chainId,
|
|
30
|
+
version: vars.version,
|
|
31
|
+
slotNumber: vars.slotNumber,
|
|
32
|
+
timestamp: vars.timestamp,
|
|
33
|
+
coinbase: vars.coinbase,
|
|
34
|
+
feeRecipient: vars.feeRecipient,
|
|
35
|
+
gasFees: vars.gasFees,
|
|
36
|
+
});
|
|
37
|
+
}
|
|
20
38
|
}
|
|
@@ -1,10 +1,14 @@
|
|
|
1
1
|
import { type AztecNodeConfig, AztecNodeService } from '@aztec/aztec-node';
|
|
2
2
|
import { TestCircuitVerifier } from '@aztec/bb-prover/test';
|
|
3
|
+
import { CheckpointNumber } from '@aztec/foundation/branded-types';
|
|
4
|
+
import { Fr } from '@aztec/foundation/curves/bn254';
|
|
3
5
|
import { createLogger } from '@aztec/foundation/log';
|
|
4
|
-
import type
|
|
5
|
-
import {
|
|
6
|
-
import {
|
|
6
|
+
import { type AnchorBlockStore, type ContractStore, ContractSyncService, type NoteStore } from '@aztec/pxe/server';
|
|
7
|
+
import { MessageContextService } from '@aztec/pxe/simulator';
|
|
8
|
+
import { L2Block } from '@aztec/stdlib/block';
|
|
9
|
+
import { Checkpoint, L1PublishedData, PublishedCheckpoint } from '@aztec/stdlib/checkpoint';
|
|
7
10
|
import type { AztecNode } from '@aztec/stdlib/interfaces/client';
|
|
11
|
+
import { CheckpointHeader } from '@aztec/stdlib/rollup';
|
|
8
12
|
import { getPackageVersion } from '@aztec/stdlib/update-checker';
|
|
9
13
|
|
|
10
14
|
import { TXEArchiver } from './archiver.js';
|
|
@@ -21,14 +25,18 @@ export class TXEStateMachine {
|
|
|
21
25
|
public node: AztecNode,
|
|
22
26
|
public synchronizer: TXESynchronizer,
|
|
23
27
|
public archiver: TXEArchiver,
|
|
24
|
-
public
|
|
28
|
+
public anchorBlockStore: AnchorBlockStore,
|
|
29
|
+
public contractSyncService: ContractSyncService,
|
|
30
|
+
public messageContextService: MessageContextService,
|
|
25
31
|
) {}
|
|
26
32
|
|
|
27
|
-
public static async create(
|
|
28
|
-
|
|
33
|
+
public static async create(
|
|
34
|
+
archiver: TXEArchiver,
|
|
35
|
+
anchorBlockStore: AnchorBlockStore,
|
|
36
|
+
contractStore: ContractStore,
|
|
37
|
+
noteStore: NoteStore,
|
|
38
|
+
) {
|
|
29
39
|
const synchronizer = await TXESynchronizer.create();
|
|
30
|
-
const syncDataProvider = new SyncDataProvider(db);
|
|
31
|
-
|
|
32
40
|
const aztecNodeConfig = {} as AztecNodeConfig;
|
|
33
41
|
|
|
34
42
|
const log = createLogger('txe_node');
|
|
@@ -44,6 +52,7 @@ export class TXEStateMachine {
|
|
|
44
52
|
undefined,
|
|
45
53
|
undefined,
|
|
46
54
|
undefined,
|
|
55
|
+
undefined,
|
|
47
56
|
VERSION,
|
|
48
57
|
CHAIN_ID,
|
|
49
58
|
new TXEGlobalVariablesBuilder(),
|
|
@@ -54,24 +63,58 @@ export class TXEStateMachine {
|
|
|
54
63
|
log,
|
|
55
64
|
);
|
|
56
65
|
|
|
57
|
-
|
|
66
|
+
const contractSyncService = new ContractSyncService(
|
|
67
|
+
node,
|
|
68
|
+
contractStore,
|
|
69
|
+
noteStore,
|
|
70
|
+
createLogger('txe:contract_sync'),
|
|
71
|
+
);
|
|
72
|
+
|
|
73
|
+
const messageContextService = new MessageContextService(node);
|
|
74
|
+
|
|
75
|
+
return new this(node, synchronizer, archiver, anchorBlockStore, contractSyncService, messageContextService);
|
|
58
76
|
}
|
|
59
77
|
|
|
60
78
|
public async handleL2Block(block: L2Block) {
|
|
79
|
+
// Create a checkpoint from the block manually.
|
|
80
|
+
// TXE uses 1-block-per-checkpoint for testing simplicity, so we can use block number as checkpoint number.
|
|
81
|
+
// This uses the deprecated fromBlockNumber method intentionally for the TXE testing environment.
|
|
82
|
+
const checkpointNumber = CheckpointNumber.fromBlockNumber(block.number);
|
|
83
|
+
const checkpoint = new Checkpoint(
|
|
84
|
+
block.archive,
|
|
85
|
+
CheckpointHeader.from({
|
|
86
|
+
lastArchiveRoot: block.header.lastArchive.root,
|
|
87
|
+
inHash: Fr.ZERO,
|
|
88
|
+
blobsHash: Fr.ZERO,
|
|
89
|
+
blockHeadersHash: Fr.ZERO,
|
|
90
|
+
epochOutHash: Fr.ZERO,
|
|
91
|
+
slotNumber: block.header.globalVariables.slotNumber,
|
|
92
|
+
timestamp: block.header.globalVariables.timestamp,
|
|
93
|
+
coinbase: block.header.globalVariables.coinbase,
|
|
94
|
+
feeRecipient: block.header.globalVariables.feeRecipient,
|
|
95
|
+
gasFees: block.header.globalVariables.gasFees,
|
|
96
|
+
totalManaUsed: block.header.totalManaUsed,
|
|
97
|
+
}),
|
|
98
|
+
[block],
|
|
99
|
+
checkpointNumber,
|
|
100
|
+
);
|
|
101
|
+
|
|
102
|
+
const publishedCheckpoint = new PublishedCheckpoint(
|
|
103
|
+
checkpoint,
|
|
104
|
+
new L1PublishedData(
|
|
105
|
+
BigInt(block.header.globalVariables.blockNumber),
|
|
106
|
+
block.header.globalVariables.timestamp,
|
|
107
|
+
block.header.globalVariables.blockNumber.toString(),
|
|
108
|
+
),
|
|
109
|
+
[],
|
|
110
|
+
);
|
|
111
|
+
// Wipe contract sync cache when anchor block changes (mirrors BlockSynchronizer behavior)
|
|
112
|
+
this.contractSyncService.wipe();
|
|
113
|
+
|
|
61
114
|
await Promise.all([
|
|
62
115
|
this.synchronizer.handleL2Block(block),
|
|
63
|
-
this.archiver.
|
|
64
|
-
|
|
65
|
-
block,
|
|
66
|
-
l1: {
|
|
67
|
-
blockHash: block.header.globalVariables.blockNumber.toString(),
|
|
68
|
-
blockNumber: BigInt(block.header.globalVariables.blockNumber),
|
|
69
|
-
timestamp: block.header.globalVariables.timestamp,
|
|
70
|
-
},
|
|
71
|
-
attestations: [],
|
|
72
|
-
}),
|
|
73
|
-
]),
|
|
74
|
-
this.syncDataProvider.setHeader(block.getBlockHeader()),
|
|
116
|
+
this.archiver.addCheckpoints([publishedCheckpoint], undefined),
|
|
117
|
+
this.anchorBlockStore.setHeader(block.header),
|
|
75
118
|
]);
|
|
76
119
|
}
|
|
77
120
|
}
|