Aztec is a privacy focussed L2, that enables cheap private interactions with Layer 1 smart contracts and liquidity, via a process called DeFi aggregation. We use advanced zero-knowledge technology "zk-zk rollups" to add privacy and significant gas savings any Layer 1 protocol via Aztec Connect bridges.
A bridge is a Layer solidity contract deployed on Mainnet that conforms a DeFi protocol to the interface the Aztec rollup expects. This allows the Aztec rollup contract to interact with the DeFi protocol via the bridge.
A bridge contract models any layer 1 DeFi protocol as an asynchronous asset swap. You can specify up to two input assets and two output assets per bridge.
Users who have shielded fuassetsnds on Aztec can construct a zero-knowledge proof instructing the Aztec rollup contract to make an external L1 contract call.
Rollup providers batch multiple L2 transaction intents on the Aztec Network together in a rollup. The rollup contract then makes aggregate transaction against L1 DeFi contracts and returns the funds pro-rata to the users on L2.
Aztec connect transactions can be batched together into one rollup. Each user in the batch pays a base fee to cover the verification of the rollup proof and their share of the L1 DeFi transaction gas cost.
The largest batch size is 256 transactions. The cost to verify a rollup is fixed at ~400,000 gas, with a variable amount of gas for each transaction in the batch of ~3,500 gas.
Assuming 256 transactions in a batch each user would pay:
Rollup Verification gas cost: (400,000 / 256) + 3,500= 5062 gas Claim Rollup Verification gas cost: (400,000 / 256) + 3,500 = 5062 gas
Uniswap bridge gas cost: 180,000 / 256 = 703 gas
Total = ~11,000 gas for a Uniswap trade
The source of funds for any Aztec Connect transaction is an Aztec shielded asset. When a user interacts with an Aztec Connect Bridge contract, their identity is kept hidden, but balances set to the bridge are public.
Rollup providers are incentivised to batch any transaction with the same BridgeId. This reduces the cost of the L1 transaction for similar trades. A bridgeId consits of:
Aztec uses the concept of Virtual Assets or Position tokens to represent a share of assets held by a Bridge contract. This is far more gas efficient than minting ERC20 tokens.These are used when the bridge holds an asset that Aztec doesn't support i.e Uniswap Position NFT's or other non-fungible assets.
If the output asset of any interaction is specified as "VIRTUAL", the user will receive encrypted notes on Aztec representing their share of the position, but no Tokens or ETH need to be transfered. The position tokens have an assetId
that is the interactionNonce
of the DeFi Bridge call. This is globably unique. Virtual assets can be used to construct complex flows, such as entering or exiting LP positions. i.e one bridge contract can have multiple flows which are triggered using different input assets.
If the output asset of any interaction is specified as "VIRTUAL", the user will receive encrypted notes on Aztec representing their share of the position, but no Tokens or ETH need to be transfered. The position tokens have an assetId
that is the interactionNonce
of the DeFi Bridge call. This is globably unique. Virtual assets can be used to construct complex flows, such as entering or exiting LP positions. i.e one bridge contract can have multiple flows which are triggered using different input assets.
Aztec's Connect has been deployed to the Goerli testnet in Alpha. Your bridge contract needs to conform to the following interface. If so you can contact us for access to the SDK to create proofs.
// SPDX-License-Identifier: GPL-2.0-only
// Copyright 2020 Spilsbury Holdings Ltd
pragma solidity >=0.6.6 <0.8.0;
pragma experimental ABIEncoderV2;
import { Types } from "../Types.sol";
interface IDefiBridge {
/**
* Input cases:
* Case1: 1 real input.
* Case2: 1 virtual asset input.
* Case3: 1 real 1 virtual input.
*
* Output cases:
* Case1: 1 real
* Case2: 2 real
* Case3: 1 real 1 virtual
* Case4: 1 virtual
*
* Example use cases with asset mappings
* 1 1: Swapping.
* 1 2: Swapping with incentives (2nd output reward token).
* 1 3: Borrowing. Lock up collateral, get back loan asset and virtual position asset.
* 1 4: Opening lending position OR Purchasing NFT. Input real asset, get back virtual asset representing NFT or position.
* 2 1: Selling NFT. Input the virtual asset, get back a real asset.
* 2 2: Closing a lending position. Get back original asset and reward asset.
* 2 3: Claiming fees from an open position.
* 2 4: Voting on a 1 4 case.
* 3 1: Repaying a borrow. Return loan plus interest. Get collateral back.
* 3 2: Repaying a borrow. Return loan plus interest. Get collateral plus reward token. (AAVE)
* 3 3: Partial loan repayment.
* 3 4: DAO voting stuff.
*/
// @dev This function is called from the RollupProcessor.sol contract via the DefiBridgeProxy. It receives the aggreagte sum of all users funds for the input assets.
// @param AztecAsset inputAssetA a struct detailing the first input asset, this will always be set
// @param AztecAsset inputAssetB an optional struct detailing the second input asset, this is used for repaying borrows and should be virtual
// @param AztecAsset outputAssetA a struct detailing the first output asset, this will always be set
// @param AztecAsset outputAssetB a struct detailing an optional second output asset
// @param uint256 inputValue, the total amount input, if there are two input assets, equal amounts of both assets will have been input
// @param uint256 interactionNonce a globally unique identifier for this DeFi interaction. This is used as the assetId if one of the output assets is virtual
// @param uint64 auxData other data to be passed into the bridge contract (slippage / nftID etc)
// @return uint256 outputValueA the amount of outputAssetA returned from this interaction, should be 0 if async
// @return uint256 outputValueB the amount of outputAssetB returned from this interaction, should be 0 if async or bridge only returns 1 asset.
// @return bool isAsync a flag to toggle if this bridge interaction will return assets at a later date after some third party contract has interacted with it via finalise()
function convert(
Types.AztecAsset calldata inputAssetA,
Types.AztecAsset calldata inputAssetB,
Types.AztecAsset calldata outputAssetA,
Types.AztecAsset calldata outputAssetB,
uint256 inputValue,
uint256 interactionNonce,
uint64 auxData
)
external
payable
returns (
uint256 outputValueA,
uint256 outputValueB,
bool isAsync
);
// @dev This function is called from the RollupProcessor.sol contract via the DefiBridgeProxy
// @param uint256 interactionNonce
function canFinalise(uint256 interactionNonce) external view returns (bool);
// @dev This function is called from the RollupProcessor.sol contract via the DefiBridgeProxy. It receives the aggreagte sum of all users funds for the input assets.
// @param AztecAsset inputAssetA a struct detailing the first input asset, this will always be set
// @param AztecAsset inputAssetB an optional struct detailing the second input asset, this is used for repaying borrows and should be virtual
// @param AztecAsset outputAssetA a struct detailing the first output asset, this will always be set
// @param AztecAsset outputAssetB a struct detailing an optional second output asset
// @param uint256 interactionNonce
// @param uint64 auxData other data to be passed into the bridge contract (slippage / nftID etc)
// @return uint256 outputValueA the return value of output asset A
// @return uint256 outputValueB optional return value of output asset B
// @dev this function should have a modifier on it to ensure it can only be called by the Rollup Contract
function finalise(
Types.AztecAsset calldata inputAssetA,
Types.AztecAsset calldata inputAssetB,
Types.AztecAsset calldata outputAssetA,
Types.AztecAsset calldata outputAssetB,
uint256 interactionNonce,
uint64 auxData
) external payable returns (uint256 outputValueA, uint256 outputValueB);
}
If a Defi Bridge interaction is asynchronous, it must be finalised at a later date once the DeFi interaction has completed.
This is acheived by calling RollupProcessor.processAsyncDefiInteraction(uint256 interactionNonce)
. This internally will call finalise and ensure the correct amount of tokens have been transferred.
This function is called from the Aztec Rollup Contract via the DeFi Bridge Proxy. Before this function on your bridge contract is called the rollup contract will have sent you ETH or Tokens defined by the input params.
This function should interact with the DeFi protocol e.g Uniswap, and transfer tokens or ETH back to the Aztec Rollup Contract. The Rollup contract will check it received the correct amount.
If the DeFi interaction is ASYNC i.e it does not settle in the same block, the call to convert should return (0,0 true). The contract should record the interaction nonce for any Async position or if virtual assets are returned.
At a later date, this interaction can be finalised by proding the rollup contract to call finalise on the bridge.
This function checks to see if an async interaction is ready to settle. It should return true if it is.
This function will be called from the Azte Rollup contract. The Aztec rollup contract will check that it received the correct amount of ETH and Tokens specified by the return values, and trigger the settlement step on Aztec.