This repo has been built with Foundry. Given the interconnected nature of Aztec Connect Bridges with existing mainnet protocols, we decided Foundry / Forge offered the best support for testing. This repo should make debugging, mainnet-forking, impersonation and gas profiling simple. It makes sense to test Solidity contracts with Solidity, not with the added complication of Ethers / Typescript.
Developing a bridge is simple. It is done entirely in Solidity and no knowledge of cryptography underpinning Aztec Connect is required. Users of your bridge will get the full benefits of ironclad privacy and 10-30x gas savings.
Note: Currently adding a new bridge or asset to Aztec Connect is permissioned and requires our approval. Once Aztec Connect leaves beta this won't be the case anymore and developing a bridge will become completely permissionless.
To get started follow the steps below:
-
Fork this repository on GitHub, clone your fork and create a new branch:
git clone https://github.com/YOUR_USERNAME/aztec-connect-bridges.git git checkout -b your_username/bridge-name
-
Install dependencies and build the repo:
cd aztec-connect-bridges yarn setup
-
Copy and rename the following folders (e.g. rename example to uniswap):
src/bridges/example src/test/example src/client/example src/specs/example src/deployment/example
-
Implement the bridges and tests. See the example bridge, example bridge tests and documentation of IDefiBridge for more details. For a more complex example check out other bridges in this repository.
-
Debug your bridge:
forge test --match-contract YourBridge -vvv
-
Write a deployment script. Make a script that inherits from the
BaseDeployment.s.sol
file. The base provides helper functions for listing assets/bridges and a getter for the rollup address. Use the env variablesbroadcast = false|true
andnetwork=mainnet|devnet|testnet
to specify how to run it, withbroadcast = true
, thelistBridge
andlistAsset
helpers will be broadcast, otherwise they are similuated as if they came from the controller. See the example scripts from other bridges, for inspiration on how to do it.
All bridges need to be submitted via PRs to this repo. To receive a grant payment we expect the following work to be done:
- A solidity bridge that interfaces with the protocol you are bridging to (e.g AAVE),
- tests in Solidity that test the bridge with production values and the deployed protocol that is currently on mainnet (you should test a range of assets, edge cases and use Forge's fuzzing abilities),
- tests cover the full contract, there are no untested functions or lines.
- implementation of the Typescript
bridge-data.ts
class that tells a frontend developer how to use your bridge. - an explanation of the flows your bridge supports should be included as
spec.md
, - NatSpec documentation of all the functions in all the contracts which are to be deployed on mainnet,
- a deployment script to deploy the bridge with proper configuration,
- Subsidy contract has been integrated (see the Subsidy integration section for details).
Before submitting a PR for a review make sure that the following is true:
- All the tests you wrote pass (
forge test --match-contract TestName
), - there are no linting errors (
yarn lint
), - you fetched upstream changes to your fork on GitHub and your branch has been rebased against the head of the
master
branch (not merged, if you are not sure how to rebase check out this article), - the diff contains only changes related to the PR description,
- NatSpec documentation has already been written,
- a spec was written,
- a deployment script was written,
- Subsidy has been integrated.
You can find more information about setting up connections to bridge contracts with the SDK on the Ethereum Interactions page of the docs.
This repo includes an Infura key that allows forking from mainnet. We have included helpers to make testing easier (see example bridge tests).
In production a bridge is called by a user creating a client side proof via the Aztec SDK.
These transaction proofs are sent to a rollup provider for aggregation.
The rollup provider then sends the aggregate rollup proof with the sum of all users' proofs for a given bridgeCallData
to your bridge contract.
A bridgeCallData
uniquely defines the expected inputs/outputs of a DeFi interaction.
It is a uint256
that represents a bit-string containing multiple fields.
When unpacked its data is used to create a BridgeData struct in the rollup processor contract.
Structure of the bit-string is as follows (starting at the least significant bit):
bit position | bit length | definition | description |
---|---|---|---|
0 | 32 | bridgeAddressId |
id of bridge smart contract address |
32 | 30 | inputAssetA |
asset id of 1st input asset |
62 | 30 | inputAssetB |
asset id of 1st input asset |
92 | 30 | outputAssetA |
asset id of 1st output asset |
122 | 30 | outputAssetB |
asset id of 2nd output asset |
152 | 32 | bitConfig |
flags that describe asset types |
184 | 64 | auxData |
custom auxiliary data for bridge-specific logic |
Bit Config Definition:
bit | meaning |
---|---|
0 | secondInputInUse |
1 | secondOutputInUse |
Note 1: Last 8 bits of
bridgeCallData
bit-string are wasted because the circuits don't support values of full 256 bits (248 is the largest multiple of 8 that we can use).
Note 2:
bitConfig
is 32 bits large even though we only use 2 bits because we need it to be future proofed (e.g. we might add NFT support, and we would need new bit flag for that).
The rollup contract uses this to construct the function parameters to pass into your bridge contract (via convert(...)
function).
It calls your bridge with a fixed amount of gas via a delegateCall
via the DefiBridgeProxy.sol
contract.
We decided to have 2 separate approaches of bridge testing:
-
In the first one it is expected that you call convert function directly on the bridge contract. This allows for simple debugging because execution traces are simple. Disadvantage of this approach is that you have take care of transferring tokens to and from the bridge (this is handle by the DefiBridgeProxy contract in the production environment). This type of test can be considered to be a unit test and an example of such a test is here.
-
In the second approach we construct a
bridgeCallData
, we mock proof data and verifier's response, and we pass this data directly to the RollupProcessor'sprocessRollup(...)
function. The purpose of this test is to test the bridge in an environment that is as close to the final deployment as possible without spinning up all the rollup infrastructure (sequencer, proof generator etc.). This test can be considered an end-to-end test of the bridge and an example of such a test is here.
We encourage you to first write all tests as unit tests to be able to leverage simple traces while you are debugging the bridge.
Once you make the bridge work in the unit tests environment convert the relevant tests to E2E.
Converting unit tests to E2E is straightforward because we made the BridgeTestBase.sendDefiRollup(...)
function return the same values as IDefiBridge.convert(...)
.
In production, the rollup contract will supply _totalInputValue
of both input assets and use the _interactionNonce
as a globally unique ID.
For testing, you may provide this value.
The rollup contract will send _totalInputValue
of _inputAssetA
and _inputAssetB
ahead of the call to convert.
In production, the rollup contract already has these tokens as they are the users' funds.
For testing use the deal
method from the forge-std's Test
class (see ExampleUnit.t.sol for details).
This method prefunds the rollup with sufficient tokens to enable the transfer.
After extending the BridgeTestBase
or Test
class simply call:
deal(address(dai), address(rollupProcessor), amount);
This will set the balance of the rollup to the amount required.
The rollup contract expects a bridge to have approved for transfer by the rollupAddress
the ERC20 tokens that are returned via outputValueA
or outputValueB
for a given asset.
The DeFiBridgeProxy
will attempt to recover the tokens by calling transferFrom(bridgeAddress, rollupAddress, amount)
for these values once bridge.convert()
or bridge.finalise()
have executed (in production and in end-to-end tests).
In unit tests it is expected that you transfer these tokens on your own.
ETH is returned to the rollup from a bridge by calling the payable function with a msg.value
rollupContract.receiveETH(uint256 interactionNonce)
.
You must also set the outputValue
of the corresponding outputAsset
(A or B) to be the amount of ETH sent.
This repo supports TypeChain so all Typescript bindings will be auto generated and added to the typechain-types
folder.
This is a Typescript class designed to help a developer on the frontend use your bridge. You should implement the functions to fetch data from your bridge / L1.
You can also find more information on the Aztec docs site.
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 1 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 assets 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. A single Aztec Connect transaction requires 3 base fees or approximately ~8000 gas.
The user then pays their share of the L1 DeFi transaction. The rollup will call a bridge contract with a fixed deterministic amount of gas so the total fee is simple ~8000 gas + BRIDGE_GAS / BATCH_SIZE.
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 sent to the bridge are public.
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 (e.g. Uniswap Position NFTs 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 transferred.
The position tokens have an assetId
that is the interactionNonce
of the DeFi Bridge call.
This is globally unique.
Virtual assets can be used to construct complex flows, such as entering or exiting LP positions (e.g. one bridge contract can have multiple flows which are triggered using different input assets).
- Swapping - 1 real input, 1 real output
- Swapping with incentives - 1 real input, 2 real outputs (2nd output reward token)
- Borrowing - 1 real input (collateral), 1 real output (borrowed asset, e.g. Dai), 1 virtual output (represents the position, e.g. position is a vault in MakerDao)
- Purchasing an NFT - 1 real input, 1 virtual output (asset representing NFT)
- Selling an NFT - 1 virtual input (asset representing NFT), 1 real output (e.g. ETH)
- Repaying a loan - 1 real input (e.g. Dai), 1 virtual input (representing the vault/position), 1 real output (collateral, e.g. ETH)
- Repaying a loan 2 - 1 real input (e.g. USDC), 1 virtual input (representing the position), 2 real outputs ( 1st output collateral, 2nd output reward token, e.g. AAVE)
- Partial loan repaying - 1 real input (e.g. Dai), 1 virtual input (representing the vault/position), 1 real output (collateral, e.g. ETH), 1 virtual output (representing the vault/position)
- Claiming fees from Uniswap position and redepositing them - 1 virtual input (represents LP position NFT), 1 virtual output (representing the modified LP position)
There are 2 calls the bridge has to do for the subsidy to work.
The first one is calling setGasUsageAndMinGasPerMinute(...)
from constructor or any other function which is called before convert(...)
method.
This method has 3 parameters:
_criteria
is a value defining a specific bridge call. This value is expected to be computed differently in different bridges and should identify a specific bridge flow (e.g. it can be a hash of the input and output token addresses or just the_auxData
or whatever makes sense)._gasUsage
is an estimated gas consumption of the bridge call for a given_criteria
. This is used as an upper limit for subsidy. If this is set correctly, at maximum less than 100% of the call should be covered since the subsidy computation is only based on base fee and is not taking tips into consideration._minGasPerMinute
is the minimum amount of gas per minute that subsidizer has to subsidize. In general, we recommend this value to be computed in such a way that the subsidy funder has to fund at least 1 call a day (in such a case compute the value as_gasUsage / (24 * 60)
). This value is set from the bridge in order to avoid a griefing attack (malicious subsidy funder setting a subsidy with very lowminGasPerMinute
value and effectively blocking everyone else from subsidizing the given bridge and criteria).
There are 2 versions of this method. In the first one the parameters are single values. In the second one the parameters are arrays of values.
The second required call is calling the claimSubsidy(...)
function from within the convert(...)
method.
This method has 2 parameters:
_criteria
same as above,_rollupBeneficiary
is the address of the beneficiary. This value is passed to theconvert(...)
function.
For more info on how the Subsidy works see the documentation in the Subsidy.sol file.
See the example bridge for an example subsidy integration.
Please reach out on Discord with any questions. You can join our Discord here.