diff --git a/.github/workflows/devnet-deploys.yml b/.github/workflows/devnet-deploys.yml index 496f4f72833..75d0431dfda 100644 --- a/.github/workflows/devnet-deploys.yml +++ b/.github/workflows/devnet-deploys.yml @@ -546,7 +546,6 @@ jobs: echo "TF_VAR_REGISTRY_CONTRACT_ADDRESS=$(extract registryAddress)" >>$GITHUB_ENV echo "TF_VAR_INBOX_CONTRACT_ADDRESS=$(extract inboxAddress)" >>$GITHUB_ENV echo "TF_VAR_OUTBOX_CONTRACT_ADDRESS=$(extract outboxAddress)" >>$GITHUB_ENV - echo "TF_VAR_AVAILABILITY_ORACLE_CONTRACT_ADDRESS=$(extract availabilityOracleAddress)" >>$GITHUB_ENV echo "TF_VAR_FEE_JUICE_CONTRACT_ADDRESS=$(extract feeJuiceAddress)" >>$GITHUB_ENV echo "TF_VAR_FEE_JUICE_PORTAL_CONTRACT_ADDRESS=$(extract feeJuicePortalAddress)" >>$GITHUB_ENV diff --git a/.github/workflows/publish-bb.yml b/.github/workflows/publish-bb.yml index fc5e0acdce0..4b859087c54 100644 --- a/.github/workflows/publish-bb.yml +++ b/.github/workflows/publish-bb.yml @@ -17,6 +17,7 @@ on: permissions: # Necessary to upload new release artifacts contents: write + issues: write jobs: build-x86_64-linux-gnu: @@ -59,7 +60,7 @@ jobs: run: tar -cvzf barretenberg-x86_64-linux-gnu.tar.gz bb - name: Upload artifacts - uses: actions/upload-artifact@v2 + uses: actions/upload-artifact@v4 with: name: release-linux path: | @@ -128,7 +129,7 @@ jobs: # NODE_AUTH_TOKEN: ${{secrets.NPM_TOKEN}} - name: Upload artifacts - uses: actions/upload-artifact@v2 + uses: actions/upload-artifact@v4 with: name: release-wasm path: | @@ -163,7 +164,7 @@ jobs: 7z a -ttar -so -an ./dist/* | 7z a -si ./barretenberg-x86_64-apple-darwin.tar.gz - name: Upload artifact - uses: actions/upload-artifact@v3 + uses: actions/upload-artifact@v4 with: name: barretenberg-x86_64-apple-darwin path: ./barretenberg/cpp/build/bin/barretenberg-x86_64-apple-darwin.tar.gz @@ -195,7 +196,7 @@ jobs: 7z a -ttar -so -an ./dist/* | 7z a -si ./barretenberg-aarch64-apple-darwin.tar.gz - name: Upload artifact - uses: actions/upload-artifact@v3 + uses: actions/upload-artifact@v4 with: name: barretenberg-aarch64-apple-darwin path: ./barretenberg/cpp/build/bin/barretenberg-aarch64-apple-darwin.tar.gz @@ -243,22 +244,22 @@ jobs: runs-on: ubuntu-latest steps: - name: Download files from Linux Runner - uses: actions/download-artifact@v2 + uses: actions/download-artifact@v4 with: name: release-linux - name: Download files for WASM - uses: actions/download-artifact@v2 + uses: actions/download-artifact@v4 with: name: release-wasm - name: Download files from x86_64 Mac Runner - uses: actions/download-artifact@v2 + uses: actions/download-artifact@v4 with: name: barretenberg-x86_64-apple-darwin - name: Download files from aarch64 Mac Runner - uses: actions/download-artifact@v2 + uses: actions/download-artifact@v4 with: name: barretenberg-aarch64-apple-darwin diff --git a/.noir-sync-commit b/.noir-sync-commit index 9326943a310..f2fa0b3c445 100644 --- a/.noir-sync-commit +++ b/.noir-sync-commit @@ -1 +1 @@ -d6f60d70dc41640ad84f7a968927b20818bcaf2a +3c3ed1e3d28946a02071c524dd128afe131bc3da diff --git a/docker-compose.yml b/docker-compose.yml index 9106ee66811..18ca3c48335 100644 --- a/docker-compose.yml +++ b/docker-compose.yml @@ -50,7 +50,6 @@ services: REGISTRY_CONTRACT_ADDRESS: "0x589a9634c1d00c62e47b3b7a790c8dc986b3d40d" INBOX_CONTRACT_ADDRESS: "0x12d9b5effc69bf5c0c29c8258c6b6fa95a08de74" OUTBOX_CONTRACT_ADDRESS: "0x3ec4b6c68a8c2ce4c78cdd465b3019b11a568d1d" - AVAILABILITY_ORACLE_CONTRACT_ADDRESS: "0x98a4089127f3f5d555656f1c9b1801342c9d6bce" FEE_JUICE_CONTRACT_ADDRESS: "0x73c43b919973711e096bfc04c9d4b3be511ffc0b" FEE_JUICE_PORTAL_CONTRACT_ADDRESS: "0xdf25b0a34dbee9f25518f7a4d63bab8b3bb3e496" ETHEREUM_HOST: diff --git a/docs/docs/protocol-specs/data-publication-and-availability/overview.md b/docs/docs/protocol-specs/data-publication-and-availability/overview.md index ca46010fc15..c8bcc533217 100644 --- a/docs/docs/protocol-specs/data-publication-and-availability/overview.md +++ b/docs/docs/protocol-specs/data-publication-and-availability/overview.md @@ -13,7 +13,7 @@ As for that, we highly recommend reading [this very nice post](https://dba.xyz/d Essentially Data Publication $\subset$ Data Availability, since if it is available, it must also have been published. This difference might be small but becomes important in a few moments. -Progressing the state of the validating light node requires that we can convince it (and therefore the [availability oracle](./index.md#availability-oracle)) that the data was published - as it needs to compute the public inputs for the proof. +Progressing the state of the validating light node requires that we can convince it that the data was published - as it needs to compute the public inputs for the proof. The exact method of computing these public inputs can vary depending on the data layer, but generally, it could be by providing the data directly or by using data availability sampling or a data availability committee. The exact mechanism greatly impacts the security and cost of the system, and will be discussed in the following sections. @@ -246,7 +246,7 @@ Assuming that this is a decent guess, and we can estimate the data requirements Using the values from just above for transaction data requirements, we can get a ball park estimate of what we can expect to require at different throughput levels. -|Throughput | Everyone | Someone | Total | +|Throughput | Everyone | Someone | Total | |:-----:|:-----:|:-----:|:-----:| | 1 TPS | $512 \dfrac{byte}{s}$ | $1036 \dfrac{byte}{s}$ | $1548 \dfrac{byte}{s}$ | | 10 TPS | $5120 \dfrac{byte}{s}$ | $10360 \dfrac{byte}{s}$ | $15480 \dfrac{byte}{s}$ | diff --git a/docs/docs/protocol-specs/l1-smart-contracts/index.md b/docs/docs/protocol-specs/l1-smart-contracts/index.md index 19dc8f05140..845c77e36aa 100644 --- a/docs/docs/protocol-specs/l1-smart-contracts/index.md +++ b/docs/docs/protocol-specs/l1-smart-contracts/index.md @@ -21,7 +21,7 @@ When presented with a new [`ProvenBlock`](../rollup-circuits/root-rollup.md) and The `archive` used as public input is the archive after the new header is inserted (see [root rollup](./../rollup-circuits/root-rollup.md)). ```python -def process(block: ProvenBlock, proof: Proof): +def propose(block: ProvenBlock, proof: Proof): header = block.header block_number = header.global_variables.block_number @@ -57,6 +57,8 @@ Namely, we need the cross-chain messages to be published to L1, but the rest of :::info Validium or Rollup If a different data availability layer than Ethereum is used for the block body, we are effectively building a Validium. If we use Ethereum for the block body, we are building a Rollup. + +For more information around the requirements we have for the availability, see [Data Availability](../data-publication-and-availability/index.md). ::: Using the data structures defined throughout the [rollup circuits](./../rollup-circuits/index.md) section, we can outline the validating light node structure as follows: @@ -64,12 +66,6 @@ Using the data structures defined throughout the [rollup circuits](./../rollup-c ```mermaid classDiagram -class AvailabilityOracle { - available: Map[Fr => bool] - mut publish(effects: TxEffect[]): Fr - is_available(txs_hash: Fr): bool -} - class Inbox { consume(): bytes32 } @@ -84,9 +80,8 @@ class Verifier { class StateTransitioner { archive: Snapshot - process(header: Header, archive: Fr, proof: Proof) + propose(header: Header, archive: Fr, proof: Proof, body: Body) } -StateTransitioner --> AvailabilityOracle: is_available() StateTransitioner --> Inbox: consume() StateTransitioner --> Outbox: insert() StateTransitioner --> Verifier: verify() @@ -110,7 +105,6 @@ class StateTransitioner: slot_number: uint128 VERIFIER: immutable(IVerifier) - AVAILABILITY_ORACLE: immutable(IAvailabilityOracle) INBOX: immutable(IInbox) OUTBOX: immutable(IOutbox) VERSION: immutable(uint256) @@ -126,13 +120,14 @@ class StateTransitioner: self.blocks.append(BlockLog({archive: bytes32(0), slot_number: 0})) self.GENESIS_TIME = block.timestamp - def process( + def propose( self, header: Header, archive: Fr, - proof: Proof + proof: Proof, + body: Body ): - assert self.AVAILABILITY_ORACLE.is_available(header.content_commitment.txs_hash) + assert body.compute_commitment() == header.content_commitment assert self.validate_header(header) assert VERIFIER.verify(header, archive, proof) assert self.INBOX.consume() == header.content_commitment.in_hash @@ -162,25 +157,6 @@ class StateTransitioner: return True ``` -### Availability Oracle - -The state transitioner should be connected to an oracle which addresses the availability condition. - -For the case of a rollup, this "oracle" will be deriving the `TxsHash` from calldata and blobs. -For a validium it should be connected to a bridge that it can use to verify that the data is available on the other chain. - -For a generic DA that publishes data commitments to Ethereum, the oracle could be a snark proof that opens the data commitment from the bridge and computes the `TxsHash` from it. - -By having the availability oracle be independent from state progression we can even do multi-transaction blocks, e.g., use multiple transactions or commitments from other DA layers to construct the `TxsHash` for a large block. - -For more information around the requirements we have for the availability oracle, see [Data Availability](../data-publication-and-availability/index.md). - -An interesting observation around the availability oracle is that the `OutHash` and `InHash` don't need to be explicitly proven available through it. -The `InHash` is already proven as part of the L1 inbox, as we will see in a second. -And the `OutHash` consists entirely of a subset of the contents of the `TxsHash`, which is already proven available. - - - ### Registry To keep one location where all the core rollup contracts can be found, we have a registry contract. diff --git a/helm-charts/aztec-network/files/config/config-prover-env.sh b/helm-charts/aztec-network/files/config/config-prover-env.sh index a4bd508a111..125ee8dca11 100644 --- a/helm-charts/aztec-network/files/config/config-prover-env.sh +++ b/helm-charts/aztec-network/files/config/config-prover-env.sh @@ -14,7 +14,6 @@ rollup_address=$(echo "$output" | grep -oP 'Rollup Address: \K0x[a-fA-F0-9]{40}' registry_address=$(echo "$output" | grep -oP 'Registry Address: \K0x[a-fA-F0-9]{40}') inbox_address=$(echo "$output" | grep -oP 'L1 -> L2 Inbox Address: \K0x[a-fA-F0-9]{40}') outbox_address=$(echo "$output" | grep -oP 'L2 -> L1 Outbox Address: \K0x[a-fA-F0-9]{40}') -availability_oracle_address=$(echo "$output" | grep -oP 'Availability Oracle Address: \K0x[a-fA-F0-9]{40}') fee_juice_address=$(echo "$output" | grep -oP 'Fee Juice Address: \K0x[a-fA-F0-9]{40}') fee_juice_portal_address=$(echo "$output" | grep -oP 'Fee Juice Portal Address: \K0x[a-fA-F0-9]{40}') @@ -26,7 +25,6 @@ export ROLLUP_CONTRACT_ADDRESS=$rollup_address export REGISTRY_CONTRACT_ADDRESS=$registry_address export INBOX_CONTRACT_ADDRESS=$inbox_address export OUTBOX_CONTRACT_ADDRESS=$outbox_address -export AVAILABILITY_ORACLE_CONTRACT_ADDRESS=$availability_oracle_address export FEE_JUICE_CONTRACT_ADDRESS=$fee_juice_address export FEE_JUICE_PORTAL_CONTRACT_ADDRESS=$fee_juice_portal_address EOF diff --git a/helm-charts/aztec-network/files/config/config-validator-env.sh b/helm-charts/aztec-network/files/config/config-validator-env.sh index e2b29694bc7..f25834ad54d 100644 --- a/helm-charts/aztec-network/files/config/config-validator-env.sh +++ b/helm-charts/aztec-network/files/config/config-validator-env.sh @@ -14,7 +14,6 @@ rollup_address=$(echo "$output" | grep -oP 'Rollup Address: \K0x[a-fA-F0-9]{40}' registry_address=$(echo "$output" | grep -oP 'Registry Address: \K0x[a-fA-F0-9]{40}') inbox_address=$(echo "$output" | grep -oP 'L1 -> L2 Inbox Address: \K0x[a-fA-F0-9]{40}') outbox_address=$(echo "$output" | grep -oP 'L2 -> L1 Outbox Address: \K0x[a-fA-F0-9]{40}') -availability_oracle_address=$(echo "$output" | grep -oP 'Availability Oracle Address: \K0x[a-fA-F0-9]{40}') fee_juice_address=$(echo "$output" | grep -oP 'Fee Juice Address: \K0x[a-fA-F0-9]{40}') fee_juice_portal_address=$(echo "$output" | grep -oP 'Fee Juice Portal Address: \K0x[a-fA-F0-9]{40}') @@ -37,7 +36,6 @@ export ROLLUP_CONTRACT_ADDRESS=$rollup_address export REGISTRY_CONTRACT_ADDRESS=$registry_address export INBOX_CONTRACT_ADDRESS=$inbox_address export OUTBOX_CONTRACT_ADDRESS=$outbox_address -export AVAILABILITY_ORACLE_CONTRACT_ADDRESS=$availability_oracle_address export FEE_JUICE_CONTRACT_ADDRESS=$fee_juice_address export FEE_JUICE_PORTAL_CONTRACT_ADDRESS=$fee_juice_portal_address export VALIDATOR_PRIVATE_KEY=$private_key diff --git a/helm-charts/aztec-network/templates/deploy-contracts.config-map.yaml b/helm-charts/aztec-network/templates/deploy-contracts.config-map.yaml index 157cb822dbd..7191a38b78f 100644 --- a/helm-charts/aztec-network/templates/deploy-contracts.config-map.yaml +++ b/helm-charts/aztec-network/templates/deploy-contracts.config-map.yaml @@ -19,7 +19,6 @@ data: registry_address=$(echo "$output" | grep -oP 'Registry Address: \K0x[a-fA-F0-9]{40}') inbox_address=$(echo "$output" | grep -oP 'L1 -> L2 Inbox Address: \K0x[a-fA-F0-9]{40}') outbox_address=$(echo "$output" | grep -oP 'L2 -> L1 Outbox Address: \K0x[a-fA-F0-9]{40}') - availability_oracle_address=$(echo "$output" | grep -oP 'Availability Oracle Address: \K0x[a-fA-F0-9]{40}') fee_juice_address=$(echo "$output" | grep -oP 'Fee Juice Address: \K0x[a-fA-F0-9]{40}') fee_juice_portal_address=$(echo "$output" | grep -oP 'Fee Juice Portal Address: \K0x[a-fA-F0-9]{40}') @@ -29,7 +28,6 @@ data: export REGISTRY_CONTRACT_ADDRESS=$registry_address export INBOX_CONTRACT_ADDRESS=$inbox_address export OUTBOX_CONTRACT_ADDRESS=$outbox_address - export AVAILABILITY_ORACLE_CONTRACT_ADDRESS=$availability_oracle_address export FEE_JUICE_CONTRACT_ADDRESS=$fee_juice_address export FEE_JUICE_PORTAL_CONTRACT_ADDRESS=$fee_juice_portal_address EOF diff --git a/l1-contracts/src/core/Rollup.sol b/l1-contracts/src/core/Rollup.sol index ca9aed08f2d..03f8181ddf2 100644 --- a/l1-contracts/src/core/Rollup.sol +++ b/l1-contracts/src/core/Rollup.sol @@ -4,7 +4,6 @@ pragma solidity >=0.8.18; // Interfaces import {IRollup, ITestRollup} from "./interfaces/IRollup.sol"; -import {IAvailabilityOracle} from "./interfaces/IAvailabilityOracle.sol"; import {IInbox} from "./interfaces/messagebridge/IInbox.sol"; import {IOutbox} from "./interfaces/messagebridge/IOutbox.sol"; import {IRegistry} from "./interfaces/messagebridge/IRegistry.sol"; @@ -19,6 +18,7 @@ import {MerkleLib} from "./libraries/MerkleLib.sol"; import {SignatureLib} from "./sequencer_selection/SignatureLib.sol"; import {SafeCast} from "@oz/utils/math/SafeCast.sol"; import {DataStructures} from "./libraries/DataStructures.sol"; +import {TxsDecoder} from "./libraries/decoders/TxsDecoder.sol"; // Contracts import {MockVerifier} from "../mock/MockVerifier.sol"; @@ -47,7 +47,6 @@ contract Rollup is Leonidas, IRollup, ITestRollup { uint256 public constant TIMELINESS_PROVING_IN_SLOTS = 100; IRegistry public immutable REGISTRY; - IAvailabilityOracle public immutable AVAILABILITY_ORACLE; IInbox public immutable INBOX; IOutbox public immutable OUTBOX; uint256 public immutable VERSION; @@ -73,7 +72,6 @@ contract Rollup is Leonidas, IRollup, ITestRollup { constructor( IRegistry _registry, - IAvailabilityOracle _availabilityOracle, IFeeJuicePortal _fpcJuicePortal, bytes32 _vkTreeRoot, address _ares, @@ -81,7 +79,6 @@ contract Rollup is Leonidas, IRollup, ITestRollup { ) Leonidas(_ares) { verifier = new MockVerifier(); REGISTRY = _registry; - AVAILABILITY_ORACLE = _availabilityOracle; FEE_JUICE_PORTAL = _fpcJuicePortal; INBOX = new Inbox(address(this), Constants.L1_TO_L2_MSG_SUBTREE_HEIGHT); OUTBOX = new Outbox(address(this)); @@ -171,9 +168,17 @@ contract Rollup is Leonidas, IRollup, ITestRollup { vkTreeRoot = _vkTreeRoot; } + function computeTxsEffectsHash(bytes calldata _body) + external + view + override(IRollup) + returns (bytes32) + { + return TxsDecoder.decode(_body); + } + /** - * @notice Published the body and propose the block - * @dev This should likely be purged in the future as it is a convenience method + * @notice Publishes the body and propose the block * @dev `eth_log_handlers` rely on this function * * @param _header - The L2 block header @@ -182,7 +187,7 @@ contract Rollup is Leonidas, IRollup, ITestRollup { * @param _signatures - Signatures from the validators * @param _body - The body of the L2 block */ - function proposeWithBody( + function propose( bytes calldata _header, bytes32 _archive, bytes32 _blockHash, @@ -190,27 +195,57 @@ contract Rollup is Leonidas, IRollup, ITestRollup { SignatureLib.Signature[] memory _signatures, bytes calldata _body ) external override(IRollup) { - AVAILABILITY_ORACLE.publish(_body); - propose(_header, _archive, _blockHash, _txHashes, _signatures); - } + bytes32 txsEffectsHash = TxsDecoder.decode(_body); - /** - * @notice Published the body and propose the block - * @dev This should likely be purged in the future as it is a convenience method - * @dev `eth_log_handlers` rely on this function - * @param _header - The L2 block header - * @param _archive - A root of the archive tree after the L2 block is applied - * @param _blockHash - The poseidon2 hash of the header added to the archive tree in the rollup circuit - * @param _body - The body of the L2 block - */ - function propose( - bytes calldata _header, - bytes32 _archive, - bytes32 _blockHash, - bytes calldata _body - ) external override(IRollup) { - AVAILABILITY_ORACLE.publish(_body); - propose(_header, _archive, _blockHash); + // Decode and validate header + HeaderLib.Header memory header = HeaderLib.decode(_header); + + bytes32 digest = keccak256(abi.encode(_archive, _txHashes)); + setupEpoch(); + _validateHeader({ + _header: header, + _signatures: _signatures, + _digest: digest, + _currentTime: block.timestamp, + _txEffectsHash: txsEffectsHash, + _flags: DataStructures.ExecutionFlags({ignoreDA: false, ignoreSignatures: false}) + }); + + blocks[pendingBlockCount++] = BlockLog({ + archive: _archive, + blockHash: _blockHash, + slotNumber: header.globalVariables.slotNumber.toUint128() + }); + + // @note The block number here will always be >=1 as the genesis block is at 0 + bytes32 inHash = INBOX.consume(header.globalVariables.blockNumber); + if (header.contentCommitment.inHash != inHash) { + revert Errors.Rollup__InvalidInHash(inHash, header.contentCommitment.inHash); + } + + // TODO(#7218): Revert to fixed height tree for outbox, currently just providing min as interim + // Min size = smallest path of the rollup tree + 1 + (uint256 min,) = MerkleLib.computeMinMaxPathLength(header.contentCommitment.numTxs); + uint256 l2ToL1TreeMinHeight = min + 1; + OUTBOX.insert( + header.globalVariables.blockNumber, header.contentCommitment.outHash, l2ToL1TreeMinHeight + ); + + emit L2BlockProposed(header.globalVariables.blockNumber); + + // Automatically flag the block as proven if we have cheated and set assumeProvenUntilBlockNumber. + if (header.globalVariables.blockNumber < assumeProvenUntilBlockNumber) { + provenBlockCount += 1; + + if (header.globalVariables.coinbase != address(0) && header.totalFees > 0) { + // @note This will currently fail if there are insufficient funds in the bridge + // which WILL happen for the old version after an upgrade where the bridge follow. + // Consider allowing a failure. See #7938. + FEE_JUICE_PORTAL.distributeFees(header.globalVariables.coinbase, header.totalFees); + } + + emit L2ProofVerified(header.globalVariables.blockNumber, "CHEAT"); + } } /** @@ -418,91 +453,11 @@ contract Rollup is Leonidas, IRollup, ITestRollup { SignatureLib.Signature[] memory _signatures, bytes32 _digest, uint256 _currentTime, + bytes32 _txsEffectsHash, DataStructures.ExecutionFlags memory _flags ) external view override(IRollup) { HeaderLib.Header memory header = HeaderLib.decode(_header); - _validateHeader(header, _signatures, _digest, _currentTime, _flags); - } - - /** - * @notice Propose an incoming L2 block with signatures - * - * @param _header - The L2 block header - * @param _archive - A root of the archive tree after the L2 block is applied - * @param _blockHash - The poseidon2 hash of the header added to the archive tree in the rollup circuit - * @param _signatures - Signatures from the validators - */ - function propose( - bytes calldata _header, - bytes32 _archive, - bytes32 _blockHash, - bytes32[] memory _txHashes, - SignatureLib.Signature[] memory _signatures - ) public override(IRollup) { - // Decode and validate header - HeaderLib.Header memory header = HeaderLib.decode(_header); - setupEpoch(); - - bytes32 digest = keccak256(abi.encode(_archive, _txHashes)); - _validateHeader({ - _header: header, - _signatures: _signatures, - _digest: digest, - _currentTime: block.timestamp, - _flags: DataStructures.ExecutionFlags({ignoreDA: false, ignoreSignatures: false}) - }); - - blocks[pendingBlockCount++] = BlockLog({ - archive: _archive, - blockHash: _blockHash, - slotNumber: header.globalVariables.slotNumber.toUint128() - }); - - // @note The block number here will always be >=1 as the genesis block is at 0 - bytes32 inHash = INBOX.consume(header.globalVariables.blockNumber); - if (header.contentCommitment.inHash != inHash) { - revert Errors.Rollup__InvalidInHash(inHash, header.contentCommitment.inHash); - } - - // TODO(#7218): Revert to fixed height tree for outbox, currently just providing min as interim - // Min size = smallest path of the rollup tree + 1 - (uint256 min,) = MerkleLib.computeMinMaxPathLength(header.contentCommitment.numTxs); - uint256 l2ToL1TreeMinHeight = min + 1; - OUTBOX.insert( - header.globalVariables.blockNumber, header.contentCommitment.outHash, l2ToL1TreeMinHeight - ); - - emit L2BlockProposed(header.globalVariables.blockNumber); - - // Automatically flag the block as proven if we have cheated and set assumeProvenUntilBlockNumber. - if (header.globalVariables.blockNumber < assumeProvenUntilBlockNumber) { - provenBlockCount += 1; - - if (header.globalVariables.coinbase != address(0) && header.totalFees > 0) { - // @note This will currently fail if there are insufficient funds in the bridge - // which WILL happen for the old version after an upgrade where the bridge follow. - // Consider allowing a failure. See #7938. - FEE_JUICE_PORTAL.distributeFees(header.globalVariables.coinbase, header.totalFees); - } - - emit L2ProofVerified(header.globalVariables.blockNumber, "CHEAT"); - } - } - - /** - * @notice Propose a L2 block without signatures - * - * @param _header - The L2 block header - * @param _archive - A root of the archive tree after the L2 block is applied - * @param _blockHash - The poseidon2 hash of the header added to the archive tree in the rollup circuit - */ - function propose(bytes calldata _header, bytes32 _archive, bytes32 _blockHash) - public - override(IRollup) - { - SignatureLib.Signature[] memory emptySignatures = new SignatureLib.Signature[](0); - bytes32[] memory emptyTxHashes = new bytes32[](0); - propose(_header, _archive, _blockHash, emptyTxHashes, emptySignatures); + _validateHeader(header, _signatures, _digest, _currentTime, _txsEffectsHash, _flags); } /** @@ -529,9 +484,10 @@ contract Rollup is Leonidas, IRollup, ITestRollup { SignatureLib.Signature[] memory _signatures, bytes32 _digest, uint256 _currentTime, + bytes32 _txEffectsHash, DataStructures.ExecutionFlags memory _flags ) internal view { - _validateHeaderForSubmissionBase(_header, _currentTime, _flags); + _validateHeaderForSubmissionBase(_header, _currentTime, _txEffectsHash, _flags); _validateHeaderForSubmissionSequencerSelection( _header.globalVariables.slotNumber, _signatures, _digest, _currentTime, _flags ); @@ -592,7 +548,7 @@ contract Rollup is Leonidas, IRollup, ITestRollup { * - The last archive root in the header MUST match the current archive * - The slot MUST be larger than the slot of the previous block (ensures single block per slot) * - The timestamp MUST be equal to GENESIS_TIME + slot * SLOT_DURATION - * - The availability oracle MUST return true for availability of txsEffectsHash + * - The `txsEffectsHash` of the header must match the computed `_txsEffectsHash` * - This can be relaxed to happen at the time of `submitProof` instead * * @param _header - The header to validate @@ -600,6 +556,7 @@ contract Rollup is Leonidas, IRollup, ITestRollup { function _validateHeaderForSubmissionBase( HeaderLib.Header memory _header, uint256 _currentTime, + bytes32 _txsEffectsHash, DataStructures.ExecutionFlags memory _flags ) internal view { if (block.chainid != _header.globalVariables.chainId) { @@ -646,10 +603,8 @@ contract Rollup is Leonidas, IRollup, ITestRollup { revert Errors.Rollup__TimestampInFuture(_currentTime, timestamp); } - // Check if the data is available using availability oracle (change availability oracle if you want a different DA layer) - if ( - !_flags.ignoreDA && !AVAILABILITY_ORACLE.isAvailable(_header.contentCommitment.txsEffectsHash) - ) { + // Check if the data is available + if (!_flags.ignoreDA && _header.contentCommitment.txsEffectsHash != _txsEffectsHash) { revert Errors.Rollup__UnavailableTxs(_header.contentCommitment.txsEffectsHash); } } diff --git a/l1-contracts/src/core/availability_oracle/AvailabilityOracle.sol b/l1-contracts/src/core/availability_oracle/AvailabilityOracle.sol deleted file mode 100644 index 5553d5a2143..00000000000 --- a/l1-contracts/src/core/availability_oracle/AvailabilityOracle.sol +++ /dev/null @@ -1,32 +0,0 @@ -// SPDX-License-Identifier: Apache-2.0 -// Copyright 2024 Aztec Labs. -pragma solidity >=0.8.18; - -// Interfaces -import {IAvailabilityOracle} from "./../interfaces/IAvailabilityOracle.sol"; - -// Libraries -import {TxsDecoder} from "./../libraries/decoders/TxsDecoder.sol"; - -/** - * @title AvailabilityOracle - * @author Aztec Labs - * @notice An availability oracle that uses L1 calldata for publication - */ -contract AvailabilityOracle is IAvailabilityOracle { - mapping(bytes32 txsHash => bool available) public override(IAvailabilityOracle) isAvailable; - - /** - * @notice Publishes transactions and marks its commitment, the TxsEffectsHash, as available - * @param _body - The block body - * @return txsEffectsHash - The TxsEffectsHash - */ - function publish(bytes calldata _body) external override(IAvailabilityOracle) returns (bytes32) { - bytes32 txsEffectsHash = TxsDecoder.decode(_body); - isAvailable[txsEffectsHash] = true; - - emit TxsPublished(txsEffectsHash); - - return txsEffectsHash; - } -} diff --git a/l1-contracts/src/core/interfaces/IAvailabilityOracle.sol b/l1-contracts/src/core/interfaces/IAvailabilityOracle.sol deleted file mode 100644 index 9c5f23ab093..00000000000 --- a/l1-contracts/src/core/interfaces/IAvailabilityOracle.sol +++ /dev/null @@ -1,11 +0,0 @@ -// SPDX-License-Identifier: Apache-2.0 -// Copyright 2024 Aztec Labs. -pragma solidity >=0.8.18; - -interface IAvailabilityOracle { - event TxsPublished(bytes32 txsEffectsHash); - - function publish(bytes calldata _body) external returns (bytes32); - - function isAvailable(bytes32 _txsEffectsHash) external view returns (bool); -} diff --git a/l1-contracts/src/core/interfaces/IRollup.sol b/l1-contracts/src/core/interfaces/IRollup.sol index 23c3012372b..ce99bd64921 100644 --- a/l1-contracts/src/core/interfaces/IRollup.sol +++ b/l1-contracts/src/core/interfaces/IRollup.sol @@ -25,6 +25,7 @@ interface IRollup { SignatureLib.Signature[] memory _signatures, bytes32 _digest, uint256 _currentTime, + bytes32 _txsEffecstHash, DataStructures.ExecutionFlags memory _flags ) external view; @@ -34,7 +35,7 @@ interface IRollup { function OUTBOX() external view returns (IOutbox); - function proposeWithBody( + function propose( bytes calldata _header, bytes32 _archive, bytes32 _blockHash, @@ -42,20 +43,6 @@ interface IRollup { SignatureLib.Signature[] memory _signatures, bytes calldata _body ) external; - function propose( - bytes calldata _header, - bytes32 _archive, - bytes32 _blockHash, - bytes calldata _body - ) external; - function propose(bytes calldata _header, bytes32 _archive, bytes32 _blockHash) external; - function propose( - bytes calldata _header, - bytes32 _archive, - bytes32 _blockHash, - bytes32[] memory _txHashes, - SignatureLib.Signature[] memory _signatures - ) external; function submitBlockRootProof( bytes calldata _header, @@ -79,4 +66,5 @@ interface IRollup { function archive() external view returns (bytes32); function archiveAt(uint256 _blockNumber) external view returns (bytes32); + function computeTxsEffectsHash(bytes calldata _body) external view returns (bytes32); } diff --git a/l1-contracts/terraform/main.tf b/l1-contracts/terraform/main.tf index 4b7a1877459..7b9c55131aa 100644 --- a/l1-contracts/terraform/main.tf +++ b/l1-contracts/terraform/main.tf @@ -20,15 +20,6 @@ output "rollup_contract_address" { value = var.ROLLUP_CONTRACT_ADDRESS } -variable "AVAILABILITY_ORACLE_CONTRACT_ADDRESS" { - type = string - default = "" -} - -output "availability_oracle_contract_address" { - value = var.AVAILABILITY_ORACLE_CONTRACT_ADDRESS -} - variable "REGISTRY_CONTRACT_ADDRESS" { type = string default = "" diff --git a/l1-contracts/test/Rollup.t.sol b/l1-contracts/test/Rollup.t.sol index c908099f6a0..e9b1f899eaa 100644 --- a/l1-contracts/test/Rollup.t.sol +++ b/l1-contracts/test/Rollup.t.sol @@ -15,7 +15,7 @@ import {Rollup} from "../src/core/Rollup.sol"; import {IFeeJuicePortal} from "../src/core/interfaces/IFeeJuicePortal.sol"; import {FeeJuicePortal} from "../src/core/FeeJuicePortal.sol"; import {Leonidas} from "../src/core/sequencer_selection/Leonidas.sol"; -import {AvailabilityOracle} from "../src/core/availability_oracle/AvailabilityOracle.sol"; +import {SignatureLib} from "../src/core/sequencer_selection/SignatureLib.sol"; import {NaiveMerkle} from "./merkle/Naive.sol"; import {MerkleTestUtil} from "./merkle/TestUtil.sol"; import {PortalERC20} from "./portals/PortalERC20.sol"; @@ -37,7 +37,7 @@ contract RollupTest is DecoderBase { PortalERC20 internal portalERC20; FeeJuicePortal internal feeJuicePortal; - AvailabilityOracle internal availabilityOracle; + SignatureLib.Signature[] internal signatures; /** * @notice Set up the contracts needed for the tests with time aligned to the provided block name @@ -53,7 +53,6 @@ contract RollupTest is DecoderBase { } registry = new Registry(address(this)); - availabilityOracle = new AvailabilityOracle(); portalERC20 = new PortalERC20(); feeJuicePortal = new FeeJuicePortal(address(this)); portalERC20.mint(address(feeJuicePortal), Constants.FEE_JUICE_INITIAL_MINT); @@ -62,7 +61,6 @@ contract RollupTest is DecoderBase { ); rollup = new Rollup( registry, - availabilityOracle, IFeeJuicePortal(address(feeJuicePortal)), bytes32(0), address(this), @@ -83,15 +81,12 @@ contract RollupTest is DecoderBase { bytes memory header = data.header; bytes32 archive = data.archive; bytes memory body = data.body; - - // Progress time as necessary - vm.warp(max(block.timestamp, data.decodedHeader.globalVariables.timestamp)); - availabilityOracle.publish(body); + bytes32[] memory txHashes = new bytes32[](0); // We jump to the time of the block. (unless it is in the past) vm.warp(max(block.timestamp, data.decodedHeader.globalVariables.timestamp)); - rollup.propose(header, archive, bytes32(0)); + rollup.propose(header, archive, bytes32(0), txHashes, signatures, body); rollup.submitBlockRootProof(header, archive, bytes32(0), "", ""); @@ -192,6 +187,7 @@ contract RollupTest is DecoderBase { bytes memory header = data.header; bytes32 archive = data.archive; bytes memory body = data.body; + bytes32[] memory txHashes = new bytes32[](0); // Progress time as necessary vm.warp(max(block.timestamp, data.decodedHeader.globalVariables.timestamp)); @@ -199,7 +195,6 @@ contract RollupTest is DecoderBase { assembly { mstore(add(header, add(0x20, 0x0248)), feeAmount) } - availabilityOracle.publish(body); assertEq(portalERC20.balanceOf(address(rollup)), 0, "invalid rollup balance"); @@ -213,7 +208,7 @@ contract RollupTest is DecoderBase { assertEq(coinbaseBalance, 0, "invalid initial coinbase balance"); // Assert that balance have NOT been increased by proposing the block - rollup.propose(header, archive, bytes32(0)); + rollup.propose(header, archive, bytes32(0), txHashes, signatures, body); assertEq(portalERC20.balanceOf(coinbase), 0, "invalid coinbase balance"); vm.expectRevert( @@ -258,10 +253,10 @@ contract RollupTest is DecoderBase { bytes memory header = data.header; bytes32 archive = data.archive; bytes memory body = data.body; + bytes32[] memory txHashes = new bytes32[](0); vm.warp(max(block.timestamp, data.decodedHeader.globalVariables.timestamp)); - availabilityOracle.publish(body); - rollup.propose(header, archive, bytes32(0)); + rollup.propose(header, archive, bytes32(0), txHashes, signatures, body); vm.expectRevert(abi.encodeWithSelector(Errors.Rollup__NonSequentialProving.selector)); rollup.submitBlockRootProof(header, archive, bytes32(0), "", ""); @@ -290,16 +285,15 @@ contract RollupTest is DecoderBase { bytes memory header = data.header; bytes32 archive = data.archive; bytes memory body = data.body; + bytes32[] memory txHashes = new bytes32[](0); assembly { // TODO: Hardcoding offsets in the middle of tests is annoying to say the least. mstore(add(header, add(0x20, 0x0174)), 0x420) } - availabilityOracle.publish(body); - vm.expectRevert(abi.encodeWithSelector(Errors.Rollup__InvalidBlockNumber.selector, 1, 0x420)); - rollup.propose(header, archive, bytes32(0)); + rollup.propose(header, archive, bytes32(0), txHashes, signatures, body); } function testRevertInvalidChainId() public setUpFor("empty_block_1") { @@ -307,16 +301,14 @@ contract RollupTest is DecoderBase { bytes memory header = data.header; bytes32 archive = data.archive; bytes memory body = data.body; + bytes32[] memory txHashes = new bytes32[](0); assembly { - // TODO: Hardcoding offsets in the middle of tests is annoying to say the least. mstore(add(header, add(0x20, 0x0134)), 0x420) } - availabilityOracle.publish(body); - vm.expectRevert(abi.encodeWithSelector(Errors.Rollup__InvalidChainId.selector, 31337, 0x420)); - rollup.propose(header, archive, bytes32(0)); + rollup.propose(header, archive, bytes32(0), txHashes, signatures, body); } function testRevertInvalidVersion() public setUpFor("empty_block_1") { @@ -324,15 +316,14 @@ contract RollupTest is DecoderBase { bytes memory header = data.header; bytes32 archive = data.archive; bytes memory body = data.body; + bytes32[] memory txHashes = new bytes32[](0); assembly { mstore(add(header, add(0x20, 0x0154)), 0x420) } - availabilityOracle.publish(body); - vm.expectRevert(abi.encodeWithSelector(Errors.Rollup__InvalidVersion.selector, 1, 0x420)); - rollup.propose(header, archive, bytes32(0)); + rollup.propose(header, archive, bytes32(0), txHashes, signatures, body); } function testRevertInvalidTimestamp() public setUpFor("empty_block_1") { @@ -340,6 +331,7 @@ contract RollupTest is DecoderBase { bytes memory header = data.header; bytes32 archive = data.archive; bytes memory body = data.body; + bytes32[] memory txHashes = new bytes32[](0); uint256 realTs = data.decodedHeader.globalVariables.timestamp; uint256 badTs = realTs + 1; @@ -350,10 +342,8 @@ contract RollupTest is DecoderBase { mstore(add(header, add(0x20, 0x01b4)), badTs) } - availabilityOracle.publish(body); - vm.expectRevert(abi.encodeWithSelector(Errors.Rollup__InvalidTimestamp.selector, realTs, badTs)); - rollup.propose(header, archive, bytes32(0)); + rollup.propose(header, archive, bytes32(0), txHashes, signatures, body); } function testBlocksWithAssumeProven() public setUpFor("mixed_block_1") { @@ -434,6 +424,7 @@ contract RollupTest is DecoderBase { bytes32 archive = full.block.archive; bytes memory body = full.block.body; uint32 numTxs = full.block.numTxs; + bytes32[] memory txHashes = new bytes32[](0); // Overwrite some timestamps if needed if (_slotNumber != 0) { @@ -452,9 +443,7 @@ contract RollupTest is DecoderBase { _populateInbox(full.populate.sender, full.populate.recipient, full.populate.l1ToL2Content); - availabilityOracle.publish(body); - - rollup.propose(header, archive, bytes32(0)); + rollup.propose(header, archive, bytes32(0), txHashes, signatures, body); if (_submitProof) { uint256 pre = rollup.provenBlockCount(); diff --git a/l1-contracts/test/portals/TokenPortal.t.sol b/l1-contracts/test/portals/TokenPortal.t.sol index 6241c57880e..96104d54ddb 100644 --- a/l1-contracts/test/portals/TokenPortal.t.sol +++ b/l1-contracts/test/portals/TokenPortal.t.sol @@ -4,7 +4,6 @@ import "forge-std/Test.sol"; // Rollup Processor import {Rollup} from "../../src/core/Rollup.sol"; -import {AvailabilityOracle} from "../../src/core/availability_oracle/AvailabilityOracle.sol"; import {Constants} from "../../src/core/libraries/ConstantsGen.sol"; import {Registry} from "../../src/core/messagebridge/Registry.sol"; import {DataStructures} from "../../src/core/libraries/DataStructures.sol"; @@ -61,14 +60,8 @@ contract TokenPortalTest is Test { function setUp() public { registry = new Registry(address(this)); portalERC20 = new PortalERC20(); - rollup = new Rollup( - registry, - new AvailabilityOracle(), - IFeeJuicePortal(address(0)), - bytes32(0), - address(this), - new address[](0) - ); + rollup = + new Rollup(registry, IFeeJuicePortal(address(0)), bytes32(0), address(this), new address[](0)); inbox = rollup.INBOX(); outbox = rollup.OUTBOX(); diff --git a/l1-contracts/test/portals/UniswapPortal.t.sol b/l1-contracts/test/portals/UniswapPortal.t.sol index 06269177402..0f506b279ef 100644 --- a/l1-contracts/test/portals/UniswapPortal.t.sol +++ b/l1-contracts/test/portals/UniswapPortal.t.sol @@ -4,7 +4,6 @@ import "forge-std/Test.sol"; // Rollup Processor import {Rollup} from "../../src/core/Rollup.sol"; -import {AvailabilityOracle} from "../../src/core/availability_oracle/AvailabilityOracle.sol"; import {Registry} from "../../src/core/messagebridge/Registry.sol"; import {DataStructures} from "../../src/core/libraries/DataStructures.sol"; import {DataStructures as PortalDataStructures} from "./DataStructures.sol"; @@ -53,14 +52,8 @@ contract UniswapPortalTest is Test { vm.selectFork(forkId); registry = new Registry(address(this)); - rollup = new Rollup( - registry, - new AvailabilityOracle(), - IFeeJuicePortal(address(0)), - bytes32(0), - address(this), - new address[](0) - ); + rollup = + new Rollup(registry, IFeeJuicePortal(address(0)), bytes32(0), address(this), new address[](0)); registry.upgrade(address(rollup)); daiTokenPortal = new TokenPortal(); diff --git a/l1-contracts/test/sparta/Sparta.t.sol b/l1-contracts/test/sparta/Sparta.t.sol index b0f18cf37b7..dc559ede666 100644 --- a/l1-contracts/test/sparta/Sparta.t.sol +++ b/l1-contracts/test/sparta/Sparta.t.sol @@ -15,7 +15,6 @@ import {Outbox} from "../../src/core/messagebridge/Outbox.sol"; import {Errors} from "../../src/core/libraries/Errors.sol"; import {Rollup} from "../../src/core/Rollup.sol"; import {Leonidas} from "../../src/core/sequencer_selection/Leonidas.sol"; -import {AvailabilityOracle} from "../../src/core/availability_oracle/AvailabilityOracle.sol"; import {NaiveMerkle} from "../merkle/Naive.sol"; import {MerkleTestUtil} from "../merkle/TestUtil.sol"; import {PortalERC20} from "../portals/PortalERC20.sol"; @@ -37,8 +36,6 @@ contract SpartaTest is DecoderBase { TxsDecoderHelper internal txsHelper; PortalERC20 internal portalERC20; - AvailabilityOracle internal availabilityOracle; - mapping(address validator => uint256 privateKey) internal privateKeys; SignatureLib.Signature internal emptySignature; @@ -66,15 +63,9 @@ contract SpartaTest is DecoderBase { } registry = new Registry(address(this)); - availabilityOracle = new AvailabilityOracle(); portalERC20 = new PortalERC20(); rollup = new Rollup( - registry, - availabilityOracle, - IFeeJuicePortal(address(0)), - bytes32(0), - address(this), - initialValidators + registry, IFeeJuicePortal(address(0)), bytes32(0), address(this), initialValidators ); inbox = Inbox(address(rollup.INBOX())); outbox = Outbox(address(rollup.OUTBOX())); @@ -175,8 +166,6 @@ contract SpartaTest is DecoderBase { _populateInbox(full.populate.sender, full.populate.recipient, full.populate.l1ToL2Content); - availabilityOracle.publish(body); - ree.proposer = rollup.getCurrentProposer(); ree.shouldRevert = false; @@ -222,13 +211,15 @@ contract SpartaTest is DecoderBase { } vm.prank(ree.proposer); - rollup.propose(header, archive, bytes32(0), txHashes, signatures); + rollup.propose(header, archive, bytes32(0), txHashes, signatures, body); if (ree.shouldRevert) { return; } } else { - rollup.propose(header, archive, bytes32(0)); + SignatureLib.Signature[] memory signatures = new SignatureLib.Signature[](0); + + rollup.propose(header, archive, bytes32(0), txHashes, signatures, body); } assertEq(_expectRevert, ree.shouldRevert, "Does not match revert expectation"); diff --git a/noir/noir-repo/compiler/noirc_frontend/src/elaborator/types.rs b/noir/noir-repo/compiler/noirc_frontend/src/elaborator/types.rs index 39ef4e0bb8e..46c779dc4ff 100644 --- a/noir/noir-repo/compiler/noirc_frontend/src/elaborator/types.rs +++ b/noir/noir-repo/compiler/noirc_frontend/src/elaborator/types.rs @@ -118,7 +118,15 @@ impl<'context> Elaborator<'context> { let fields = self.resolve_type_inner(*fields, kind); Type::FmtString(Box::new(resolved_size), Box::new(fields)) } - Quoted(quoted) => Type::Quoted(quoted), + Quoted(quoted) => { + let in_function = matches!(self.current_item, Some(DependencyId::Function(_))); + if in_function && !self.in_comptime_context() { + let span = typ.span; + let typ = quoted.to_string(); + self.push_err(ResolverError::ComptimeTypeInRuntimeCode { span, typ }); + } + Type::Quoted(quoted) + } Unit => Type::Unit, Unspecified => { let span = typ.span; diff --git a/noir/noir-repo/compiler/noirc_frontend/src/hir/comptime/interpreter/builtin.rs b/noir/noir-repo/compiler/noirc_frontend/src/hir/comptime/interpreter/builtin.rs index 899d62ecb61..166c6479b15 100644 --- a/noir/noir-repo/compiler/noirc_frontend/src/hir/comptime/interpreter/builtin.rs +++ b/noir/noir-repo/compiler/noirc_frontend/src/hir/comptime/interpreter/builtin.rs @@ -189,6 +189,9 @@ impl<'local, 'context> Interpreter<'local, 'context> { "typed_expr_as_function_definition" => { typed_expr_as_function_definition(interner, arguments, return_type, location) } + "typed_expr_get_type" => { + typed_expr_get_type(interner, arguments, return_type, location) + } "unresolved_type_is_field" => unresolved_type_is_field(interner, arguments, location), "zeroed" => zeroed(return_type), _ => { @@ -1070,6 +1073,7 @@ fn trait_impl_trait_generic_args( Ok(Value::Slice(trait_generics, slice_type)) } +// fn as_function_definition(self) -> Option fn typed_expr_as_function_definition( interner: &NodeInterner, arguments: Vec<(Value, Location)>, @@ -1087,6 +1091,28 @@ fn typed_expr_as_function_definition( option(return_type, option_value) } +// fn get_type(self) -> Option +fn typed_expr_get_type( + interner: &NodeInterner, + arguments: Vec<(Value, Location)>, + return_type: Type, + location: Location, +) -> IResult { + let self_argument = check_one_argument(arguments, location)?; + let typed_expr = get_typed_expr(self_argument)?; + let option_value = if let TypedExpr::ExprId(expr_id) = typed_expr { + let typ = interner.id_type(expr_id); + if typ == Type::Error { + None + } else { + Some(Value::Type(typ)) + } + } else { + None + }; + option(return_type, option_value) +} + // fn is_field(self) -> bool fn unresolved_type_is_field( interner: &NodeInterner, diff --git a/noir/noir-repo/compiler/noirc_frontend/src/hir/resolution/errors.rs b/noir/noir-repo/compiler/noirc_frontend/src/hir/resolution/errors.rs index 5abc94b89a2..ec22c8f1986 100644 --- a/noir/noir-repo/compiler/noirc_frontend/src/hir/resolution/errors.rs +++ b/noir/noir-repo/compiler/noirc_frontend/src/hir/resolution/errors.rs @@ -126,6 +126,8 @@ pub enum ResolverError { OverflowInType { lhs: u32, op: crate::BinaryTypeOperator, rhs: u32, span: Span }, #[error("`quote` cannot be used in runtime code")] QuoteInRuntimeCode { span: Span }, + #[error("Comptime-only type `{typ}` cannot be used in runtime code")] + ComptimeTypeInRuntimeCode { typ: String, span: Span }, } impl ResolverError { @@ -513,6 +515,13 @@ impl<'a> From<&'a ResolverError> for Diagnostic { *span, ) }, + ResolverError::ComptimeTypeInRuntimeCode { typ, span } => { + Diagnostic::simple_error( + format!("Comptime-only type `{typ}` cannot be used in runtime code"), + "Comptime-only type used here".to_string(), + *span, + ) + }, } } } diff --git a/noir/noir-repo/compiler/noirc_frontend/src/monomorphization/errors.rs b/noir/noir-repo/compiler/noirc_frontend/src/monomorphization/errors.rs index 66db72eef55..7f4172017e2 100644 --- a/noir/noir-repo/compiler/noirc_frontend/src/monomorphization/errors.rs +++ b/noir/noir-repo/compiler/noirc_frontend/src/monomorphization/errors.rs @@ -9,6 +9,7 @@ pub enum MonomorphizationError { InternalError { message: &'static str, location: Location }, InterpreterError(InterpreterError), ComptimeFnInRuntimeCode { name: String, location: Location }, + ComptimeTypeInRuntimeCode { typ: String, location: Location }, } impl MonomorphizationError { @@ -17,6 +18,7 @@ impl MonomorphizationError { MonomorphizationError::UnknownArrayLength { location, .. } | MonomorphizationError::InternalError { location, .. } | MonomorphizationError::ComptimeFnInRuntimeCode { location, .. } + | MonomorphizationError::ComptimeTypeInRuntimeCode { location, .. } | MonomorphizationError::NoDefaultType { location, .. } => *location, MonomorphizationError::InterpreterError(error) => error.get_location(), } @@ -51,6 +53,11 @@ impl MonomorphizationError { "Comptime functions must be in a comptime block to be called".into(); return CustomDiagnostic::simple_error(message, secondary, location.span); } + MonomorphizationError::ComptimeTypeInRuntimeCode { typ, location } => { + let message = format!("Comptime-only type `{typ}` used in runtime code"); + let secondary = "Comptime type used here".into(); + return CustomDiagnostic::simple_error(message, secondary, location.span); + } }; let location = self.location(); diff --git a/noir/noir-repo/compiler/noirc_frontend/src/monomorphization/mod.rs b/noir/noir-repo/compiler/noirc_frontend/src/monomorphization/mod.rs index 9357cc65c14..fd06a2b04a8 100644 --- a/noir/noir-repo/compiler/noirc_frontend/src/monomorphization/mod.rs +++ b/noir/noir-repo/compiler/noirc_frontend/src/monomorphization/mod.rs @@ -1056,7 +1056,10 @@ impl<'interner> Monomorphizer<'interner> { let message = "Unexpected Type::Error found during monomorphization"; return Err(MonomorphizationError::InternalError { message, location }); } - HirType::Quoted(_) => unreachable!("Tried to translate Code type into runtime code"), + HirType::Quoted(typ) => { + let typ = typ.to_string(); + return Err(MonomorphizationError::ComptimeTypeInRuntimeCode { typ, location }); + } }) } diff --git a/noir/noir-repo/compiler/noirc_frontend/src/parser/errors.rs b/noir/noir-repo/compiler/noirc_frontend/src/parser/errors.rs index ad6f6b928ab..6ba4cb68500 100644 --- a/noir/noir-repo/compiler/noirc_frontend/src/parser/errors.rs +++ b/noir/noir-repo/compiler/noirc_frontend/src/parser/errors.rs @@ -1,6 +1,7 @@ use crate::ast::{Expression, IntegerBitSize}; use crate::lexer::errors::LexerErrorKind; use crate::lexer::token::Token; +use crate::token::TokenKind; use small_ord_set::SmallOrdSet; use thiserror::Error; @@ -211,8 +212,17 @@ impl<'a> From<&'a ParserError> for Diagnostic { other => Diagnostic::simple_error(format!("{other}"), String::new(), error.span), }, None => { - let primary = error.to_string(); - Diagnostic::simple_error(primary, String::new(), error.span) + if matches!( + error.found.kind(), + TokenKind::InnerDocComment | TokenKind::OuterDocComment + ) { + let primary = "This doc comment doesn't document anything".to_string(); + let secondary = "Consider changing it to a regular `//` comment".to_string(); + Diagnostic::simple_error(primary, secondary, error.span) + } else { + let primary = error.to_string(); + Diagnostic::simple_error(primary, String::new(), error.span) + } } } } diff --git a/noir/noir-repo/compiler/noirc_frontend/src/tests.rs b/noir/noir-repo/compiler/noirc_frontend/src/tests.rs index 64c9b7471b4..414bfa5fa5b 100644 --- a/noir/noir-repo/compiler/noirc_frontend/src/tests.rs +++ b/noir/noir-repo/compiler/noirc_frontend/src/tests.rs @@ -3451,3 +3451,14 @@ fn constrained_reference_to_unconstrained() { panic!("Expected an error about passing a constrained reference to unconstrained"); }; } + +#[test] +fn comptime_type_in_runtime_code() { + let source = "pub fn foo(_f: FunctionDefinition) {}"; + let errors = get_program_errors(source); + assert_eq!(errors.len(), 1); + assert!(matches!( + errors[0].0, + CompilationError::ResolverError(ResolverError::ComptimeTypeInRuntimeCode { .. }) + )); +} diff --git a/noir/noir-repo/docs/docs/noir/standard_library/meta/typed_expr.md b/noir/noir-repo/docs/docs/noir/standard_library/meta/typed_expr.md index eacfd9c1230..24b23f2ec19 100644 --- a/noir/noir-repo/docs/docs/noir/standard_library/meta/typed_expr.md +++ b/noir/noir-repo/docs/docs/noir/standard_library/meta/typed_expr.md @@ -6,8 +6,14 @@ title: TypedExpr ## Methods -### as_function_definition +### get_type #include_code as_function_definition noir_stdlib/src/meta/typed_expr.nr rust -If this expression refers to a function definitions, returns it. Otherwise returns `Option::none()`. \ No newline at end of file +If this expression refers to a function definitions, returns it. Otherwise returns `Option::none()`. + +### get_type + +#include_code get_type noir_stdlib/src/meta/typed_expr.nr rust + +Returns the type of the expression, or `Option::none()` if there were errors when the expression was previously resolved. \ No newline at end of file diff --git a/noir/noir-repo/noir_stdlib/src/meta/expr.nr b/noir/noir-repo/noir_stdlib/src/meta/expr.nr index 642dbecc36b..72e1a88cea8 100644 --- a/noir/noir-repo/noir_stdlib/src/meta/expr.nr +++ b/noir/noir-repo/noir_stdlib/src/meta/expr.nr @@ -5,129 +5,129 @@ use crate::meta::op::BinaryOp; impl Expr { #[builtin(expr_as_array)] // docs:start:as_array - fn as_array(self) -> Option<[Expr]> {} + comptime fn as_array(self) -> Option<[Expr]> {} // docs:end:as_array #[builtin(expr_as_assert)] // docs:start:as_assert - fn as_assert(self) -> Option<(Expr, Option)> {} + comptime fn as_assert(self) -> Option<(Expr, Option)> {} // docs:end:as_assert #[builtin(expr_as_assert_eq)] // docs:start:as_assert_eq - fn as_assert_eq(self) -> Option<(Expr, Expr, Option)> {} + comptime fn as_assert_eq(self) -> Option<(Expr, Expr, Option)> {} // docs:end:as_assert_eq #[builtin(expr_as_assign)] // docs:start:as_assign - fn as_assign(self) -> Option<(Expr, Expr)> {} + comptime fn as_assign(self) -> Option<(Expr, Expr)> {} // docs:end:as_assign #[builtin(expr_as_integer)] // docs:start:as_integer - fn as_integer(self) -> Option<(Field, bool)> {} + comptime fn as_integer(self) -> Option<(Field, bool)> {} // docs:end:as_integer #[builtin(expr_as_binary_op)] // docs:start:as_binary_op - fn as_binary_op(self) -> Option<(Expr, BinaryOp, Expr)> {} + comptime fn as_binary_op(self) -> Option<(Expr, BinaryOp, Expr)> {} // docs:end:as_binary_op #[builtin(expr_as_block)] // docs:start:as_block - fn as_block(self) -> Option<[Expr]> {} + comptime fn as_block(self) -> Option<[Expr]> {} // docs:end:as_block #[builtin(expr_as_bool)] // docs:start:as_bool - fn as_bool(self) -> Option {} + comptime fn as_bool(self) -> Option {} // docs:end:as_bool #[builtin(expr_as_cast)] - fn as_cast(self) -> Option<(Expr, UnresolvedType)> {} + comptime fn as_cast(self) -> Option<(Expr, UnresolvedType)> {} #[builtin(expr_as_comptime)] // docs:start:as_comptime - fn as_comptime(self) -> Option<[Expr]> {} + comptime fn as_comptime(self) -> Option<[Expr]> {} // docs:end:as_comptime #[builtin(expr_as_function_call)] // docs:start:as_function_call - fn as_function_call(self) -> Option<(Expr, [Expr])> {} + comptime fn as_function_call(self) -> Option<(Expr, [Expr])> {} // docs:end:as_function_call #[builtin(expr_as_if)] // docs:start:as_if - fn as_if(self) -> Option<(Expr, Expr, Option)> {} + comptime fn as_if(self) -> Option<(Expr, Expr, Option)> {} // docs:end:as_if #[builtin(expr_as_index)] // docs:start:as_index - fn as_index(self) -> Option<(Expr, Expr)> {} + comptime fn as_index(self) -> Option<(Expr, Expr)> {} // docs:end:as_index #[builtin(expr_as_let)] // docs:start:as_let - fn as_let(self) -> Option<(Expr, Option, Expr)> {} + comptime fn as_let(self) -> Option<(Expr, Option, Expr)> {} // docs:end:as_let #[builtin(expr_as_member_access)] // docs:start:as_member_access - fn as_member_access(self) -> Option<(Expr, Quoted)> {} + comptime fn as_member_access(self) -> Option<(Expr, Quoted)> {} // docs:end:as_member_access #[builtin(expr_as_method_call)] // docs:start:as_method_call - fn as_method_call(self) -> Option<(Expr, Quoted, [UnresolvedType], [Expr])> {} + comptime fn as_method_call(self) -> Option<(Expr, Quoted, [UnresolvedType], [Expr])> {} // docs:end:as_method_call #[builtin(expr_as_repeated_element_array)] // docs:start:as_repeated_element_array - fn as_repeated_element_array(self) -> Option<(Expr, Expr)> {} + comptime fn as_repeated_element_array(self) -> Option<(Expr, Expr)> {} // docs:end:as_repeated_element_array #[builtin(expr_as_repeated_element_slice)] // docs:start:as_repeated_element_slice - fn as_repeated_element_slice(self) -> Option<(Expr, Expr)> {} + comptime fn as_repeated_element_slice(self) -> Option<(Expr, Expr)> {} // docs:end:as_repeated_element_slice #[builtin(expr_as_slice)] // docs:start:as_slice - fn as_slice(self) -> Option<[Expr]> {} + comptime fn as_slice(self) -> Option<[Expr]> {} // docs:end:as_slice #[builtin(expr_as_tuple)] // docs:start:as_tuple - fn as_tuple(self) -> Option<[Expr]> {} + comptime fn as_tuple(self) -> Option<[Expr]> {} // docs:end:as_tuple #[builtin(expr_as_unary_op)] // docs:start:as_unary_op - fn as_unary_op(self) -> Option<(UnaryOp, Expr)> {} + comptime fn as_unary_op(self) -> Option<(UnaryOp, Expr)> {} // docs:end:as_unary_op #[builtin(expr_as_unsafe)] // docs:start:as_unsafe - fn as_unsafe(self) -> Option<[Expr]> {} + comptime fn as_unsafe(self) -> Option<[Expr]> {} // docs:end:as_unsafe #[builtin(expr_has_semicolon)] // docs:start:has_semicolon - fn has_semicolon(self) -> bool {} + comptime fn has_semicolon(self) -> bool {} // docs:end:has_semicolon #[builtin(expr_is_break)] // docs:start:is_break - fn is_break(self) -> bool {} + comptime fn is_break(self) -> bool {} // docs:end:is_break #[builtin(expr_is_continue)] // docs:start:is_continue - fn is_continue(self) -> bool {} + comptime fn is_continue(self) -> bool {} // docs:end:is_continue // docs:start:modify - fn modify(self, f: fn[Env](Expr) -> Option) -> Expr { + comptime fn modify(self, f: fn[Env](Expr) -> Option) -> Expr { // docs:end:modify let result = modify_array(self, f); let result = result.or_else(|| modify_assert(self, f)); @@ -166,11 +166,11 @@ impl Expr { #[builtin(expr_resolve)] // docs:start:resolve - fn resolve(self, in_function: Option) -> TypedExpr {} + comptime fn resolve(self, in_function: Option) -> TypedExpr {} // docs:end:resolve } -fn modify_array(expr: Expr, f: fn[Env](Expr) -> Option) -> Option { +comptime fn modify_array(expr: Expr, f: fn[Env](Expr) -> Option) -> Option { expr.as_array().map( |exprs: [Expr]| { let exprs = modify_expressions(exprs, f); @@ -179,7 +179,7 @@ fn modify_array(expr: Expr, f: fn[Env](Expr) -> Option) -> Option(expr: Expr, f: fn[Env](Expr) -> Option) -> Option { +comptime fn modify_assert(expr: Expr, f: fn[Env](Expr) -> Option) -> Option { expr.as_assert().map( |expr: (Expr, Option)| { let (predicate, msg) = expr; @@ -190,7 +190,7 @@ fn modify_assert(expr: Expr, f: fn[Env](Expr) -> Option) -> Option(expr: Expr, f: fn[Env](Expr) -> Option) -> Option { +comptime fn modify_assert_eq(expr: Expr, f: fn[Env](Expr) -> Option) -> Option { expr.as_assert_eq().map( |expr: (Expr, Expr, Option)| { let (lhs, rhs, msg) = expr; @@ -202,7 +202,7 @@ fn modify_assert_eq(expr: Expr, f: fn[Env](Expr) -> Option) -> Option ) } -fn modify_assign(expr: Expr, f: fn[Env](Expr) -> Option) -> Option { +comptime fn modify_assign(expr: Expr, f: fn[Env](Expr) -> Option) -> Option { expr.as_assign().map( |expr: (Expr, Expr)| { let (lhs, rhs) = expr; @@ -213,7 +213,7 @@ fn modify_assign(expr: Expr, f: fn[Env](Expr) -> Option) -> Option(expr: Expr, f: fn[Env](Expr) -> Option) -> Option { +comptime fn modify_binary_op(expr: Expr, f: fn[Env](Expr) -> Option) -> Option { expr.as_binary_op().map( |expr: (Expr, BinaryOp, Expr)| { let (lhs, op, rhs) = expr; @@ -224,7 +224,7 @@ fn modify_binary_op(expr: Expr, f: fn[Env](Expr) -> Option) -> Option ) } -fn modify_block(expr: Expr, f: fn[Env](Expr) -> Option) -> Option { +comptime fn modify_block(expr: Expr, f: fn[Env](Expr) -> Option) -> Option { expr.as_block().map( |exprs: [Expr]| { let exprs = modify_expressions(exprs, f); @@ -233,7 +233,7 @@ fn modify_block(expr: Expr, f: fn[Env](Expr) -> Option) -> Option(expr: Expr, f: fn[Env](Expr) -> Option) -> Option { +comptime fn modify_cast(expr: Expr, f: fn[Env](Expr) -> Option) -> Option { expr.as_cast().map( |expr: (Expr, UnresolvedType)| { let (expr, typ) = expr; @@ -243,7 +243,7 @@ fn modify_cast(expr: Expr, f: fn[Env](Expr) -> Option) -> Option(expr: Expr, f: fn[Env](Expr) -> Option) -> Option { +comptime fn modify_comptime(expr: Expr, f: fn[Env](Expr) -> Option) -> Option { expr.as_comptime().map( |exprs: [Expr]| { let exprs = exprs.map(|expr: Expr| expr.modify(f)); @@ -252,7 +252,7 @@ fn modify_comptime(expr: Expr, f: fn[Env](Expr) -> Option) -> Option< ) } -fn modify_function_call(expr: Expr, f: fn[Env](Expr) -> Option) -> Option { +comptime fn modify_function_call(expr: Expr, f: fn[Env](Expr) -> Option) -> Option { expr.as_function_call().map( |expr: (Expr, [Expr])| { let (function, arguments) = expr; @@ -263,7 +263,7 @@ fn modify_function_call(expr: Expr, f: fn[Env](Expr) -> Option) -> Op ) } -fn modify_if(expr: Expr, f: fn[Env](Expr) -> Option) -> Option { +comptime fn modify_if(expr: Expr, f: fn[Env](Expr) -> Option) -> Option { expr.as_if().map( |expr: (Expr, Expr, Option)| { let (condition, consequence, alternative) = expr; @@ -275,7 +275,7 @@ fn modify_if(expr: Expr, f: fn[Env](Expr) -> Option) -> Option ) } -fn modify_index(expr: Expr, f: fn[Env](Expr) -> Option) -> Option { +comptime fn modify_index(expr: Expr, f: fn[Env](Expr) -> Option) -> Option { expr.as_index().map( |expr: (Expr, Expr)| { let (object, index) = expr; @@ -286,7 +286,7 @@ fn modify_index(expr: Expr, f: fn[Env](Expr) -> Option) -> Option(expr: Expr, f: fn[Env](Expr) -> Option) -> Option { +comptime fn modify_let(expr: Expr, f: fn[Env](Expr) -> Option) -> Option { expr.as_let().map( |expr: (Expr, Option, Expr)| { let (pattern, typ, expr) = expr; @@ -297,7 +297,7 @@ fn modify_let(expr: Expr, f: fn[Env](Expr) -> Option) -> Option ) } -fn modify_member_access(expr: Expr, f: fn[Env](Expr) -> Option) -> Option { +comptime fn modify_member_access(expr: Expr, f: fn[Env](Expr) -> Option) -> Option { expr.as_member_access().map( |expr: (Expr, Quoted)| { let (object, name) = expr; @@ -307,7 +307,7 @@ fn modify_member_access(expr: Expr, f: fn[Env](Expr) -> Option) -> Op ) } -fn modify_method_call(expr: Expr, f: fn[Env](Expr) -> Option) -> Option { +comptime fn modify_method_call(expr: Expr, f: fn[Env](Expr) -> Option) -> Option { expr.as_method_call().map( |expr: (Expr, Quoted, [UnresolvedType], [Expr])| { let (object, name, generics, arguments) = expr; @@ -318,7 +318,7 @@ fn modify_method_call(expr: Expr, f: fn[Env](Expr) -> Option) -> Opti ) } -fn modify_repeated_element_array(expr: Expr, f: fn[Env](Expr) -> Option) -> Option { +comptime fn modify_repeated_element_array(expr: Expr, f: fn[Env](Expr) -> Option) -> Option { expr.as_repeated_element_array().map( |expr: (Expr, Expr)| { let (expr, length) = expr; @@ -329,7 +329,7 @@ fn modify_repeated_element_array(expr: Expr, f: fn[Env](Expr) -> Option(expr: Expr, f: fn[Env](Expr) -> Option) -> Option { +comptime fn modify_repeated_element_slice(expr: Expr, f: fn[Env](Expr) -> Option) -> Option { expr.as_repeated_element_slice().map( |expr: (Expr, Expr)| { let (expr, length) = expr; @@ -340,7 +340,7 @@ fn modify_repeated_element_slice(expr: Expr, f: fn[Env](Expr) -> Option(expr: Expr, f: fn[Env](Expr) -> Option) -> Option { +comptime fn modify_slice(expr: Expr, f: fn[Env](Expr) -> Option) -> Option { expr.as_slice().map( |exprs: [Expr]| { let exprs = modify_expressions(exprs, f); @@ -349,7 +349,7 @@ fn modify_slice(expr: Expr, f: fn[Env](Expr) -> Option) -> Option(expr: Expr, f: fn[Env](Expr) -> Option) -> Option { +comptime fn modify_tuple(expr: Expr, f: fn[Env](Expr) -> Option) -> Option { expr.as_tuple().map( |exprs: [Expr]| { let exprs = modify_expressions(exprs, f); @@ -358,7 +358,7 @@ fn modify_tuple(expr: Expr, f: fn[Env](Expr) -> Option) -> Option(expr: Expr, f: fn[Env](Expr) -> Option) -> Option { +comptime fn modify_unary_op(expr: Expr, f: fn[Env](Expr) -> Option) -> Option { expr.as_unary_op().map( |expr: (UnaryOp, Expr)| { let (op, rhs) = expr; @@ -368,7 +368,7 @@ fn modify_unary_op(expr: Expr, f: fn[Env](Expr) -> Option) -> Option< ) } -fn modify_unsafe(expr: Expr, f: fn[Env](Expr) -> Option) -> Option { +comptime fn modify_unsafe(expr: Expr, f: fn[Env](Expr) -> Option) -> Option { expr.as_unsafe().map( |exprs: [Expr]| { let exprs = exprs.map(|expr: Expr| expr.modify(f)); @@ -377,7 +377,7 @@ fn modify_unsafe(expr: Expr, f: fn[Env](Expr) -> Option) -> Option(exprs: [Expr], f: fn[Env](Expr) -> Option) -> [Expr] { +comptime fn modify_expressions(exprs: [Expr], f: fn[Env](Expr) -> Option) -> [Expr] { exprs.map(|expr: Expr| expr.modify(f)) } @@ -498,7 +498,7 @@ comptime fn new_unsafe(exprs: [Expr]) -> Expr { quote { unsafe { $exprs }}.as_expr().unwrap() } -fn join_expressions(exprs: [Expr], separator: Quoted) -> Quoted { +comptime fn join_expressions(exprs: [Expr], separator: Quoted) -> Quoted { exprs.map(|expr: Expr| expr.quoted()).join(separator) } diff --git a/noir/noir-repo/noir_stdlib/src/meta/format_string.nr b/noir/noir-repo/noir_stdlib/src/meta/format_string.nr index 44b69719efe..075a69fdafb 100644 --- a/noir/noir-repo/noir_stdlib/src/meta/format_string.nr +++ b/noir/noir-repo/noir_stdlib/src/meta/format_string.nr @@ -1,6 +1,6 @@ impl fmtstr { #[builtin(fmtstr_quoted_contents)] // docs:start:quoted_contents - fn quoted_contents(self) -> Quoted {} + comptime fn quoted_contents(self) -> Quoted {} // docs:end:quoted_contents } diff --git a/noir/noir-repo/noir_stdlib/src/meta/function_def.nr b/noir/noir-repo/noir_stdlib/src/meta/function_def.nr index 5ce3dbeabff..5d536d60ae1 100644 --- a/noir/noir-repo/noir_stdlib/src/meta/function_def.nr +++ b/noir/noir-repo/noir_stdlib/src/meta/function_def.nr @@ -1,66 +1,66 @@ impl FunctionDefinition { #[builtin(function_def_add_attribute)] // docs:start:add_attribute - fn add_attribute(self, attribute: str) {} + comptime fn add_attribute(self, attribute: str) {} // docs:end:add_attribute #[builtin(function_def_body)] // docs:start:body - fn body(self) -> Expr {} + comptime fn body(self) -> Expr {} // docs:end:body #[builtin(function_def_has_named_attribute)] // docs:start:has_named_attribute - fn has_named_attribute(self, name: Quoted) -> bool {} + comptime fn has_named_attribute(self, name: Quoted) -> bool {} // docs:end:has_named_attribute #[builtin(function_def_is_unconstrained)] // docs:start:is_unconstrained - fn is_unconstrained(self) -> bool {} + comptime fn is_unconstrained(self) -> bool {} // docs:end:is_unconstrained #[builtin(function_def_module)] // docs:start:module - fn module(self) -> Module {} + comptime fn module(self) -> Module {} // docs:end:module #[builtin(function_def_name)] // docs:start:name - fn name(self) -> Quoted {} + comptime fn name(self) -> Quoted {} // docs:end:name #[builtin(function_def_parameters)] // docs:start:parameters - fn parameters(self) -> [(Quoted, Type)] {} + comptime fn parameters(self) -> [(Quoted, Type)] {} // docs:end:parameters #[builtin(function_def_return_type)] // docs:start:return_type - fn return_type(self) -> Type {} + comptime fn return_type(self) -> Type {} // docs:end:return_type #[builtin(function_def_set_body)] // docs:start:set_body - fn set_body(self, body: Expr) {} + comptime fn set_body(self, body: Expr) {} // docs:end:set_body #[builtin(function_def_set_parameters)] // docs:start:set_parameters - fn set_parameters(self, parameters: [(Quoted, Type)]) {} + comptime fn set_parameters(self, parameters: [(Quoted, Type)]) {} // docs:end:set_parameters #[builtin(function_def_set_return_type)] // docs:start:set_return_type - fn set_return_type(self, return_type: Type) {} + comptime fn set_return_type(self, return_type: Type) {} // docs:end:set_return_type #[builtin(function_def_set_return_public)] // docs:start:set_return_public - fn set_return_public(self, public: bool) {} + comptime fn set_return_public(self, public: bool) {} // docs:end:set_return_public #[builtin(function_def_set_unconstrained)] // docs:start:set_unconstrained - fn set_unconstrained(self, value: bool) {} + comptime fn set_unconstrained(self, value: bool) {} // docs:end:set_unconstrained } diff --git a/noir/noir-repo/noir_stdlib/src/meta/module.nr b/noir/noir-repo/noir_stdlib/src/meta/module.nr index bee6612e1bf..6f936bf4c57 100644 --- a/noir/noir-repo/noir_stdlib/src/meta/module.nr +++ b/noir/noir-repo/noir_stdlib/src/meta/module.nr @@ -1,26 +1,26 @@ impl Module { #[builtin(module_add_item)] // docs:start:add_item - fn add_item(self, item: Quoted) {} + comptime fn add_item(self, item: Quoted) {} // docs:end:add_item #[builtin(module_has_named_attribute)] // docs:start:has_named_attribute - fn has_named_attribute(self, name: Quoted) -> bool {} + comptime fn has_named_attribute(self, name: Quoted) -> bool {} // docs:end:has_named_attribute #[builtin(module_is_contract)] // docs:start:is_contract - fn is_contract(self) -> bool {} + comptime fn is_contract(self) -> bool {} // docs:end:is_contract #[builtin(module_functions)] // docs:start:functions - fn functions(self) -> [FunctionDefinition] {} + comptime fn functions(self) -> [FunctionDefinition] {} // docs:end:functions #[builtin(module_name)] // docs:start:name - fn name(self) -> Quoted {} + comptime fn name(self) -> Quoted {} // docs:end:name } diff --git a/noir/noir-repo/noir_stdlib/src/meta/quoted.nr b/noir/noir-repo/noir_stdlib/src/meta/quoted.nr index 3fab43359c1..ff74580ce20 100644 --- a/noir/noir-repo/noir_stdlib/src/meta/quoted.nr +++ b/noir/noir-repo/noir_stdlib/src/meta/quoted.nr @@ -4,36 +4,36 @@ use crate::option::Option; impl Quoted { #[builtin(quoted_as_expr)] // docs:start:as_expr - fn as_expr(self) -> Option {} + comptime fn as_expr(self) -> Option {} // docs:end:as_expr #[builtin(quoted_as_module)] // docs:start:as_module - fn as_module(self) -> Option {} + comptime fn as_module(self) -> Option {} // docs:end:as_module #[builtin(quoted_as_trait_constraint)] // docs:start:as_trait_constraint - fn as_trait_constraint(self) -> TraitConstraint {} + comptime fn as_trait_constraint(self) -> TraitConstraint {} // docs:end:as_trait_constraint #[builtin(quoted_as_type)] // docs:start:as_type - fn as_type(self) -> Type {} + comptime fn as_type(self) -> Type {} // docs:end:as_type #[builtin(quoted_tokens)] // docs:start:tokens - fn tokens(self) -> [Quoted] {} + comptime fn tokens(self) -> [Quoted] {} // docs:end:tokens } impl Eq for Quoted { - fn eq(self, other: Quoted) -> bool { + comptime fn eq(self, other: Quoted) -> bool { quoted_eq(self, other) } } #[builtin(quoted_eq)] -fn quoted_eq(_first: Quoted, _second: Quoted) -> bool {} +comptime fn quoted_eq(_first: Quoted, _second: Quoted) -> bool {} diff --git a/noir/noir-repo/noir_stdlib/src/meta/struct_def.nr b/noir/noir-repo/noir_stdlib/src/meta/struct_def.nr index 5db720b91d3..e3621b3482e 100644 --- a/noir/noir-repo/noir_stdlib/src/meta/struct_def.nr +++ b/noir/noir-repo/noir_stdlib/src/meta/struct_def.nr @@ -1,47 +1,47 @@ impl StructDefinition { #[builtin(struct_def_add_attribute)] // docs:start:add_attribute - fn add_attribute(self, attribute: str) {} + comptime fn add_attribute(self, attribute: str) {} // docs:end:add_attribute #[builtin(struct_def_add_generic)] // docs:start:add_generic - fn add_generic(self, generic_name: str) -> Type {} + comptime fn add_generic(self, generic_name: str) -> Type {} // docs:end:add_generic /// Return a syntactic version of this struct definition as a type. /// For example, `as_type(quote { type Foo { ... } })` would return `Foo` #[builtin(struct_def_as_type)] // docs:start:as_type - fn as_type(self) -> Type {} + comptime fn as_type(self) -> Type {} // docs:end:as_type #[builtin(struct_def_has_named_attribute)] // docs:start:has_named_attribute - fn has_named_attribute(self, name: Quoted) -> bool {} + comptime fn has_named_attribute(self, name: Quoted) -> bool {} // docs:end:has_named_attribute /// Return each generic on this struct. #[builtin(struct_def_generics)] // docs:start:generics - fn generics(self) -> [Type] {} + comptime fn generics(self) -> [Type] {} // docs:end:generics /// Returns (name, type) pairs of each field in this struct. Each type is as-is /// with any generic arguments unchanged. #[builtin(struct_def_fields)] // docs:start:fields - fn fields(self) -> [(Quoted, Type)] {} + comptime fn fields(self) -> [(Quoted, Type)] {} // docs:end:fields #[builtin(struct_def_module)] // docs:start:module - fn module(self) -> Module {} + comptime fn module(self) -> Module {} // docs:end:module #[builtin(struct_def_name)] // docs:start:name - fn name(self) -> Quoted {} + comptime fn name(self) -> Quoted {} // docs:end:name /// Sets the fields of this struct to the given fields list. @@ -50,6 +50,6 @@ impl StructDefinition { /// Each name is expected to be a single identifier. #[builtin(struct_def_set_fields)] // docs:start:set_fields - fn set_fields(self, new_fields: [(Quoted, Type)]) {} + comptime fn set_fields(self, new_fields: [(Quoted, Type)]) {} // docs:end:set_fields } diff --git a/noir/noir-repo/noir_stdlib/src/meta/trait_constraint.nr b/noir/noir-repo/noir_stdlib/src/meta/trait_constraint.nr index f0276608974..b90f0b590d5 100644 --- a/noir/noir-repo/noir_stdlib/src/meta/trait_constraint.nr +++ b/noir/noir-repo/noir_stdlib/src/meta/trait_constraint.nr @@ -2,19 +2,19 @@ use crate::hash::{Hash, Hasher}; use crate::cmp::Eq; impl Eq for TraitConstraint { - fn eq(self, other: Self) -> bool { + comptime fn eq(self, other: Self) -> bool { constraint_eq(self, other) } } impl Hash for TraitConstraint { - fn hash(self, state: &mut H) where H: Hasher { + comptime fn hash(self, state: &mut H) where H: Hasher { state.write(constraint_hash(self)); } } #[builtin(trait_constraint_eq)] -fn constraint_eq(_first: TraitConstraint, _second: TraitConstraint) -> bool {} +comptime fn constraint_eq(_first: TraitConstraint, _second: TraitConstraint) -> bool {} #[builtin(trait_constraint_hash)] -fn constraint_hash(_constraint: TraitConstraint) -> Field {} +comptime fn constraint_hash(_constraint: TraitConstraint) -> Field {} diff --git a/noir/noir-repo/noir_stdlib/src/meta/trait_def.nr b/noir/noir-repo/noir_stdlib/src/meta/trait_def.nr index c26b571240b..51676efbc34 100644 --- a/noir/noir-repo/noir_stdlib/src/meta/trait_def.nr +++ b/noir/noir-repo/noir_stdlib/src/meta/trait_def.nr @@ -4,24 +4,24 @@ use crate::cmp::Eq; impl TraitDefinition { #[builtin(trait_def_as_trait_constraint)] // docs:start:as_trait_constraint - fn as_trait_constraint(_self: Self) -> TraitConstraint {} + comptime fn as_trait_constraint(_self: Self) -> TraitConstraint {} // docs:end:as_trait_constraint } impl Eq for TraitDefinition { - fn eq(self, other: Self) -> bool { + comptime fn eq(self, other: Self) -> bool { trait_def_eq(self, other) } } impl Hash for TraitDefinition { - fn hash(self, state: &mut H) where H: Hasher { + comptime fn hash(self, state: &mut H) where H: Hasher { state.write(trait_def_hash(self)); } } #[builtin(trait_def_eq)] -fn trait_def_eq(_first: TraitDefinition, _second: TraitDefinition) -> bool {} +comptime fn trait_def_eq(_first: TraitDefinition, _second: TraitDefinition) -> bool {} #[builtin(trait_def_hash)] -fn trait_def_hash(_def: TraitDefinition) -> Field {} +comptime fn trait_def_hash(_def: TraitDefinition) -> Field {} diff --git a/noir/noir-repo/noir_stdlib/src/meta/trait_impl.nr b/noir/noir-repo/noir_stdlib/src/meta/trait_impl.nr index 15b02eac6bd..6755a5c2031 100644 --- a/noir/noir-repo/noir_stdlib/src/meta/trait_impl.nr +++ b/noir/noir-repo/noir_stdlib/src/meta/trait_impl.nr @@ -1,11 +1,11 @@ impl TraitImpl { #[builtin(trait_impl_trait_generic_args)] // docs:start:trait_generic_args - fn trait_generic_args(self) -> [Type] {} + comptime fn trait_generic_args(self) -> [Type] {} // docs:end:trait_generic_args #[builtin(trait_impl_methods)] // docs:start:methods - fn methods(self) -> [FunctionDefinition] {} + comptime fn methods(self) -> [FunctionDefinition] {} // docs:end:methods } diff --git a/noir/noir-repo/noir_stdlib/src/meta/typ.nr b/noir/noir-repo/noir_stdlib/src/meta/typ.nr index 71bd6fd7f1c..d692f6e5a7e 100644 --- a/noir/noir-repo/noir_stdlib/src/meta/typ.nr +++ b/noir/noir-repo/noir_stdlib/src/meta/typ.nr @@ -3,71 +3,71 @@ use crate::option::Option; #[builtin(fresh_type_variable)] // docs:start:fresh_type_variable -pub fn fresh_type_variable() -> Type {} +pub comptime fn fresh_type_variable() -> Type {} // docs:end:fresh_type_variable impl Type { #[builtin(type_as_array)] // docs:start:as_array - fn as_array(self) -> Option<(Type, Type)> {} + comptime fn as_array(self) -> Option<(Type, Type)> {} // docs:end:as_array #[builtin(type_as_constant)] // docs:start:as_constant - fn as_constant(self) -> Option {} + comptime fn as_constant(self) -> Option {} // docs:end:as_constant #[builtin(type_as_integer)] // docs:start:as_integer - fn as_integer(self) -> Option<(bool, u8)> {} + comptime fn as_integer(self) -> Option<(bool, u8)> {} // docs:end:as_integer #[builtin(type_as_slice)] // docs:start:as_slice - fn as_slice(self) -> Option {} + comptime fn as_slice(self) -> Option {} // docs:end:as_slice #[builtin(type_as_str)] // docs:start:as_str - fn as_str(self) -> Option {} + comptime fn as_str(self) -> Option {} // docs:end:as_str #[builtin(type_as_struct)] // docs:start:as_struct - fn as_struct(self) -> Option<(StructDefinition, [Type])> {} + comptime fn as_struct(self) -> Option<(StructDefinition, [Type])> {} // docs:end:as_struct #[builtin(type_as_tuple)] // docs:start:as_tuple - fn as_tuple(self) -> Option<[Type]> {} + comptime fn as_tuple(self) -> Option<[Type]> {} // docs:end:as_tuple #[builtin(type_get_trait_impl)] // docs:start:get_trait_impl - fn get_trait_impl(self, constraint: TraitConstraint) -> Option {} + comptime fn get_trait_impl(self, constraint: TraitConstraint) -> Option {} // docs:end:get_trait_impl #[builtin(type_implements)] // docs:start:implements - fn implements(self, constraint: TraitConstraint) -> bool {} + comptime fn implements(self, constraint: TraitConstraint) -> bool {} // docs:end:implements #[builtin(type_is_bool)] // docs:start:is_bool - fn is_bool(self) -> bool {} + comptime fn is_bool(self) -> bool {} // docs:end:is_bool #[builtin(type_is_field)] // docs:start:is_field - fn is_field(self) -> bool {} + comptime fn is_field(self) -> bool {} // docs:end:is_field } impl Eq for Type { - fn eq(self, other: Self) -> bool { + comptime fn eq(self, other: Self) -> bool { type_eq(self, other) } } #[builtin(type_eq)] -fn type_eq(_first: Type, _second: Type) -> bool {} +comptime fn type_eq(_first: Type, _second: Type) -> bool {} diff --git a/noir/noir-repo/noir_stdlib/src/meta/typed_expr.nr b/noir/noir-repo/noir_stdlib/src/meta/typed_expr.nr index 8daede97438..1d3b073b6da 100644 --- a/noir/noir-repo/noir_stdlib/src/meta/typed_expr.nr +++ b/noir/noir-repo/noir_stdlib/src/meta/typed_expr.nr @@ -1,8 +1,16 @@ +//! Contains methods on the built-in `TypedExpr` type for resolved and type-checked expressions. use crate::option::Option; impl TypedExpr { + /// If this expression refers to a function definitions, returns it. Otherwise returns `Option::none()`. #[builtin(typed_expr_as_function_definition)] // docs:start:as_function_definition - fn as_function_definition(self) -> Option {} + comptime fn as_function_definition(self) -> Option {} // docs:end:as_function_definition + + /// Returns the type of the expression, if the expression could be resolved without errors. + #[builtin(typed_expr_get_type)] + // docs:start:get_type + comptime fn get_type(self) -> Option {} + // docs:end:get_type } diff --git a/noir/noir-repo/noir_stdlib/src/meta/unresolved_type.nr b/noir/noir-repo/noir_stdlib/src/meta/unresolved_type.nr index 2589174ed64..f53635414cc 100644 --- a/noir/noir-repo/noir_stdlib/src/meta/unresolved_type.nr +++ b/noir/noir-repo/noir_stdlib/src/meta/unresolved_type.nr @@ -1,6 +1,6 @@ impl UnresolvedType { #[builtin(unresolved_type_is_field)] // docs:start:is_field - fn is_field(self) -> bool {} + comptime fn is_field(self) -> bool {} // docs:end:is_field } diff --git a/noir/noir-repo/test_programs/compile_success_empty/comptime_fmt_strings/src/main.nr b/noir/noir-repo/test_programs/compile_success_empty/comptime_fmt_strings/src/main.nr index 0e2d459a00f..ca337c822d8 100644 --- a/noir/noir-repo/test_programs/compile_success_empty/comptime_fmt_strings/src/main.nr +++ b/noir/noir-repo/test_programs/compile_success_empty/comptime_fmt_strings/src/main.nr @@ -17,7 +17,7 @@ fn main() { call!(glue(quote { hello }, quote { world })); } -fn glue(x: Quoted, y: Quoted) -> Quoted { +comptime fn glue(x: Quoted, y: Quoted) -> Quoted { f"{x}_{y}".quoted_contents() } diff --git a/noir/noir-repo/test_programs/compile_success_empty/comptime_function_definition/src/main.nr b/noir/noir-repo/test_programs/compile_success_empty/comptime_function_definition/src/main.nr index 6bfd8ef9699..e01ff4a71f1 100644 --- a/noir/noir-repo/test_programs/compile_success_empty/comptime_function_definition/src/main.nr +++ b/noir/noir-repo/test_programs/compile_success_empty/comptime_function_definition/src/main.nr @@ -66,7 +66,7 @@ contract some_contract { } } -fn set_pub_return(f: FunctionDefinition) { +comptime fn set_pub_return(f: FunctionDefinition) { f.set_return_public(true); } diff --git a/noir/noir-repo/test_programs/compile_success_empty/comptime_typed_expr/Nargo.toml b/noir/noir-repo/test_programs/compile_success_empty/comptime_typed_expr/Nargo.toml new file mode 100644 index 00000000000..151b8b8e9d0 --- /dev/null +++ b/noir/noir-repo/test_programs/compile_success_empty/comptime_typed_expr/Nargo.toml @@ -0,0 +1,7 @@ +[package] +name = "comptime_typed_expr" +type = "bin" +authors = [""] +compiler_version = ">=0.31.0" + +[dependencies] \ No newline at end of file diff --git a/noir/noir-repo/test_programs/compile_success_empty/comptime_typed_expr/src/main.nr b/noir/noir-repo/test_programs/compile_success_empty/comptime_typed_expr/src/main.nr new file mode 100644 index 00000000000..ba2594f1da9 --- /dev/null +++ b/noir/noir-repo/test_programs/compile_success_empty/comptime_typed_expr/src/main.nr @@ -0,0 +1,7 @@ +fn main() { + comptime + { + let expr = quote { [1, 3] }.as_expr().unwrap().resolve(Option::none()); + assert_eq(expr.get_type().unwrap(), quote { [Field; 2] }.as_type()); + } +} diff --git a/noir/noir-repo/tooling/lsp/src/requests/completion/completion_items.rs b/noir/noir-repo/tooling/lsp/src/requests/completion/completion_items.rs index 7e7511cdaa3..163a4e15d00 100644 --- a/noir/noir-repo/tooling/lsp/src/requests/completion/completion_items.rs +++ b/noir/noir-repo/tooling/lsp/src/requests/completion/completion_items.rs @@ -213,11 +213,14 @@ impl<'a> NodeFinder<'a> { let name = if self_prefix { format!("self.{}", name) } else { name.clone() }; let name = &name; let description = func_meta_type_to_string(func_meta, func_self_type.is_some()); + let mut has_arguments = false; let completion_item = match function_completion_kind { - FunctionCompletionKind::Name => { - simple_completion_item(name, CompletionItemKind::FUNCTION, Some(description)) - } + FunctionCompletionKind::Name => simple_completion_item( + name, + CompletionItemKind::FUNCTION, + Some(description.clone()), + ), FunctionCompletionKind::NameAndParameters => { let kind = CompletionItemKind::FUNCTION; let skip_first_argument = attribute_first_type.is_some(); @@ -227,20 +230,25 @@ impl<'a> NodeFinder<'a> { function_kind, skip_first_argument, ); - let (label, insert_text) = if insert_text.ends_with("()") { - if skip_first_argument { - (name.to_string(), insert_text.strip_suffix("()").unwrap().to_string()) - } else { - (format!("{}()", name), insert_text) - } - } else { - (format!("{}(…)", name), insert_text) - }; - snippet_completion_item(label, kind, insert_text, Some(description)) + if insert_text.ends_with("()") { + let label = + if skip_first_argument { name.to_string() } else { format!("{}()", name) }; + simple_completion_item(label, kind, Some(description.clone())) + } else { + has_arguments = true; + snippet_completion_item( + format!("{}(…)", name), + kind, + insert_text, + Some(description.clone()), + ) + } } }; + let completion_item = completion_item_with_detail(completion_item, description); + let completion_item = if is_operator { completion_item_with_sort_text(completion_item, operator_sort_text()) } else if function_kind == FunctionKind::Any && name == "new" { @@ -254,11 +262,16 @@ impl<'a> NodeFinder<'a> { let completion_item = match function_completion_kind { FunctionCompletionKind::Name => completion_item, FunctionCompletionKind::NameAndParameters => { - completion_item_with_trigger_parameter_hints_command(completion_item) + if has_arguments { + completion_item_with_trigger_parameter_hints_command(completion_item) + } else { + completion_item + } } }; let completion_item = self.completion_item_with_doc_comments(ReferenceId::Function(func_id), completion_item); + Some(completion_item) } @@ -494,3 +507,10 @@ pub(super) fn completion_item_with_trigger_parameter_hints_command( ..completion_item } } + +pub(super) fn completion_item_with_detail( + completion_item: CompletionItem, + detail: String, +) -> CompletionItem { + CompletionItem { detail: Some(detail), ..completion_item } +} diff --git a/noir/noir-repo/tooling/lsp/src/requests/completion/tests.rs b/noir/noir-repo/tooling/lsp/src/requests/completion/tests.rs index 1b04e060e2c..7e11e0d9d52 100644 --- a/noir/noir-repo/tooling/lsp/src/requests/completion/tests.rs +++ b/noir/noir-repo/tooling/lsp/src/requests/completion/tests.rs @@ -5,7 +5,7 @@ mod completion_tests { requests::{ completion::{ completion_items::{ - completion_item_with_sort_text, + completion_item_with_detail, completion_item_with_sort_text, completion_item_with_trigger_parameter_hints_command, module_completion_item, simple_completion_item, snippet_completion_item, }, @@ -107,12 +107,22 @@ mod completion_tests { insert_text: impl Into, description: impl Into, ) -> CompletionItem { - completion_item_with_trigger_parameter_hints_command(snippet_completion_item( - label, - CompletionItemKind::FUNCTION, - insert_text, - Some(description.into()), - )) + let insert_text: String = insert_text.into(); + let description: String = description.into(); + + let has_arguments = insert_text.ends_with(')') && !insert_text.ends_with("()"); + let completion_item = if has_arguments { + completion_item_with_trigger_parameter_hints_command(snippet_completion_item( + label, + CompletionItemKind::FUNCTION, + insert_text, + Some(description.clone()), + )) + } else { + simple_completion_item(label, CompletionItemKind::FUNCTION, Some(description.clone())) + }; + + completion_item_with_detail(completion_item, description) } fn field_completion_item(field: &str, typ: impl Into) -> CompletionItem { @@ -191,15 +201,8 @@ mod completion_tests { use foo::>|< "#; - assert_completion( - src, - vec![simple_completion_item( - "bar", - CompletionItemKind::FUNCTION, - Some("fn(i32) -> u64".to_string()), - )], - ) - .await; + assert_completion(src, vec![function_completion_item("bar", "bar()", "fn(i32) -> u64")]) + .await; } #[test] @@ -1672,11 +1675,7 @@ mod completion_tests { "#; assert_completion_excluding_auto_import( src, - vec![simple_completion_item( - "hello_world", - CompletionItemKind::FUNCTION, - Some("fn()".to_string()), - )], + vec![function_completion_item("hello_world", "hello_world()", "fn()")], ) .await; } @@ -1694,11 +1693,7 @@ mod completion_tests { "#; assert_completion_excluding_auto_import( src, - vec![simple_completion_item( - "bar", - CompletionItemKind::FUNCTION, - Some("fn()".to_string()), - )], + vec![function_completion_item("bar", "bar()", "fn()")], ) .await; } @@ -1716,11 +1711,7 @@ mod completion_tests { "#; assert_completion_excluding_auto_import( src, - vec![simple_completion_item( - "bar", - CompletionItemKind::FUNCTION, - Some("fn()".to_string()), - )], + vec![function_completion_item("bar", "bar()", "fn()")], ) .await; } @@ -1738,11 +1729,7 @@ mod completion_tests { "#; assert_completion_excluding_auto_import( src, - vec![simple_completion_item( - "bar", - CompletionItemKind::FUNCTION, - Some("fn()".to_string()), - )], + vec![function_completion_item("bar", "bar()", "fn()")], ) .await; } @@ -1762,11 +1749,7 @@ mod completion_tests { "#; assert_completion_excluding_auto_import( src, - vec![simple_completion_item( - "bar", - CompletionItemKind::FUNCTION, - Some("fn(self)".to_string()), - )], + vec![function_completion_item("bar", "bar()", "fn(self)")], ) .await; } @@ -1786,11 +1769,7 @@ mod completion_tests { "#; assert_completion_excluding_auto_import( src, - vec![simple_completion_item( - "bar", - CompletionItemKind::FUNCTION, - Some("fn(self)".to_string()), - )], + vec![function_completion_item("bar", "bar()", "fn(self)")], ) .await; } diff --git a/yarn-project/archiver/src/archiver/archiver.test.ts b/yarn-project/archiver/src/archiver/archiver.test.ts index ee7f955f954..9b9232b0ddb 100644 --- a/yarn-project/archiver/src/archiver/archiver.test.ts +++ b/yarn-project/archiver/src/archiver/archiver.test.ts @@ -1,5 +1,4 @@ import { - type Body, EncryptedL2BlockL2Logs, EncryptedNoteL2BlockL2Logs, L2Block, @@ -9,7 +8,7 @@ import { import { EthAddress } from '@aztec/foundation/eth-address'; import { Fr } from '@aztec/foundation/fields'; import { sleep } from '@aztec/foundation/sleep'; -import { AvailabilityOracleAbi, type InboxAbi, RollupAbi } from '@aztec/l1-artifacts'; +import { type InboxAbi, RollupAbi } from '@aztec/l1-artifacts'; import { type MockProxy, mock } from 'jest-mock-extended'; import { @@ -31,7 +30,6 @@ describe('Archiver', () => { const rollupAddress = EthAddress.ZERO; const inboxAddress = EthAddress.ZERO; const registryAddress = EthAddress.ZERO; - const availabilityOracleAddress = EthAddress.ZERO; const blockNumbers = [1, 2, 3]; let publicClient: MockProxy>; @@ -62,7 +60,6 @@ describe('Archiver', () => { archiver = new Archiver( publicClient, rollupAddress, - availabilityOracleAddress, inboxAddress, registryAddress, archiverStore, @@ -75,14 +72,12 @@ describe('Archiver', () => { const blocks = blockNumbers.map(x => L2Block.random(x, 4, x, x + 1, 2, 2)); blocks.forEach((b, i) => (b.header.globalVariables.timestamp = new Fr(now + 1000 * (i + 1)))); - const publishTxs = blocks.map(block => block.body).map(makePublishTx); const rollupTxs = blocks.map(makeRollupTx); publicClient.getBlockNumber.mockResolvedValueOnce(2500n).mockResolvedValueOnce(2600n).mockResolvedValueOnce(2700n); mockGetLogs({ messageSent: [makeMessageSentEvent(98n, 1n, 0n), makeMessageSentEvent(99n, 1n, 1n)], - txPublished: [makeTxsPublishedEvent(101n, blocks[0].body.getTxsEffectsHash())], L2BlockProposed: [makeL2BlockProposedEvent(101n, 1n)], proofVerified: [makeProofVerifiedEvent(102n, 1n, proverId)], }); @@ -94,17 +89,11 @@ describe('Archiver', () => { makeMessageSentEvent(2505n, 2n, 2n), makeMessageSentEvent(2506n, 3n, 1n), ], - txPublished: [ - makeTxsPublishedEvent(2510n, blocks[1].body.getTxsEffectsHash()), - makeTxsPublishedEvent(2520n, blocks[2].body.getTxsEffectsHash()), - ], L2BlockProposed: [makeL2BlockProposedEvent(2510n, 2n), makeL2BlockProposedEvent(2520n, 3n)], }); - publicClient.getTransaction.mockResolvedValueOnce(publishTxs[0]); publicClient.getTransaction.mockResolvedValueOnce(rollupTxs[0]); - publishTxs.slice(1).forEach(tx => publicClient.getTransaction.mockResolvedValueOnce(tx)); rollupTxs.slice(1).forEach(tx => publicClient.getTransaction.mockResolvedValueOnce(tx)); await archiver.start(false); @@ -183,7 +172,6 @@ describe('Archiver', () => { archiver = new Archiver( publicClient, rollupAddress, - availabilityOracleAddress, inboxAddress, registryAddress, archiverStore, @@ -196,7 +184,6 @@ describe('Archiver', () => { const blocks = blockNumbers.map(x => L2Block.random(x, 4, x, x + 1, 2, 2)); - const publishTxs = blocks.map(block => block.body).map(makePublishTx); const rollupTxs = blocks.map(makeRollupTx); // Here we set the current L1 block number to 102. L1 to L2 messages after this should not be read. @@ -204,16 +191,11 @@ describe('Archiver', () => { mockGetLogs({ messageSent: [makeMessageSentEvent(66n, 1n, 0n), makeMessageSentEvent(68n, 1n, 1n)], - txPublished: [ - makeTxsPublishedEvent(70n, blocks[0].body.getTxsEffectsHash()), - makeTxsPublishedEvent(80n, blocks[1].body.getTxsEffectsHash()), - ], L2BlockProposed: [makeL2BlockProposedEvent(70n, 1n), makeL2BlockProposedEvent(80n, 2n)], }); mockGetLogs({}); - publishTxs.slice(0, numL2BlocksInTest).forEach(tx => publicClient.getTransaction.mockResolvedValueOnce(tx)); rollupTxs.slice(0, numL2BlocksInTest).forEach(tx => publicClient.getTransaction.mockResolvedValueOnce(tx)); await archiver.start(false); @@ -230,13 +212,11 @@ describe('Archiver', () => { // logs should be created in order of how archiver syncs. const mockGetLogs = (logs: { messageSent?: ReturnType[]; - txPublished?: ReturnType[]; L2BlockProposed?: ReturnType[]; proofVerified?: ReturnType[]; }) => { publicClient.getLogs .mockResolvedValueOnce(logs.messageSent ?? []) - .mockResolvedValueOnce(logs.txPublished ?? []) .mockResolvedValueOnce(logs.L2BlockProposed ?? []) .mockResolvedValueOnce(logs.proofVerified ?? []); }; @@ -256,21 +236,6 @@ function makeL2BlockProposedEvent(l1BlockNum: bigint, l2BlockNum: bigint) { } as Log; } -/** - * Makes a fake TxsPublished event for testing purposes. - * @param l1BlockNum - L1 block number. - * @param txsEffectsHash - txsEffectsHash for the body. - * @returns A TxsPublished event log. - */ -function makeTxsPublishedEvent(l1BlockNum: bigint, txsEffectsHash: Buffer) { - return { - blockNumber: l1BlockNum, - args: { - txsEffectsHash: txsEffectsHash.toString('hex'), - }, - } as Log; -} - /** * Makes fake L1ToL2 MessageSent events for testing purposes. * @param l1BlockNum - L1 block number. @@ -306,27 +271,13 @@ function makeProofVerifiedEvent(l1BlockNum: bigint, l2BlockNumber: bigint, prove */ function makeRollupTx(l2Block: L2Block) { const header = toHex(l2Block.header.toBuffer()); + const body = toHex(l2Block.body.toBuffer()); const archive = toHex(l2Block.archive.root.toBuffer()); const blockHash = toHex(l2Block.header.hash().toBuffer()); const input = encodeFunctionData({ abi: RollupAbi, functionName: 'propose', - args: [header, archive, blockHash], - }); - return { input } as Transaction; -} - -/** - * Makes a fake availability oracle tx for testing purposes. - * @param blockBody - The block body posted by the simulated tx. - * @returns A fake tx with calldata that corresponds to calling publish in the Availability Oracle contract. - */ -function makePublishTx(blockBody: Body) { - const body = toHex(blockBody.toBuffer()); - const input = encodeFunctionData({ - abi: AvailabilityOracleAbi, - functionName: 'publish', - args: [body], + args: [header, archive, blockHash, [], [], body], }); return { input } as Transaction; } diff --git a/yarn-project/archiver/src/archiver/archiver.ts b/yarn-project/archiver/src/archiver/archiver.ts index dec28dc5228..5015c0a1e35 100644 --- a/yarn-project/archiver/src/archiver/archiver.ts +++ b/yarn-project/archiver/src/archiver/archiver.ts @@ -2,7 +2,7 @@ import { type FromLogType, type GetUnencryptedLogsResponse, type L1ToL2MessageSource, - L2Block, + type L2Block, type L2BlockL2Logs, type L2BlockSource, type L2LogsSource, @@ -46,16 +46,10 @@ import { type Chain, type HttpTransport, type PublicClient, createPublicClient, import { type ArchiverDataStore } from './archiver_store.js'; import { type ArchiverConfig } from './config.js'; -import { - retrieveBlockBodiesFromAvailabilityOracle, - retrieveBlockMetadataFromRollup, - retrieveL1ToL2Messages, - retrieveL2ProofVerifiedEvents, -} from './data_retrieval.js'; +import { retrieveBlockFromRollup, retrieveL1ToL2Messages, retrieveL2ProofVerifiedEvents } from './data_retrieval.js'; import { getL1BlockTime } from './eth_log_handlers.js'; import { ArchiverInstrumentation } from './instrumentation.js'; import { type SingletonDataRetrieval } from './structs/data_retrieval.js'; -import { type L1Published } from './structs/published.js'; /** * Helper interface to combine all sources this archiver implementation provides. @@ -86,7 +80,6 @@ export class Archiver implements ArchiveSource { constructor( private readonly publicClient: PublicClient, private readonly rollupAddress: EthAddress, - private readonly availabilityOracleAddress: EthAddress, private readonly inboxAddress: EthAddress, private readonly registryAddress: EthAddress, private readonly store: ArchiverDataStore, @@ -119,7 +112,6 @@ export class Archiver implements ArchiveSource { const archiver = new Archiver( publicClient, config.l1Contracts.rollupAddress, - config.l1Contracts.availabilityOracleAddress, config.l1Contracts.inboxAddress, config.l1Contracts.registryAddress, archiverStore, @@ -247,72 +239,26 @@ export class Archiver implements ArchiveSource { // Read all data from chain and then write to our stores at the end const nextExpectedL2BlockNum = BigInt((await this.store.getSynchedL2BlockNumber()) + 1); - this.log.debug(`Retrieving block bodies from ${blockBodiesSynchedTo + 1n} to ${currentL1BlockNumber}`); - const retrievedBlockBodies = await retrieveBlockBodiesFromAvailabilityOracle( + this.log.debug(`Retrieving blocks from ${blocksSynchedTo + 1n} to ${currentL1BlockNumber}`); + const retrievedBlocks = await retrieveBlockFromRollup( this.publicClient, - this.availabilityOracleAddress, + this.rollupAddress, blockUntilSynced, - blockBodiesSynchedTo + 1n, + blocksSynchedTo + 1n, currentL1BlockNumber, + nextExpectedL2BlockNum, ); - this.log.debug( - `Retrieved ${retrievedBlockBodies.retrievedData.length} block bodies up to L1 block ${retrievedBlockBodies.lastProcessedL1BlockNumber}`, - ); - await this.store.addBlockBodies(retrievedBlockBodies); - - // Now that we have block bodies we will retrieve block metadata and build L2 blocks from the bodies and the metadata - let retrievedBlocks: L1Published[]; - let lastProcessedL1BlockNumber: bigint; - { - // @todo @LHerskind Investigate how necessary that nextExpectedL2BlockNum really is. - // Also, I would expect it to break horribly if we have a reorg. - this.log.debug(`Retrieving block metadata from ${blocksSynchedTo + 1n} to ${currentL1BlockNumber}`); - const retrievedBlockMetadata = await retrieveBlockMetadataFromRollup( - this.publicClient, - this.rollupAddress, - blockUntilSynced, - blocksSynchedTo + 1n, - currentL1BlockNumber, - nextExpectedL2BlockNum, - ); - - const retrievedBodyHashes = retrievedBlockMetadata.map(([header]) => header.contentCommitment.txsEffectsHash); - - // @note @LHerskind We will occasionally be hitting this point BEFORE, we have actually retrieved the bodies. - // The main reason this have not been an issue earlier is because: - // i) the design previously published the body in one tx and the header in another, - // which in an anvil auto mine world mean that they are separate blocks. - // ii) We have been lucky that latency have been small enough to not matter. - const blockBodiesFromStore = await this.store.getBlockBodies(retrievedBodyHashes); + // Add the body - if (retrievedBlockMetadata.length !== blockBodiesFromStore.length) { - this.log.warn('Block headers length does not equal block bodies length'); - } - - const blocks: L1Published[] = []; - for (let i = 0; i < retrievedBlockMetadata.length; i++) { - const [header, archive, l1] = retrievedBlockMetadata[i]; - const blockBody = blockBodiesFromStore[i]; - if (blockBody) { - blocks.push({ data: new L2Block(archive, header, blockBody), l1 }); - } else { - this.log.warn(`Block body not found for block ${header.globalVariables.blockNumber.toBigInt()}.`); - } - } - - (blocks.length ? this.log.verbose : this.log.debug)( - `Retrieved ${blocks.length || 'no'} new L2 blocks between L1 blocks ${ - blocksSynchedTo + 1n - } and ${currentL1BlockNumber}.`, - ); + (retrievedBlocks.length ? this.log.verbose : this.log.debug)( + `Retrieved ${retrievedBlocks.length || 'no'} new L2 blocks between L1 blocks ${ + blocksSynchedTo + 1n + } and ${currentL1BlockNumber}.`, + ); - retrievedBlocks = blocks; - lastProcessedL1BlockNumber = - retrievedBlockMetadata.length > 0 - ? retrievedBlockMetadata[retrievedBlockMetadata.length - 1][2].blockNumber - : blocksSynchedTo; - } + const lastProcessedL1BlockNumber = + retrievedBlocks.length > 0 ? retrievedBlocks[retrievedBlocks.length - 1].l1.blockNumber : blocksSynchedTo; this.log.debug( `Processing retrieved blocks ${retrievedBlocks @@ -320,29 +266,33 @@ export class Archiver implements ArchiveSource { .join(',')} with last processed L1 block ${lastProcessedL1BlockNumber}`, ); - await Promise.all( - retrievedBlocks.map(block => { - const noteEncryptedLogs = block.data.body.noteEncryptedLogs; - const encryptedLogs = block.data.body.encryptedLogs; - const unencryptedLogs = block.data.body.unencryptedLogs; - return this.store.addLogs(noteEncryptedLogs, encryptedLogs, unencryptedLogs, block.data.number); - }), - ); + if (retrievedBlocks.length > 0) { + await Promise.all( + retrievedBlocks.map(block => { + const noteEncryptedLogs = block.data.body.noteEncryptedLogs; + const encryptedLogs = block.data.body.encryptedLogs; + const unencryptedLogs = block.data.body.unencryptedLogs; + return this.store.addLogs(noteEncryptedLogs, encryptedLogs, unencryptedLogs, block.data.number); + }), + ); - // Unroll all logs emitted during the retrieved blocks and extract any contract classes and instances from them - await Promise.all( - retrievedBlocks.map(async block => { - const blockLogs = block.data.body.txEffects - .flatMap(txEffect => (txEffect ? [txEffect.unencryptedLogs] : [])) - .flatMap(txLog => txLog.unrollLogs()); - await this.storeRegisteredContractClasses(blockLogs, block.data.number); - await this.storeDeployedContractInstances(blockLogs, block.data.number); - await this.storeBroadcastedIndividualFunctions(blockLogs, block.data.number); - }), - ); + // Unroll all logs emitted during the retrieved blocks and extract any contract classes and instances from them + await Promise.all( + retrievedBlocks.map(async block => { + const blockLogs = block.data.body.txEffects + .flatMap(txEffect => (txEffect ? [txEffect.unencryptedLogs] : [])) + .flatMap(txLog => txLog.unrollLogs()); + await this.storeRegisteredContractClasses(blockLogs, block.data.number); + await this.storeDeployedContractInstances(blockLogs, block.data.number); + await this.storeBroadcastedIndividualFunctions(blockLogs, block.data.number); + }), + ); - if (retrievedBlocks.length > 0) { const timer = new Timer(); + await this.store.addBlockBodies({ + lastProcessedL1BlockNumber: lastProcessedL1BlockNumber, + retrievedData: retrievedBlocks.map(b => b.data.body), + }); await this.store.addBlocks(retrievedBlocks); this.instrumentation.processNewBlocks( timer.ms() / retrievedBlocks.length, diff --git a/yarn-project/archiver/src/archiver/data_retrieval.ts b/yarn-project/archiver/src/archiver/data_retrieval.ts index 81aca60a783..5827296d3f1 100644 --- a/yarn-project/archiver/src/archiver/data_retrieval.ts +++ b/yarn-project/archiver/src/archiver/data_retrieval.ts @@ -1,5 +1,5 @@ -import { type Body, type InboxLeaf } from '@aztec/circuit-types'; -import { type AppendOnlyTreeSnapshot, Fr, type Header, type Proof } from '@aztec/circuits.js'; +import { type InboxLeaf, type L2Block } from '@aztec/circuit-types'; +import { Fr, type Proof } from '@aztec/circuits.js'; import { type EthAddress } from '@aztec/foundation/eth-address'; import { type DebugLogger, createDebugLogger } from '@aztec/foundation/log'; import { RollupAbi } from '@aztec/l1-artifacts'; @@ -10,25 +10,23 @@ import { getBlockProofFromSubmitProofTx, getL2BlockProposedLogs, getMessageSentLogs, - getTxsPublishedLogs, processL2BlockProposedLogs, processMessageSentLogs, - processTxsPublishedLogs, } from './eth_log_handlers.js'; import { type DataRetrieval } from './structs/data_retrieval.js'; -import { type L1PublishedData } from './structs/published.js'; +import { type L1Published } from './structs/published.js'; /** - * Fetches new L2 block metadata (header, archive snapshot). + * Fetches new L2 blocks. * @param publicClient - The viem public client to use for transaction retrieval. * @param rollupAddress - The address of the rollup contract. * @param blockUntilSynced - If true, blocks until the archiver has fully synced. * @param searchStartBlock - The block number to use for starting the search. * @param searchEndBlock - The highest block number that we should search up to. * @param expectedNextL2BlockNum - The next L2 block number that we expect to find. - * @returns An array of tuples representing block metadata including the header, archive tree snapshot; as well as the next eth block to search from. + * @returns An array of block; as well as the next eth block to search from. */ -export async function retrieveBlockMetadataFromRollup( +export async function retrieveBlockFromRollup( publicClient: PublicClient, rollupAddress: EthAddress, blockUntilSynced: boolean, @@ -36,8 +34,8 @@ export async function retrieveBlockMetadataFromRollup( searchEndBlock: bigint, expectedNextL2BlockNum: bigint, logger: DebugLogger = createDebugLogger('aztec:archiver'), -): Promise<[Header, AppendOnlyTreeSnapshot, L1PublishedData][]> { - const retrievedBlockMetadata: [Header, AppendOnlyTreeSnapshot, L1PublishedData][] = []; +): Promise[]> { + const retrievedBlocks: L1Published[] = []; do { if (searchStartBlock > searchEndBlock) { break; @@ -57,56 +55,12 @@ export async function retrieveBlockMetadataFromRollup( `Got L2 block processed logs for ${L2BlockProposedLogs[0].blockNumber}-${lastLog.blockNumber} between ${searchStartBlock}-${searchEndBlock} L1 blocks`, ); - const newBlockMetadata = await processL2BlockProposedLogs( - publicClient, - expectedNextL2BlockNum, - L2BlockProposedLogs, - ); - retrievedBlockMetadata.push(...newBlockMetadata); + const newBlocks = await processL2BlockProposedLogs(publicClient, expectedNextL2BlockNum, L2BlockProposedLogs); + retrievedBlocks.push(...newBlocks); searchStartBlock = lastLog.blockNumber! + 1n; - expectedNextL2BlockNum += BigInt(newBlockMetadata.length); - } while (blockUntilSynced && searchStartBlock <= searchEndBlock); - return retrievedBlockMetadata; -} - -/** - * Fetches new L2 block bodies and their hashes. - * @param publicClient - The viem public client to use for transaction retrieval. - * @param availabilityOracleAddress - The address of the availability oracle contract. - * @param blockUntilSynced - If true, blocks until the archiver has fully synced. - * @param searchStartBlock - The block number to use for starting the search. - * @param searchEndBlock - The highest block number that we should search up to. - * @returns A array of L2 block bodies as well as the next eth block to search from - */ -export async function retrieveBlockBodiesFromAvailabilityOracle( - publicClient: PublicClient, - availabilityOracleAddress: EthAddress, - blockUntilSynced: boolean, - searchStartBlock: bigint, - searchEndBlock: bigint, -): Promise> { - const retrievedBlockBodies: Body[] = []; - - do { - if (searchStartBlock > searchEndBlock) { - break; - } - const l2TxsPublishedLogs = await getTxsPublishedLogs( - publicClient, - availabilityOracleAddress, - searchStartBlock, - searchEndBlock, - ); - if (l2TxsPublishedLogs.length === 0) { - break; - } - - const newBlockBodies = await processTxsPublishedLogs(publicClient, l2TxsPublishedLogs); - retrievedBlockBodies.push(...newBlockBodies.map(([body]) => body)); - searchStartBlock = l2TxsPublishedLogs[l2TxsPublishedLogs.length - 1].blockNumber + 1n; + expectedNextL2BlockNum += BigInt(newBlocks.length); } while (blockUntilSynced && searchStartBlock <= searchEndBlock); - - return { lastProcessedL1BlockNumber: searchStartBlock - 1n, retrievedData: retrievedBlockBodies }; + return retrievedBlocks; } /** diff --git a/yarn-project/archiver/src/archiver/eth_log_handlers.ts b/yarn-project/archiver/src/archiver/eth_log_handlers.ts index 68a5173db39..fa218fb2086 100644 --- a/yarn-project/archiver/src/archiver/eth_log_handlers.ts +++ b/yarn-project/archiver/src/archiver/eth_log_handlers.ts @@ -1,22 +1,13 @@ -import { Body, InboxLeaf } from '@aztec/circuit-types'; +import { Body, InboxLeaf, L2Block, type ViemSignature } from '@aztec/circuit-types'; import { AppendOnlyTreeSnapshot, Header, Proof } from '@aztec/circuits.js'; import { type EthAddress } from '@aztec/foundation/eth-address'; import { Fr } from '@aztec/foundation/fields'; import { numToUInt32BE } from '@aztec/foundation/serialize'; -import { AvailabilityOracleAbi, InboxAbi, RollupAbi } from '@aztec/l1-artifacts'; +import { InboxAbi, RollupAbi } from '@aztec/l1-artifacts'; -import { - type Hex, - type Log, - type PublicClient, - decodeFunctionData, - getAbiItem, - getAddress, - hexToBytes, - slice, -} from 'viem'; +import { type Hex, type Log, type PublicClient, decodeFunctionData, getAbiItem, getAddress, hexToBytes } from 'viem'; -import { type L1PublishedData } from './structs/published.js'; +import { type L1Published, type L1PublishedData } from './structs/published.js'; /** * Processes newly received MessageSent (L1 to L2) logs. @@ -39,25 +30,21 @@ export function processMessageSentLogs( * @param publicClient - The viem public client to use for transaction retrieval. * @param expectedL2BlockNumber - The next expected L2 block number. * @param logs - L2BlockProposed logs. - * @returns - An array of tuples representing block metadata including the header, archive tree snapshot. + * @returns - An array blocks. */ export async function processL2BlockProposedLogs( publicClient: PublicClient, expectedL2BlockNumber: bigint, logs: Log[], -): Promise<[Header, AppendOnlyTreeSnapshot, L1PublishedData][]> { - const retrievedBlockMetadata: [Header, AppendOnlyTreeSnapshot, L1PublishedData][] = []; +): Promise[]> { + const retrievedBlocks: L1Published[] = []; for (const log of logs) { const blockNum = log.args.blockNumber; if (blockNum !== expectedL2BlockNumber) { throw new Error('Block number mismatch. Expected: ' + expectedL2BlockNumber + ' but got: ' + blockNum + '.'); } // TODO: Fetch blocks from calldata in parallel - const [header, archive] = await getBlockMetadataFromRollupTx( - publicClient, - log.transactionHash!, - log.args.blockNumber, - ); + const block = await getBlockFromRollupTx(publicClient, log.transactionHash!, log.args.blockNumber); const l1: L1PublishedData = { blockNumber: log.blockNumber, @@ -65,11 +52,11 @@ export async function processL2BlockProposedLogs( timestamp: await getL1BlockTime(publicClient, log.blockNumber), }; - retrievedBlockMetadata.push([header, archive, l1]); + retrievedBlocks.push({ data: block, l1 }); expectedL2BlockNumber++; } - return retrievedBlockMetadata; + return retrievedBlocks; } export async function getL1BlockTime(publicClient: PublicClient, blockNumber: bigint): Promise { @@ -77,45 +64,33 @@ export async function getL1BlockTime(publicClient: PublicClient, blockNumber: bi return block.timestamp; } -export async function processTxsPublishedLogs( - publicClient: PublicClient, - logs: Log[], -): Promise<[Body, Buffer][]> { - const retrievedBlockBodies: [Body, Buffer][] = []; - for (const log of logs) { - const newBlockBody = await getBlockBodiesFromAvailabilityOracleTx(publicClient, log.transactionHash!); - retrievedBlockBodies.push([newBlockBody, Buffer.from(hexToBytes(log.args.txsEffectsHash))]); - } - - return retrievedBlockBodies; -} - /** - * Gets block metadata (header and archive snapshot) from the calldata of an L1 transaction. + * Gets block from the calldata of an L1 transaction. * Assumes that the block was published from an EOA. * TODO: Add retries and error management. * @param publicClient - The viem public client to use for transaction retrieval. * @param txHash - Hash of the tx that published it. * @param l2BlockNum - L2 block number. - * @returns L2 block metadata (header and archive) from the calldata, deserialized + * @returns L2 block from the calldata, deserialized */ -async function getBlockMetadataFromRollupTx( +async function getBlockFromRollupTx( publicClient: PublicClient, txHash: `0x${string}`, l2BlockNum: bigint, -): Promise<[Header, AppendOnlyTreeSnapshot]> { +): Promise { const { input: data } = await publicClient.getTransaction({ hash: txHash }); const { functionName, args } = decodeFunctionData({ abi: RollupAbi, data, }); - if (!(functionName === 'propose' || functionName === 'proposeWithBody')) { + if (!(functionName === 'propose')) { throw new Error(`Unexpected method called ${functionName}`); } - const [headerHex, archiveRootHex, _] = args! as readonly [Hex, Hex, Hex]; + const [headerHex, archiveRootHex, , , , bodyHex] = args! as readonly [Hex, Hex, Hex, Hex[], ViemSignature[], Hex]; const header = Header.fromBuffer(Buffer.from(hexToBytes(headerHex))); + const blockBody = Body.fromBuffer(Buffer.from(hexToBytes(bodyHex))); const blockNumberFromHeader = header.globalVariables.blockNumber.toBigInt(); @@ -130,59 +105,7 @@ async function getBlockMetadataFromRollupTx( ]), ); - return [header, archive]; -} - -/** - * Gets block bodies from calldata of an L1 transaction, and deserializes them into Body objects. - * @note Assumes that the block was published using `propose` or `publish`. - * TODO: Add retries and error management. - * @param publicClient - The viem public client to use for transaction retrieval. - * @param txHash - Hash of the tx that published it. - * @returns An L2 block body from the calldata, deserialized - */ -async function getBlockBodiesFromAvailabilityOracleTx( - publicClient: PublicClient, - txHash: `0x${string}`, -): Promise { - const { input: data } = await publicClient.getTransaction({ hash: txHash }); - - // @note Use `forge inspect Rollup methodIdentifiers to get this, - // If using `forge sig` you will get an INVALID value for the case with a struct. - // [ - // "propose(bytes,bytes32,bytes32,(bool,uint8,bytes32,bytes32)[],bytes)": "08978fe9", - // "propose(bytes,bytes32,bytes32,bytes)": "81e6f472", - // "proposeWithBody(bytes,bytes32,bytes32,bytes32[],(bool,uint8,bytes32,bytes32)[],bytes)": "b2283b07", - // "publish(bytes calldata _body)" - // ] - const DATA_INDEX = [4, 3, 5, 0]; // index where the body is, in the parameters list - const SUPPORTED_SIGS = ['0x08978fe9', '0x81e6f472', '0xb2283b07', '0x7fd28346']; - - const signature = slice(data, 0, 4); - - if (!SUPPORTED_SIGS.includes(signature)) { - throw new Error(`Unexpected method called ${signature}`); - } - - // Check if explicitly calling the DA oracle - if (signature === SUPPORTED_SIGS[SUPPORTED_SIGS.length - 1]) { - const { args } = decodeFunctionData({ - abi: AvailabilityOracleAbi, - data, - }); - const [bodyHex] = args! as [Hex]; - const blockBody = Body.fromBuffer(Buffer.from(hexToBytes(bodyHex))); - return blockBody; - } else { - const { args } = decodeFunctionData({ - abi: RollupAbi, - data, - }); - const index = SUPPORTED_SIGS.indexOf(signature); - const bodyHex = args![DATA_INDEX[index]] as Hex; - const blockBody = Body.fromBuffer(Buffer.from(hexToBytes(bodyHex))); - return blockBody; - } + return new L2Block(archive, header, blockBody); } /** @@ -210,31 +133,6 @@ export function getL2BlockProposedLogs( }); } -/** - * Gets relevant `TxsPublished` logs from chain. - * @param publicClient - The viem public client to use for transaction retrieval. - * @param dataAvailabilityOracleAddress - The address of the availability oracle contract. - * @param fromBlock - First block to get logs from (inclusive). - * @param toBlock - Last block to get logs from (inclusive). - * @returns An array of `TxsPublished` logs. - */ -export function getTxsPublishedLogs( - publicClient: PublicClient, - dataAvailabilityOracleAddress: EthAddress, - fromBlock: bigint, - toBlock: bigint, -): Promise[]> { - return publicClient.getLogs({ - address: getAddress(dataAvailabilityOracleAddress.toString()), - event: getAbiItem({ - abi: AvailabilityOracleAbi, - name: 'TxsPublished', - }), - fromBlock, - toBlock: toBlock + 1n, // the toBlock argument in getLogs is exclusive - }); -} - /** * Get relevant `MessageSent` logs emitted by Inbox on chain. * @param publicClient - The viem public client to use for transaction retrieval. diff --git a/yarn-project/archiver/src/index.ts b/yarn-project/archiver/src/index.ts index 37b0f6da3b6..f23f973101c 100644 --- a/yarn-project/archiver/src/index.ts +++ b/yarn-project/archiver/src/index.ts @@ -13,7 +13,7 @@ export * from './archiver/index.js'; export * from './rpc/index.js'; export * from './factory.js'; -export { retrieveL2ProofVerifiedEvents, retrieveBlockMetadataFromRollup } from './archiver/data_retrieval.js'; +export { retrieveL2ProofVerifiedEvents, retrieveBlockFromRollup } from './archiver/data_retrieval.js'; export { getL2BlockProposedLogs } from './archiver/eth_log_handlers.js'; @@ -37,7 +37,6 @@ async function main() { const archiver = new Archiver( publicClient, l1Contracts.rollupAddress, - l1Contracts.availabilityOracleAddress, l1Contracts.inboxAddress, l1Contracts.registryAddress, archiverStore, diff --git a/yarn-project/aztec-node/src/aztec-node/server.test.ts b/yarn-project/aztec-node/src/aztec-node/server.test.ts index f753431b918..7dbd162fbb9 100644 --- a/yarn-project/aztec-node/src/aztec-node/server.test.ts +++ b/yarn-project/aztec-node/src/aztec-node/server.test.ts @@ -69,7 +69,6 @@ describe('aztec node', () => { registryAddress: EthAddress.ZERO, inboxAddress: EthAddress.ZERO, outboxAddress: EthAddress.ZERO, - availabilityOracleAddress: EthAddress.ZERO, }, }, p2p, diff --git a/yarn-project/aztec-node/src/aztec-node/server.ts b/yarn-project/aztec-node/src/aztec-node/server.ts index ed438882142..be8d69c9af8 100644 --- a/yarn-project/aztec-node/src/aztec-node/server.ts +++ b/yarn-project/aztec-node/src/aztec-node/server.ts @@ -115,8 +115,7 @@ export class AztecNodeService implements AztecNode { `Rollup: ${config.l1Contracts.rollupAddress.toString()}\n` + `Registry: ${config.l1Contracts.registryAddress.toString()}\n` + `Inbox: ${config.l1Contracts.inboxAddress.toString()}\n` + - `Outbox: ${config.l1Contracts.outboxAddress.toString()}\n` + - `Availability Oracle: ${config.l1Contracts.availabilityOracleAddress.toString()}`; + `Outbox: ${config.l1Contracts.outboxAddress.toString()}`; this.log.info(message); } diff --git a/yarn-project/aztec.js/src/contract/contract.test.ts b/yarn-project/aztec.js/src/contract/contract.test.ts index f13bbfd954a..73076d9b456 100644 --- a/yarn-project/aztec.js/src/contract/contract.test.ts +++ b/yarn-project/aztec.js/src/contract/contract.test.ts @@ -22,7 +22,6 @@ describe('Contract Class', () => { const mockTxReceipt = { type: 'TxReceipt' } as any as TxReceipt; const mockUnconstrainedResultValue = 1; const l1Addresses: L1ContractAddresses = { - availabilityOracleAddress: EthAddress.random(), rollupAddress: EthAddress.random(), registryAddress: EthAddress.random(), inboxAddress: EthAddress.random(), diff --git a/yarn-project/aztec/src/cli/aztec_start_options.ts b/yarn-project/aztec/src/cli/aztec_start_options.ts index 953f07adcb3..4b551d2dc24 100644 --- a/yarn-project/aztec/src/cli/aztec_start_options.ts +++ b/yarn-project/aztec/src/cli/aztec_start_options.ts @@ -132,12 +132,6 @@ export const aztecStartOptions: { [key: string]: AztecStartOption[] } = { defaultValue: undefined, envVar: 'OUTBOX_CONTRACT_ADDRESS', }, - { - flag: '--availability-oracle-address ', - description: 'The deployed L1 availability oracle contract address', - defaultValue: undefined, - envVar: 'AVAILABILITY_ORACLE_CONTRACT_ADDRESS', - }, { flag: '--fee-juice-address ', description: 'The deployed L1 Fee Juice contract address', diff --git a/yarn-project/aztec/src/sandbox.ts b/yarn-project/aztec/src/sandbox.ts index 81e24a94856..200cdcbe04f 100644 --- a/yarn-project/aztec/src/sandbox.ts +++ b/yarn-project/aztec/src/sandbox.ts @@ -14,8 +14,6 @@ import { import { createDebugLogger } from '@aztec/foundation/log'; import { retryUntil } from '@aztec/foundation/retry'; import { - AvailabilityOracleAbi, - AvailabilityOracleBytecode, FeeJuicePortalAbi, FeeJuicePortalBytecode, InboxAbi, @@ -105,10 +103,6 @@ export async function deployContractsToL1( contractAbi: OutboxAbi, contractBytecode: OutboxBytecode, }, - availabilityOracle: { - contractAbi: AvailabilityOracleAbi, - contractBytecode: AvailabilityOracleBytecode, - }, rollup: { contractAbi: RollupAbi, contractBytecode: RollupBytecode, diff --git a/yarn-project/aztec/terraform/node/main.tf b/yarn-project/aztec/terraform/node/main.tf index 4d684d4ed1b..9933df21a32 100644 --- a/yarn-project/aztec/terraform/node/main.tf +++ b/yarn-project/aztec/terraform/node/main.tf @@ -315,10 +315,6 @@ resource "aws_ecs_task_definition" "aztec-node" { name = "REGISTRY_CONTRACT_ADDRESS" value = data.terraform_remote_state.l1_contracts.outputs.registry_contract_address }, - { - name = "AVAILABILITY_ORACLE_CONTRACT_ADDRESS" - value = data.terraform_remote_state.l1_contracts.outputs.availability_oracle_contract_address - }, { name = "FEE_JUICE_CONTRACT_ADDRESS" value = data.terraform_remote_state.l1_contracts.outputs.fee_juice_contract_address diff --git a/yarn-project/aztec/terraform/prover-node/main.tf b/yarn-project/aztec/terraform/prover-node/main.tf index 21bbf2fe783..a9c9a9708ca 100644 --- a/yarn-project/aztec/terraform/prover-node/main.tf +++ b/yarn-project/aztec/terraform/prover-node/main.tf @@ -273,7 +273,6 @@ resource "aws_ecs_task_definition" "aztec-prover-node" { { name = "INBOX_CONTRACT_ADDRESS", value = data.terraform_remote_state.l1_contracts.outputs.inbox_contract_address }, { name = "OUTBOX_CONTRACT_ADDRESS", value = data.terraform_remote_state.l1_contracts.outputs.outbox_contract_address }, { name = "REGISTRY_CONTRACT_ADDRESS", value = data.terraform_remote_state.l1_contracts.outputs.registry_contract_address }, - { name = "AVAILABILITY_ORACLE_CONTRACT_ADDRESS", value = data.terraform_remote_state.l1_contracts.outputs.availability_oracle_contract_address }, { name = "FEE_JUICE_CONTRACT_ADDRESS", value = data.terraform_remote_state.l1_contracts.outputs.fee_juice_contract_address }, { name = "FEE_JUICE_PORTAL_CONTRACT_ADDRESS", value = data.terraform_remote_state.l1_contracts.outputs.FEE_JUICE_PORTAL_CONTRACT_ADDRESS }, diff --git a/yarn-project/circuit-types/src/body.ts b/yarn-project/circuit-types/src/body.ts index c0e63aa43d3..21eba85c30e 100644 --- a/yarn-project/circuit-types/src/body.ts +++ b/yarn-project/circuit-types/src/body.ts @@ -48,7 +48,7 @@ export class Body { /** * Computes the transactions effects hash for the L2 block - * This hash is also computed in the `AvailabilityOracle`. + * This hash is also computed in the `TxDecoder`. * @returns The txs effects hash. */ getTxsEffectsHash() { diff --git a/yarn-project/cli/src/cmds/l1/deploy_l1_contracts.ts b/yarn-project/cli/src/cmds/l1/deploy_l1_contracts.ts index b57bae95ee9..08a4cc2cbf5 100644 --- a/yarn-project/cli/src/cmds/l1/deploy_l1_contracts.ts +++ b/yarn-project/cli/src/cmds/l1/deploy_l1_contracts.ts @@ -27,7 +27,6 @@ export async function deployL1Contracts( log(`Registry Address: ${l1ContractAddresses.registryAddress.toString()}`); log(`L1 -> L2 Inbox Address: ${l1ContractAddresses.inboxAddress.toString()}`); log(`L2 -> L1 Outbox Address: ${l1ContractAddresses.outboxAddress.toString()}`); - log(`Availability Oracle Address: ${l1ContractAddresses.availabilityOracleAddress.toString()}`); log(`Fee Juice Address: ${l1ContractAddresses.feeJuiceAddress.toString()}`); log(`Fee Juice Portal Address: ${l1ContractAddresses.feeJuicePortalAddress.toString()}`); } diff --git a/yarn-project/cli/src/cmds/pxe/get_node_info.ts b/yarn-project/cli/src/cmds/pxe/get_node_info.ts index 8128de79104..a3347f792d5 100644 --- a/yarn-project/cli/src/cmds/pxe/get_node_info.ts +++ b/yarn-project/cli/src/cmds/pxe/get_node_info.ts @@ -13,7 +13,6 @@ export async function getNodeInfo(rpcUrl: string, debugLogger: DebugLogger, log: log(` Registry Address: ${info.l1ContractAddresses.registryAddress.toString()}`); log(` L1 -> L2 Inbox Address: ${info.l1ContractAddresses.inboxAddress.toString()}`); log(` L2 -> L1 Outbox Address: ${info.l1ContractAddresses.outboxAddress.toString()}`); - log(` Availability Oracle Address: ${info.l1ContractAddresses.availabilityOracleAddress.toString()}`); log(` Fee Juice Address: ${info.l1ContractAddresses.feeJuiceAddress.toString()}`); log(` Fee Juice Portal Address: ${info.l1ContractAddresses.feeJuicePortalAddress.toString()}`); diff --git a/yarn-project/cli/src/utils/aztec.ts b/yarn-project/cli/src/utils/aztec.ts index 06f234a1207..921fd4ac56d 100644 --- a/yarn-project/cli/src/utils/aztec.ts +++ b/yarn-project/cli/src/utils/aztec.ts @@ -69,8 +69,7 @@ export async function deployAztecContracts( RegistryBytecode, RollupAbi, RollupBytecode, - AvailabilityOracleAbi, - AvailabilityOracleBytecode, + FeeJuicePortalAbi, FeeJuicePortalBytecode, PortalERC20Abi, @@ -96,10 +95,6 @@ export async function deployAztecContracts( contractAbi: OutboxAbi, contractBytecode: OutboxBytecode, }, - availabilityOracle: { - contractAbi: AvailabilityOracleAbi, - contractBytecode: AvailabilityOracleBytecode, - }, rollup: { contractAbi: RollupAbi, contractBytecode: RollupBytecode, diff --git a/yarn-project/end-to-end/src/composed/integration_l1_publisher.test.ts b/yarn-project/end-to-end/src/composed/integration_l1_publisher.test.ts index 89041117a11..21c8d5a7d5e 100644 --- a/yarn-project/end-to-end/src/composed/integration_l1_publisher.test.ts +++ b/yarn-project/end-to-end/src/composed/integration_l1_publisher.test.ts @@ -2,7 +2,6 @@ import { type ArchiveSource } from '@aztec/archiver'; import { getConfigEnvVars } from '@aztec/aztec-node'; import { AztecAddress, - Body, EthCheatCodes, Fr, GlobalVariables, @@ -38,7 +37,7 @@ import { fr, makeScopedL2ToL1Message } from '@aztec/circuits.js/testing'; import { type L1ContractAddresses, createEthereumChain } from '@aztec/ethereum'; import { makeTuple, range } from '@aztec/foundation/array'; import { openTmpStore } from '@aztec/kv-store/utils'; -import { AvailabilityOracleAbi, OutboxAbi, RollupAbi } from '@aztec/l1-artifacts'; +import { OutboxAbi, RollupAbi } from '@aztec/l1-artifacts'; import { SHA256Trunc, StandardTree } from '@aztec/merkle-tree'; import { getVKTreeRoot } from '@aztec/noir-protocol-circuits-types'; import { TxProver } from '@aztec/prover-client'; @@ -57,7 +56,6 @@ import { type HttpTransport, type PublicClient, type WalletClient, - decodeEventLog, encodeFunctionData, getAbiItem, getAddress, @@ -323,29 +321,6 @@ describe('L1Publisher integration', () => { return blockTicket; }; - it('Block body is correctly published to AvailabilityOracle', async () => { - const body = Body.random(); - // `sendPublishTx` function is private so I am hacking around TS here. I think it's ok for test purposes. - const txHash = await (publisher as any).sendPublishTx(body.toBuffer()); - const txReceipt = await publicClient.waitForTransactionReceipt({ - hash: txHash, - }); - - // Exactly 1 event should be emitted in the transaction - expect(txReceipt.logs.length).toBe(1); - - // We decode the event log before checking it - const txLog = txReceipt.logs[0]; - const topics = decodeEventLog({ - abi: AvailabilityOracleAbi, - data: txLog.data, - topics: txLog.topics, - }); - // Sol gives bytes32 txsHash, so we pad the ts bytes31 version - // We check that the txsHash in the TxsPublished event is as expected - expect(topics.args.txsEffectsHash).toEqual(`0x${body.getTxsEffectsHash().toString('hex').padStart(64, '0')}`); - }); - it(`Build ${numberOfConsecutiveBlocks} blocks of 4 bloated txs building on each other`, async () => { const archiveInRollup_ = await rollup.read.archive(); expect(hexStringToBuffer(archiveInRollup_.toString())).toEqual(new Fr(GENESIS_ARCHIVE_ROOT).toBuffer()); @@ -427,7 +402,7 @@ describe('L1Publisher integration', () => { const expectedData = encodeFunctionData({ abi: RollupAbi, - functionName: 'proposeWithBody', + functionName: 'propose', args: [ `0x${block.header.toBuffer().toString('hex')}`, `0x${block.archive.root.toBuffer().toString('hex')}`, @@ -527,31 +502,18 @@ describe('L1Publisher integration', () => { hash: logs[i].transactionHash!, }); - const expectedData = - i == 0 - ? encodeFunctionData({ - abi: RollupAbi, - functionName: 'proposeWithBody', - args: [ - `0x${block.header.toBuffer().toString('hex')}`, - `0x${block.archive.root.toBuffer().toString('hex')}`, - `0x${block.header.hash().toBuffer().toString('hex')}`, - [], - [], - `0x${block.body.toBuffer().toString('hex')}`, - ], - }) - : encodeFunctionData({ - abi: RollupAbi, - functionName: 'propose', - args: [ - `0x${block.header.toBuffer().toString('hex')}`, - `0x${block.archive.root.toBuffer().toString('hex')}`, - `0x${block.header.hash().toBuffer().toString('hex')}`, - [], - [], - ], - }); + const expectedData = encodeFunctionData({ + abi: RollupAbi, + functionName: 'propose', + args: [ + `0x${block.header.toBuffer().toString('hex')}`, + `0x${block.archive.root.toBuffer().toString('hex')}`, + `0x${block.header.hash().toBuffer().toString('hex')}`, + [], + [], + `0x${block.body.toBuffer().toString('hex')}`, + ], + }); expect(ethTx.input).toEqual(expectedData); await progressTimeBySlot(); diff --git a/yarn-project/end-to-end/src/fixtures/setup_l1_contracts.ts b/yarn-project/end-to-end/src/fixtures/setup_l1_contracts.ts index 4cc48f528fa..62099e72a14 100644 --- a/yarn-project/end-to-end/src/fixtures/setup_l1_contracts.ts +++ b/yarn-project/end-to-end/src/fixtures/setup_l1_contracts.ts @@ -1,7 +1,5 @@ import { type DebugLogger, type L1ContractArtifactsForDeployment, deployL1Contracts } from '@aztec/aztec.js'; import { - AvailabilityOracleAbi, - AvailabilityOracleBytecode, FeeJuicePortalAbi, FeeJuicePortalBytecode, InboxAbi, @@ -41,10 +39,6 @@ export const setupL1Contracts = async ( contractAbi: OutboxAbi, contractBytecode: OutboxBytecode, }, - availabilityOracle: { - contractAbi: AvailabilityOracleAbi, - contractBytecode: AvailabilityOracleBytecode, - }, rollup: { contractAbi: RollupAbi, contractBytecode: RollupBytecode, diff --git a/yarn-project/end-to-end/src/fixtures/utils.ts b/yarn-project/end-to-end/src/fixtures/utils.ts index 94c65a0c088..1045d838954 100644 --- a/yarn-project/end-to-end/src/fixtures/utils.ts +++ b/yarn-project/end-to-end/src/fixtures/utils.ts @@ -44,8 +44,6 @@ import { NULL_KEY, isAnvilTestChain } from '@aztec/ethereum'; import { bufferAsFields } from '@aztec/foundation/abi'; import { makeBackoff, retry, retryUntil } from '@aztec/foundation/retry'; import { - AvailabilityOracleAbi, - AvailabilityOracleBytecode, FeeJuicePortalAbi, FeeJuicePortalBytecode, InboxAbi, @@ -133,10 +131,6 @@ export const setupL1Contracts = async ( contractAbi: OutboxAbi, contractBytecode: OutboxBytecode, }, - availabilityOracle: { - contractAbi: AvailabilityOracleAbi, - contractBytecode: AvailabilityOracleBytecode, - }, rollup: { contractAbi: RollupAbi, contractBytecode: RollupBytecode, diff --git a/yarn-project/ethereum/src/deploy_l1_contracts.ts b/yarn-project/ethereum/src/deploy_l1_contracts.ts index eaf4f83f13b..290d0728a68 100644 --- a/yarn-project/ethereum/src/deploy_l1_contracts.ts +++ b/yarn-project/ethereum/src/deploy_l1_contracts.ts @@ -73,10 +73,6 @@ export interface L1ContractArtifactsForDeployment { * Outbox contract artifacts */ outbox: ContractArtifacts; - /** - * Availability Oracle contract artifacts - */ - availabilityOracle: ContractArtifacts; /** * Registry contract artifacts */ @@ -186,9 +182,6 @@ export const deployL1Contracts = async ( const registryAddress = await deployer.deploy(contractsToDeploy.registry, [account.address.toString()]); logger.info(`Deployed Registry at ${registryAddress}`); - const availabilityOracleAddress = await deployer.deploy(contractsToDeploy.availabilityOracle); - logger.info(`Deployed AvailabilityOracle at ${availabilityOracleAddress}`); - const feeJuiceAddress = await deployer.deploy(contractsToDeploy.feeJuice); logger.info(`Deployed Fee Juice at ${feeJuiceAddress}`); @@ -199,7 +192,6 @@ export const deployL1Contracts = async ( const rollupAddress = await deployer.deploy(contractsToDeploy.rollup, [ getAddress(registryAddress.toString()), - getAddress(availabilityOracleAddress.toString()), getAddress(feeJuicePortalAddress.toString()), args.vkTreeRoot.toString(), account.address.toString(), @@ -318,7 +310,6 @@ export const deployL1Contracts = async ( logger.verbose(`All transactions for L1 deployment have been mined`); const l1Contracts: L1ContractAddresses = { - availabilityOracleAddress, rollupAddress, registryAddress, inboxAddress, diff --git a/yarn-project/ethereum/src/l1_contract_addresses.ts b/yarn-project/ethereum/src/l1_contract_addresses.ts index 6ec61e8b24c..0230ad3d491 100644 --- a/yarn-project/ethereum/src/l1_contract_addresses.ts +++ b/yarn-project/ethereum/src/l1_contract_addresses.ts @@ -8,7 +8,6 @@ import type { DebugLogger } from '@aztec/foundation/log'; * For reference: https://github.com/AztecProtocol/aztec-packages/pull/5553 */ export const l1ContractsNames = [ - 'availabilityOracleAddress', 'rollupAddress', 'registryAddress', 'inboxAddress', @@ -27,11 +26,6 @@ export type L1ContractAddresses = { const parseEnv = (val: string) => EthAddress.fromString(val); export const l1ContractAddressesMapping: ConfigMappingsType = { - availabilityOracleAddress: { - env: 'AVAILABILITY_ORACLE_CONTRACT_ADDRESS', - description: 'The deployed L1 availability oracle contract address.', - parseEnv, - }, rollupAddress: { env: 'ROLLUP_CONTRACT_ADDRESS', description: 'The deployed L1 rollup contract address.', diff --git a/yarn-project/foundation/src/config/env_var.ts b/yarn-project/foundation/src/config/env_var.ts index a99708bf74a..be9b805d42a 100644 --- a/yarn-project/foundation/src/config/env_var.ts +++ b/yarn-project/foundation/src/config/env_var.ts @@ -11,7 +11,6 @@ export type EnvVar = | 'REGISTRY_CONTRACT_ADDRESS' | 'INBOX_CONTRACT_ADDRESS' | 'OUTBOX_CONTRACT_ADDRESS' - | 'AVAILABILITY_ORACLE_CONTRACT_ADDRESS' | 'FEE_JUICE_CONTRACT_ADDRESS' | 'FEE_JUICE_PORTAL_CONTRACT_ADDRESS' | 'ARCHIVER_URL' diff --git a/yarn-project/l1-artifacts/scripts/generate-artifacts.sh b/yarn-project/l1-artifacts/scripts/generate-artifacts.sh index 520619a558b..1d82db63c06 100755 --- a/yarn-project/l1-artifacts/scripts/generate-artifacts.sh +++ b/yarn-project/l1-artifacts/scripts/generate-artifacts.sh @@ -10,7 +10,6 @@ target_dir=./generated # - a .{CONTRACT_NAME}Abi.ts containing the contract ABI. CONTRACTS=( - "l1-contracts:AvailabilityOracle" "l1-contracts:Registry" "l1-contracts:Inbox" "l1-contracts:Outbox" diff --git a/yarn-project/pxe/src/pxe_service/test/pxe_service.test.ts b/yarn-project/pxe/src/pxe_service/test/pxe_service.test.ts index b8931137f01..b634eb62777 100644 --- a/yarn-project/pxe/src/pxe_service/test/pxe_service.test.ts +++ b/yarn-project/pxe/src/pxe_service/test/pxe_service.test.ts @@ -26,7 +26,6 @@ function createPXEService(): Promise { node.getVersion.mockResolvedValue(1); node.getChainId.mockResolvedValue(1); const mockedContracts: L1ContractAddresses = { - availabilityOracleAddress: EthAddress.random(), rollupAddress: EthAddress.random(), registryAddress: EthAddress.random(), inboxAddress: EthAddress.random(), diff --git a/yarn-project/sequencer-client/src/publisher/l1-publisher.test.ts b/yarn-project/sequencer-client/src/publisher/l1-publisher.test.ts index 6722ded9101..d06fdf32116 100644 --- a/yarn-project/sequencer-client/src/publisher/l1-publisher.test.ts +++ b/yarn-project/sequencer-client/src/publisher/l1-publisher.test.ts @@ -1,6 +1,7 @@ import { L2Block, type ViemSignature } from '@aztec/circuit-types'; import { EthAddress } from '@aztec/circuits.js'; import { sleep } from '@aztec/foundation/sleep'; +import { RollupAbi } from '@aztec/l1-artifacts'; import { NoopTelemetryClient } from '@aztec/telemetry-client/noop'; import { type MockProxy, mock } from 'jest-mock-extended'; @@ -9,31 +10,11 @@ import { type GetTransactionReceiptReturnType, type PrivateKeyAccount } from 'vi import { type PublisherConfig, type TxSenderConfig } from './config.js'; import { L1Publisher } from './l1-publisher.js'; -interface MockAvailabilityOracleWrite { - publish: (args: readonly [`0x${string}`], options: { account: PrivateKeyAccount }) => Promise<`0x${string}`>; -} - -interface MockAvailabilityOracleEstimate { - publish: (args: readonly [`0x${string}`], options: { account: PrivateKeyAccount }) => Promise; -} - -interface MockAvailabilityOracleRead { - isAvailable: (args: readonly [`0x${string}`]) => Promise; -} - -class MockAvailabilityOracle { - constructor( - public write: MockAvailabilityOracleWrite, - public simulate: MockAvailabilityOracleWrite, - public estimateGas: MockAvailabilityOracleEstimate, - public read: MockAvailabilityOracleRead, - ) {} -} - interface MockPublicClient { getTransactionReceipt: ({ hash }: { hash: '0x${string}' }) => Promise; getBlock(): Promise<{ timestamp: bigint }>; getTransaction: ({ hash }: { hash: '0x${string}' }) => Promise<{ input: `0x${string}`; hash: `0x${string}` }>; + estimateGas: ({ to, data }: { to: '0x${string}'; data: '0x${string}' }) => Promise; } interface MockRollupContractWrite { @@ -58,30 +39,17 @@ interface MockRollupContractRead { } class MockRollupContract { - constructor( - public write: MockRollupContractWrite, - public simulate: MockRollupContractWrite, - public read: MockRollupContractRead, - ) {} + constructor(public write: MockRollupContractWrite, public read: MockRollupContractRead, public abi = RollupAbi) {} } describe('L1Publisher', () => { let rollupContractRead: MockProxy; let rollupContractWrite: MockProxy; - let rollupContractSimulate: MockProxy; let rollupContract: MockRollupContract; - let availabilityOracleRead: MockProxy; - let availabilityOracleWrite: MockProxy; - let availabilityOracleSimulate: MockProxy; - let availabilityOracleEstimate: MockProxy; - let availabilityOracle: MockAvailabilityOracle; - let publicClient: MockProxy; - let processTxHash: `0x${string}`; let proposeTxHash: `0x${string}`; - let processTxReceipt: GetTransactionReceiptReturnType; let proposeTxReceipt: GetTransactionReceiptReturnType; let l2Block: L2Block; @@ -104,14 +72,8 @@ describe('L1Publisher', () => { blockHash = l2Block.header.hash().toBuffer(); body = l2Block.body.toBuffer(); - processTxHash = `0x${Buffer.from('txHashProcess').toString('hex')}`; // random tx hash proposeTxHash = `0x${Buffer.from('txHashPropose').toString('hex')}`; // random tx hash - processTxReceipt = { - transactionHash: processTxHash, - status: 'success', - logs: [], - } as unknown as GetTransactionReceiptReturnType; proposeTxReceipt = { transactionHash: proposeTxHash, status: 'success', @@ -119,20 +81,8 @@ describe('L1Publisher', () => { } as unknown as GetTransactionReceiptReturnType; rollupContractWrite = mock(); - rollupContractSimulate = mock(); rollupContractRead = mock(); - rollupContract = new MockRollupContract(rollupContractWrite, rollupContractSimulate, rollupContractRead); - - availabilityOracleWrite = mock(); - availabilityOracleRead = mock(); - availabilityOracleSimulate = mock(); - availabilityOracleEstimate = mock(); - availabilityOracle = new MockAvailabilityOracle( - availabilityOracleWrite, - availabilityOracleSimulate, - availabilityOracleEstimate, - availabilityOracleRead, - ); + rollupContract = new MockRollupContract(rollupContractWrite, rollupContractRead); publicClient = mock(); @@ -141,7 +91,6 @@ describe('L1Publisher', () => { l1ChainId: 1, publisherPrivateKey: `0xac0974bec39a17e36ba4a6b4d238ff944bacb478cbed5efcae784d7bf4f2ff80`, l1Contracts: { - availabilityOracleAddress: EthAddress.ZERO.toString(), rollupAddress: EthAddress.ZERO.toString(), }, l1PublishRetryIntervalMS: 1, @@ -149,21 +98,19 @@ describe('L1Publisher', () => { publisher = new L1Publisher(config, new NoopTelemetryClient()); - (publisher as any)['availabilityOracleContract'] = availabilityOracle; (publisher as any)['rollupContract'] = rollupContract; (publisher as any)['publicClient'] = publicClient; account = (publisher as any)['account']; rollupContractRead.getCurrentSlot.mockResolvedValue(l2Block.header.globalVariables.slotNumber.toBigInt()); - availabilityOracleEstimate.publish.mockResolvedValueOnce(GAS_GUESS); publicClient.getBlock.mockResolvedValue({ timestamp: 12n }); + publicClient.estimateGas.mockResolvedValue(GAS_GUESS); }); it('publishes and propose l2 block to l1', async () => { rollupContractRead.archive.mockResolvedValue(l2Block.header.lastArchive.root.toString() as `0x${string}`); rollupContractWrite.propose.mockResolvedValueOnce(proposeTxHash); - rollupContractSimulate.propose.mockResolvedValueOnce(proposeTxHash); publicClient.getTransactionReceipt.mockResolvedValueOnce(proposeTxReceipt); @@ -180,42 +127,14 @@ describe('L1Publisher', () => { ] as const; expect(rollupContractWrite.propose).toHaveBeenCalledWith(args, { account: account, - gas: L1Publisher.PROPOSE_GAS_GUESS + GAS_GUESS * 2n, + gas: L1Publisher.PROPOSE_GAS_GUESS + GAS_GUESS, }); expect(publicClient.getTransactionReceipt).toHaveBeenCalledWith({ hash: proposeTxHash }); }); - it('publishes l2 block to l1 (already published body)', async () => { - availabilityOracleRead.isAvailable.mockResolvedValueOnce(true); - rollupContractRead.archive.mockResolvedValue(l2Block.header.lastArchive.root.toString() as `0x${string}`); - rollupContractWrite.propose.mockResolvedValueOnce(processTxHash); - rollupContractSimulate.propose.mockResolvedValueOnce(processTxHash); - publicClient.getTransactionReceipt.mockResolvedValueOnce(processTxReceipt); - - const result = await publisher.processL2Block(l2Block); - - expect(result).toEqual(true); - const args = [ - `0x${header.toString('hex')}`, - `0x${archive.toString('hex')}`, - `0x${blockHash.toString('hex')}`, - [], - ] as const; - expect(rollupContractWrite.propose).toHaveBeenCalledWith(args, { account, gas: L1Publisher.PROPOSE_GAS_GUESS }); - expect(publicClient.getTransactionReceipt).toHaveBeenCalledWith({ hash: processTxHash }); - }); - it('does not retry if sending a propose tx fails', async () => { - availabilityOracleRead.isAvailable.mockResolvedValueOnce(true); rollupContractRead.archive.mockResolvedValue(l2Block.header.lastArchive.root.toString() as `0x${string}`); - rollupContractWrite.propose - .mockRejectedValueOnce(new Error()) - .mockResolvedValueOnce(processTxHash as `0x${string}`); - - // Note that simulate will be valid both times - rollupContractSimulate.propose - .mockResolvedValueOnce(processTxHash as `0x${string}`) - .mockResolvedValueOnce(processTxHash as `0x${string}`); + rollupContractWrite.propose.mockRejectedValueOnce(new Error()).mockResolvedValueOnce(proposeTxHash); const result = await publisher.processL2Block(l2Block); @@ -235,7 +154,6 @@ describe('L1Publisher', () => { it('does not retry if sending a publish and propose tx fails', async () => { rollupContractRead.archive.mockResolvedValue(l2Block.header.lastArchive.root.toString() as `0x${string}`); - rollupContractSimulate.propose.mockResolvedValueOnce(proposeTxHash as `0x${string}`); rollupContractWrite.propose.mockRejectedValueOnce(new Error()); const result = await publisher.processL2Block(l2Block); @@ -245,11 +163,9 @@ describe('L1Publisher', () => { }); it('retries if fetching the receipt fails (propose)', async () => { - availabilityOracleRead.isAvailable.mockResolvedValueOnce(true); rollupContractRead.archive.mockResolvedValue(l2Block.header.lastArchive.root.toString() as `0x${string}`); - rollupContractSimulate.propose.mockResolvedValueOnce(processTxHash); - rollupContractWrite.propose.mockResolvedValueOnce(processTxHash); - publicClient.getTransactionReceipt.mockRejectedValueOnce(new Error()).mockResolvedValueOnce(processTxReceipt); + rollupContractWrite.propose.mockResolvedValueOnce(proposeTxHash); + publicClient.getTransactionReceipt.mockRejectedValueOnce(new Error()).mockResolvedValueOnce(proposeTxReceipt); const result = await publisher.processL2Block(l2Block); @@ -259,7 +175,6 @@ describe('L1Publisher', () => { it('retries if fetching the receipt fails (publish propose)', async () => { rollupContractRead.archive.mockResolvedValue(l2Block.header.lastArchive.root.toString() as `0x${string}`); - rollupContractSimulate.propose.mockResolvedValueOnce(proposeTxHash as `0x${string}`); rollupContractWrite.propose.mockResolvedValueOnce(proposeTxHash as `0x${string}`); publicClient.getTransactionReceipt.mockRejectedValueOnce(new Error()).mockResolvedValueOnce(proposeTxReceipt); @@ -280,10 +195,9 @@ describe('L1Publisher', () => { }); it('returns false if propose tx reverts', async () => { - availabilityOracleRead.isAvailable.mockResolvedValueOnce(true); rollupContractRead.archive.mockResolvedValue(l2Block.header.lastArchive.root.toString() as `0x${string}`); - publicClient.getTransactionReceipt.mockResolvedValueOnce({ ...processTxReceipt, status: 'reverted' }); + publicClient.getTransactionReceipt.mockResolvedValueOnce({ ...proposeTxReceipt, status: 'reverted' }); const result = await publisher.processL2Block(l2Block); @@ -303,9 +217,8 @@ describe('L1Publisher', () => { }); it('returns false if sending propose tx is interrupted', async () => { - availabilityOracleRead.isAvailable.mockResolvedValueOnce(true); rollupContractRead.archive.mockResolvedValue(l2Block.header.lastArchive.root.toString() as `0x${string}`); - rollupContractWrite.propose.mockImplementationOnce(() => sleep(10, processTxHash) as Promise<`0x${string}`>); + rollupContractWrite.propose.mockImplementationOnce(() => sleep(10, proposeTxHash) as Promise<`0x${string}`>); const resultPromise = publisher.processL2Block(l2Block); publisher.interrupt(); diff --git a/yarn-project/sequencer-client/src/publisher/l1-publisher.ts b/yarn-project/sequencer-client/src/publisher/l1-publisher.ts index 2aa3c4173f6..b92561f6808 100644 --- a/yarn-project/sequencer-client/src/publisher/l1-publisher.ts +++ b/yarn-project/sequencer-client/src/publisher/l1-publisher.ts @@ -8,7 +8,7 @@ import { createDebugLogger } from '@aztec/foundation/log'; import { serializeToBuffer } from '@aztec/foundation/serialize'; import { InterruptibleSleep } from '@aztec/foundation/sleep'; import { Timer } from '@aztec/foundation/timer'; -import { AvailabilityOracleAbi, RollupAbi } from '@aztec/l1-artifacts'; +import { RollupAbi } from '@aztec/l1-artifacts'; import { type TelemetryClient } from '@aztec/telemetry-client'; import pick from 'lodash.pick'; @@ -22,6 +22,7 @@ import { type WalletClient, createPublicClient, createWalletClient, + encodeFunctionData, getAddress, getContract, hexToBytes, @@ -107,10 +108,6 @@ export class L1Publisher { private metrics: L1PublisherMetrics; private log = createDebugLogger('aztec:sequencer:publisher'); - private availabilityOracleContract: GetContractReturnType< - typeof AvailabilityOracleAbi, - WalletClient - >; private rollupContract: GetContractReturnType< typeof RollupAbi, WalletClient @@ -139,11 +136,6 @@ export class L1Publisher { transport: http(chain.rpcUrl), }); - this.availabilityOracleContract = getContract({ - address: getAddress(l1Contracts.availabilityOracleAddress.toString()), - abi: AvailabilityOracleAbi, - client: walletClient, - }); this.rollupContract = getContract({ address: getAddress(l1Contracts.rollupAddress.toString()), abi: RollupAbi, @@ -196,6 +188,7 @@ export class L1Publisher { formattedSignatures, `0x${attestationData.digest.toString('hex')}`, ts, + `0x${header.contentCommitment.txsEffectsHash.toString('hex')}`, flags, ] as const; @@ -218,11 +211,6 @@ export class L1Publisher { return committee.map(EthAddress.fromString); } - checkIfTxsAreAvailable(block: L2Block): Promise { - const args = [`0x${block.body.getTxsEffectsHash().toString('hex').padStart(64, '0')}`] as const; - return this.availabilityOracleContract.read.isAvailable(args); - } - async getTransactionStats(txHash: string): Promise { const tx = await this.publicClient.getTransaction({ hash: txHash as Hex }); if (!tx) { @@ -260,11 +248,8 @@ export class L1Publisher { // Publish body and propose block (if not already published) if (!this.interrupted) { - let txHash; const timer = new Timer(); - const isAvailable = await this.checkIfTxsAreAvailable(block); - // @note This will make sure that we are passing the checks for our header ASSUMING that the data is also made available // This means that we can avoid the simulation issues in later checks. // By simulation issue, I mean the fact that the block.timestamp is equal to the last block, not the next, which @@ -274,12 +259,7 @@ export class L1Publisher { signatures: attestations ?? [], }); - if (isAvailable) { - this.log.verbose(`Transaction effects of block ${block.number} already published.`, ctx); - txHash = await this.sendProposeWithoutBodyTx(proposeTxArgs); - } else { - txHash = await this.sendProposeTx(proposeTxArgs); - } + const txHash = await this.sendProposeTx(proposeTxArgs); if (!txHash) { this.log.info(`Failed to publish block ${block.number} to L1`, ctx); @@ -412,59 +392,19 @@ export class L1Publisher { } } - // This is used in `integration_l1_publisher.test.ts` currently. Could be removed though. - private async sendPublishTx(encodedBody: Buffer): Promise { + private async sendProposeTx(encodedData: L1ProcessArgs): Promise { if (!this.interrupted) { try { - this.log.info(`TxEffects size=${encodedBody.length} bytes`); - const args = [`0x${encodedBody.toString('hex')}`] as const; - - await this.availabilityOracleContract.simulate.publish(args, { - account: this.account, + // We have to jump a few hoops because viem is not happy around estimating gas for view functions + const computeTxsEffectsHashGas = await this.publicClient.estimateGas({ + to: this.rollupContract.address, + data: encodeFunctionData({ + abi: this.rollupContract.abi, + functionName: 'computeTxsEffectsHash', + args: [`0x${encodedData.body.toString('hex')}`], + }), }); - return await this.availabilityOracleContract.write.publish(args, { - account: this.account, - }); - } catch (err) { - this.log.error(`TxEffects publish failed`, err); - return undefined; - } - } - } - - private async sendProposeWithoutBodyTx(encodedData: L1ProcessArgs): Promise { - if (!this.interrupted) { - try { - const attestations = encodedData.attestations - ? encodedData.attestations.map(attest => attest.toViemSignature()) - : []; - const txHashes = encodedData.txHashes ? encodedData.txHashes.map(txHash => txHash.to0xString()) : []; - const args = [ - `0x${encodedData.header.toString('hex')}`, - `0x${encodedData.archive.toString('hex')}`, - `0x${encodedData.blockHash.toString('hex')}`, - txHashes, - attestations, - ] as const; - - return await this.rollupContract.write.propose(args, { - account: this.account, - gas: L1Publisher.PROPOSE_GAS_GUESS, - }); - } catch (err) { - this.log.error(`Rollup publish failed`, err); - return undefined; - } - } - } - - private async sendProposeTx(encodedData: L1ProcessArgs): Promise { - if (!this.interrupted) { - try { - const publishGas = await this.availabilityOracleContract.estimateGas.publish([ - `0x${encodedData.body.toString('hex')}`, - ]); const min = (a: bigint, b: bigint) => (a > b ? b : a); // @note We perform this guesstimate instead of the usual `gasEstimate` since @@ -472,7 +412,7 @@ export class L1Publisher { // we will fail estimation in the case where we are simulating for the // first ethereum block within our slot (as current time is not in the // slot yet). - const gasGuesstimate = min(publishGas * 2n + L1Publisher.PROPOSE_GAS_GUESS, 15_000_000n); + const gasGuesstimate = min(computeTxsEffectsHashGas + L1Publisher.PROPOSE_GAS_GUESS, 15_000_000n); const attestations = encodedData.attestations ? encodedData.attestations.map(attest => attest.toViemSignature()) @@ -487,7 +427,7 @@ export class L1Publisher { `0x${encodedData.body.toString('hex')}`, ] as const; - return await this.rollupContract.write.proposeWithBody(args, { + return await this.rollupContract.write.propose(args, { account: this.account, gas: gasGuesstimate, }); diff --git a/yarn-project/simulator/src/avm/serialization/bytecode_serialization.ts b/yarn-project/simulator/src/avm/serialization/bytecode_serialization.ts index 3b435b0386a..571c758d483 100644 --- a/yarn-project/simulator/src/avm/serialization/bytecode_serialization.ts +++ b/yarn-project/simulator/src/avm/serialization/bytecode_serialization.ts @@ -2,7 +2,6 @@ import { PedersenCommitment } from '../opcodes/commitment.js'; import { DAGasLeft, L2GasLeft } from '../opcodes/context_getters.js'; import { EcAdd } from '../opcodes/ec_add.js'; import { Keccak, KeccakF1600, Pedersen, Poseidon2, Sha256 } from '../opcodes/hashing.js'; -import { Instruction } from '../opcodes/index.js'; import { Add, Address, @@ -24,6 +23,7 @@ import { FieldDiv, FunctionSelector, GetContractInstance, + Instruction, InternalCall, InternalReturn, Jump,