You can either deploy all contracts at once or deploy each contract separately.
node version 18
if using nvm, run nvm use
to switch to node version 18
nvm use 18
cp .env.sample .env
Set PRIVATE_KEY
, RPC_URL
, CHAIN_ID
, CHAIN_NAME
in .env
file.
Other default values work for base sepolia. But adjust them if you are deploying to a different chain.
forge build --skip test --skip script
source .env && \
forge script script/DefaultSetupScript.s.sol:Deploy \
--rpc-url $RPC_URL \
--chain-id $CHAIN_ID \
--broadcast \
-vvv
You'll get a response like this:
== Logs ==
TokenRegistry proxy deployed at: 0xB895744454C62e9052A4332C29c4F19048054B56
TokenRegistry implementation deployed at: 0xee62Ca2257E98284eEBf66c8849cA00489Ff1E03
AllVerifiers implementation deployed at: 0x8350e89C182dD211180A8773cd91F79f2822BC9a
ECDSAOwnedDKIMRegistry deployed at: 0xA11D302310bAd2Da10216B0FA91e5A1bca631c19
Wallet implementation deployed at: 0x7a6361d28b1121EB8a030778f597A85443833cc9
Oauth core deployed at: 0x50bba18c7289bf4189E2d8e5ed40Ae86be020567
RelayerHandler proxy deployed at: 0x9F68880B9A760d15734872EbA3D519549C3996BB
RelayerHandler implementation deployed at: 0x9e119689Fc87be723886dA2dD012bF43D66c7BA4
ExtensionHandler proxy deployed at: 0x6D6bA7Ee9Ed199F67C6C26dfb3B6E56b60D9fdef
<TRUNCATED>
Save these you'll need them for setting up the relayer.
Before proceeding to deploying the relayer, you need to register the relayer data in the RelayerHandler
contract.
For this, set the following env variables first:
RPC_URL= # the RPC URL of the chain where you deployed the contracts
PRIVATE_KEY= # this is the relayer's private key
RELAYER_HANDLER= # this is the address of the RelayerHandler contract from the previous step
RELAYER_EMAIL=email@example.com
RELAYER_HOSTNAME=example.com
Then run the following command:
source .env && \
forge script script/RegisterRelayer.s.sol --rpc-url $RPC_URL --broadcast
Note: You can ignore the rest of this document if you've deployed all contracts at once and that's all you wanted to do. and you can move onto deploying the relayer
EmailWalletCore.sol
- The main contract that implements functions likehandleEmailOp
. To keep the contract simple, we have split other functionalities to many "handlers".handlers/RelayerHandler.sol
- Handles operations to register and update Relayer config.handlers/AccountHandler.sol
- Handles account related operations likecreateAccount
,initAccount
,transpotAccount
, and wallet related functions.handlers/UnclaimsHandler.sol
- Handles operations related to Unclaimed Funds and Unclaimed States.handlers/ExtensionHandler.sol
- Handles operations related to publishing and installing Extensions.
Wallet.sol
- Wallet contract deployed for each user (account). The owner of the wallet contract can execute any call data on a target. By default Core contract is the owner of the wallet contract.
interfaces/Types.sol
- Contains all the structs used in the contract including EmailOp.interfaces/Commands.sol
- Defines the commands used for EmailOps.interfaces/Events.sol
- Defines all events emitted by the contract.interfaces/IVerifier.sol
- Defines the interface for the ZK proof verifier contract.interfaces/IPriceOracle.sol
- Defines the interface for price oracle used for fee calculation.interfaces/Extension.sol
- Defines the interface for an Extension contract.
libraries/SubjectUtils.sol
- Contains helper functions to compute the subject line from an EmailOp.libraries/DecimalUtils.sol
- Contains helper functions to convert uint to decimal string.
utils/TokenRegistry.sol
- TokenRegistry contract that stores the address of ERC20 tokens by their names (and reverse) across multiple chains.utils/UniswapTWAPOracle.sol
- UniswapTWAPOracle contract that provides the TWAP price of a token in ETH.utils/ECDSAOwnedDKIMRegistry.sol
- A custom DKIM registry that allows a ECDSA signer to set the DKIM public key for a domain.
extensions/NFTExtension.sol
- NFT wallet extensions to send NFTs.extensions/UniswapExtension.sol
- Uniswap wallet extension to swap tokens on Uniswap.
verifier/Verifier.sol
- All verifiersverifier/AccountCreationVerifier.sol
- Verifier forAccountCreation
circuit.verifier/AccountInitVerifier.sol
- Verifier forAccountInit
circuit.verifier/EmailSenderVerifier.sol
- Verifier forEmailSender
circuit.verifier/ClaimVerifier.sol
- Verifier forClaim
circuit.verifier/AccountTransportVerifier.sol
- Verifier forAccountTransport
circuit.verifier/AnnouncementVerifier.sol
- Verifier forAnnouncement
circuit.
tests/*.t.sol
- Contains unit tests for all contracts/functionalities.tests/Integration.t.sol
- Contains all integration tests. Integration tests generate the proof using the circuit and verify it using contracts. Before running those tests, you need to make apackages/contracts/test/build_integration
directory, download the zip file from the following link, and place its unziped files under that directory. https://drive.google.com/file/d/1F1cw9eTlgJwPcXR4wF8kQrWPMhRsFnIy/view?usp=sharing
Make sure you have Foundry installed
Build the contracts using the below command.
forge build
Run unit tests
forge test --no-match-test "testIntegration"
Run integration tests Run each integration tests one by one as each test will consume lot of memory.
Eg: forge test --match-test 'testIntegration_Swap_Tokens' -vvv --ffi
Run the below commands to deploy each contracts. Ensure address of WETH and Uniswap Price oracle on the target chain.
PRIVATE_KEY="" \
forge script script/01_DeployTokenRegistry.s.sol:Deploy \
-vvv \
--rpc-url https://ethereum-sepolia.publicnode.com \
--chain-id 11155111 \
--broadcast \
--etherscan-api-key "" \
--verify
Copy the address from log TokenRegistry implementation deployed at: 0x9f44be9F69aF1e049dCeCDb2d9296f36C49Ceafb
PRIVATE_KEY="" \
forge script script/02_DeployAllVerifiers.s.sol:Deploy \
-vvv \
--rpc-url https://ethereum-sepolia.publicnode.com \
--chain-id 11155111 \
--broadcast \
--etherscan-api-key "" \
--verify
Copy the address from log AllVerifiers implementation deployed at: 0x9f44be9F69aF1e049dCeCDb2d9296f36C49Ceafb
PRIVATE_KEY="" \
forge script script/03_DeployDKIMRegistry.s.sol:Deploy \
-vvv \
--rpc-url https://ethereum-sepolia.publicnode.com \
--chain-id 11155111 \
--broadcast \
--etherscan-api-key "" \
--verify
Copy the address from log DKIMRegistry implementation deployed at: 0xbE66454b0Fa9E6b3D53DC1b0f9D21978bb864531
PRIVATE_KEY="" \
WETH=0x7b79995e5f793A07Bc00c21412e50Ecae098E7f9 \
UNISWAP_FACTORY=0x5C69bEe701ef814a2B6a3EDD4B1652CB9cc5aA6f \
forge script script/DeployUniswapTWAPOracle.s.sol:Deploy \
-vvv \
--rpc-url https://ethereum-sepolia.publicnode.com \
--chain-id 11155111 \
--broadcast \
--etherscan-api-key "" \
--verify
Copy the address from log UniswapTWAPOracle deployed at: 0x0000000000000000000000000000000000000000
PRIVATE_KEY="" \
WETH=0x7b79995e5f793A07Bc00c21412e50Ecae098E7f9 \
forge script script/04_DeployWallet.s.sol:Deploy \
-vvv \
--rpc-url https://ethereum-sepolia.publicnode.com \
--chain-id 11155111 \
--broadcast \
--etherscan-api-key "" \
--verify
Copy the address from log Wallet proxy deployed at: 0x0000000000000000000000000000000000000000
PRIVATE_KEY="" \
WETH=0x7b79995e5f793A07Bc00c21412e50Ecae098E7f9 \
forge script script/05_DeployHandlers.s.sol:Deploy \
-vvv \
--rpc-url https://ethereum-sepolia.publicnode.com \
--chain-id 11155111 \
--broadcast \
--etherscan-api-key "" \
--verify
Copy the addresses from log
RelayerHandler proxy deployed at: 0x0000000000000000000000000000000000000000
ExtensionHandler proxy deployed at: 0x0000000000000000000000000000000000000000
AccountHandler proxy deployed at: 0x0000000000000000000000000000000000000000
UnclaimsHandler proxy deployed at: 0x0000000000000000000000000000000000000000
PRIVATE_KEY="" \
WETH=0x7b79995e5f793A07Bc00c21412e50Ecae098E7f9 \
TOKEN_REGISTRY=0x9f44be9F69aF1e049dCeCDb2d9296f36C49Ceafb \
DKIM_REGISTRY=0xbE66454b0Fa9E6b3D53DC1b0f9D21978bb864531 \
PRICE_ORACLE=0xF5f40B12aa15286F0DE5610C4e29d87a97997ee7 \
forge script script/06_DeployEmailWalletCore.s.sol:Deploy \
-vvv \
--rpc-url https://ethereum-sepolia.publicnode.com \
--chain-id 11155111 \
--broadcast \
--etherscan-api-key "" \
--verify
Copy the addresses from log:
EmailWalletCore proxy deployed at: 0x0000000000000000000000000000000000000000
PRIVATE_KEY="" \
WETH=0x7b79995e5f793A07Bc00c21412e50Ecae098E7f9 \
TOKEN_REGISTRY=0x9f44be9F69aF1e049dCeCDb2d9296f36C49Ceafb \
DKIM_REGISTRY=0xbE66454b0Fa9E6b3D53DC1b0f9D21978bb864531 \
PRICE_ORACLE=0xF5f40B12aa15286F0DE5610C4e29d87a97997ee7 \
forge script script/07_SetDefaultExtensions.s.sol:Deploy \
-vvv \
--rpc-url https://ethereum-sepolia.publicnode.com \
--chain-id 11155111 \
--broadcast \
--etherscan-api-key "" \
--verify
Copy the addresses from log:
NFTExtension proxy deployed at: 0x0000000000000000000000000000000000000000
UniswapExtension proxy deployed at: 0x0000000000000000000000000000000000000000
Set the SIGNER
to the address of the Ethereum wallet who will be setting the DKIM public key for a domain.
PRIVATE_KEY="" \
SIGNER=0x2f6e79a6e1a982a49ca248b70b02f76e921af400 \
forge script script/DeployECDSAOwnedDKIMRegistry.s.sol:Deploy \
-vvv \
--rpc-url https://ethereum-sepolia.publicnode.com \
--chain-id 11155111 \
--broadcast \
--etherscan-api-key "S7RWR1ENYB73HZY7WTV3EER7U8CQNBBTAJ" \
--verify
Copy the address from log ECDSAOwnedDKIMRegistry deployed at: 0xB50a02E2Da524feC1209542985b2ae2917aF7265
TOKEN_REGISTRY=0xF1d24E5f7f0Ca617F0c1f3AA34A77EcFfaFedE8f \
PRIVATE_KEY=0x00 \
forge script script/XX_UpgradeTokenRegistry.s.sol:Deploy \
--rpc-url https://ethereum-sepolia.publicnode.com \
--chain-id 11155111 \
--broadcast \
-vvv