Official repository for all smart wallet contracts used by Circle web3 API/SDK.
This repository includes support for both the Hardhat and Foundry frameworks. Going forward, all new code / tests / scripts should be built on Foundry. Hardhat is currently only included for legacy contracts in the following folders:
src/account/v1src/paymaster/v1/permissioned
- Run
git submodule update --init --recursiveto update/download all libraries. - Run
yarn installto install any additional dependencies. - Run
curl -L https://foundry.paradigm.xyz | bashand follow the outputted instructions to source env file. - Run
foundryup - Create a
.envfile and provide the required API keys, wallets (can be generated for deployment), and configuration values. You can look at the.env.examplefor reference.
Run yarn lint to lint all .sol files in the src and test directories. Run yarn format:check and yarn format:write to check for, and fix formatting issues, respectively.
To run tests using Foundry, follow the steps below:
- Run
yarn build - Run
yarn test
To generate a viewable test coverage report, run:
brew install lcovif not yet installedforge coverage --ir-minimum --report lcov && genhtml lcov.info -o report --branch-coverage && open report/index.html(Note: some contracts like WeightedMultisigPlugin require using --ir-minimum because of stack depth. To build coverage faster locally, comment out this and dependent contracts and omit --ir-minimum flag.)
We use Github actions to run linter and all the tests. The workflow configuration can be found in .github/workflows/ci.yml
We are using Conventional Commit structure to automatically generate releases.
To export contract bytecode & ABI for programmable wallet:
- Execute
forge build src/msca/6900/v0.7 --force --extra-output-files abi evm. Replace v0.7 with v0.8 if you want v0.8. - Execute
make abigen - Interface files will appear under
abigenfolder - After changes are merged, release new repository tag
- update
buidl-wallet-contractsgo mod version of programmable-wallet, and import bytecode fromabigenfolder
For running integration tests in Anvil node, run make anvil-tests. This runs the python tests in test/anvil
- Run
yarn build - Run
yarn gasreport
-
anvil -
update
.env(pointing to local) andsource .env -
deploy (only choose the account type you're interested in benchmarking)
-
forge script script/<SCRIPT_NAME> --rpc-url $RPC_URL --broadcast --verify -vvvv --slow --watchExample:
forge script script/001_DeployPluginManager.s.sol --rpc-url $RPC_URL --broadcast --verify -vvvv --slow --watch
-
-
cast code $address
npx hardhat node (if using hardhat stack)
make anvil (if using foundry stack). To get a list of pre-funded addresses, you can look at the beginning of the logs in the anvil Docker container, or reference https://github.com/foundry-rs/foundry/blob/0d8302880b79fa9c3c4aa52ab446583dece19a34/crates/anvil/README.md?plain=1#L48.
- Deployment
-
Run the command
env $(grep -v '^#' .env) yarn hardhat deploy --network <chain>where<chain>is one of {mumbai,goerli}.If you only want to deploy a specific set of smart contracts, you can add the
--tagsflag, for example:env $(grep -v '^#' .env) yarn hardhat deploy --tags SponsorPaymaster --network goerli
-
- Verification
-
ECDSA wallet factory
env $(grep -v '^#' .env) npx hardhat verify --network mumbai --constructor-args script/ecdsa_account_factory_constructor_args.js ECDSA_ACCOUNT_FACTORY_ADDRESS -
Sponsor paymaster
env $(grep -v '^#' .env) npx hardhat verify --network goerli --constructor-args script/sponsor_paymaster_constructor_args.js STABLECOIN_PAYMASTER_ADDRESS -
Fallback: If the verification commands do not work for contracts deployed through the hardhat deployment scripts, you can still verify manually through etherscan's UI by submitting the standard input json.
You can find this file under
deployments/polygon/solcInputs(you can try different blockchains but I'm unsure of results). Then submit the file that you think is the one for the contract you're trying to verify. It's a bit of guessing, but you can look at the source code to try and figure it out. You may also need to verify the proxies manually through etherscan after having verified the implementation.
-
-
Set up
DEPLOYER_PRIVATE_KEY,RPC_URLandETHERSCAN_API_KEYin .env -
Run
source .env -
Run the desired numbered scripts inside the
script/folder using the below command format. Make sure any environment variable values from previous steps are updated if needed as you progress through the scripts.-
forge script script/<SCRIPT_NAME> --rpc-url $RPC_URL --broadcast --verify -vvvvExample:
forge script script/001_DeployPluginManager.s.sol --rpc-url $RPC_URL --broadcast --verify -vvvv
Tip: before executing the above command, verify the simulation works as expected by running the above command without the
--broadcastand--verifyflags. This way, you can also make sure your address will have enough tokens to cover the transaction fee estimated in the simulation.Note: if you are deploying on a local network, don't use the
--verifyflag. -
-
Include the relevant logs from the
broadcastfolder in your commit.Tip: if you did multiple runs, search the appropriate block explorer for the tx hash corresponding to the desired contract's deployment, and then search the logs for the transaction hash.
Tip: logs are organized by chain ID in the lower levels in the
broadcastfolder. Use https://chainlist.org/ to lookup IDs of common chains. -
Create or update the corresponding file in the
script/cmdfolder using the creation bytecode of the contract from the logs. See the below "Chain Expansion" section for details on how to format the files in thescript/cmdfolder. -
Verify in block explorer like etherscan using standard input json
-
Create the standard input json: run the below command
forge verify-contract <contract_address> <relative_path_to_source:classname> --show-standard-json-input > <script/verify/<filename> # Example forge verify-contract 0x03431fb00fb2e26b5bc502dfef8da30e1c8643b8 src/msca/6900/v0.7/plugins/v1_0_0/utility/DefaultTokenCallbackPlugin.sol:DefaultTokenCallbackPlugin --show-standard-json-input > script/verify/DefaultTokenCallbackPlugin.json
-
Verify and publish in block explorer (etherscan example)
-
Compiler type:
Solidity (Standard-Json-Input) -
Compiler version:
v0.8.24 -
License: Option 5 in https://etherscan.io/contract-license-types
-
Upload the JSON file
-
If the contract you are verifying took constructor arguments, input the output of the below command with the
0xprefix removed into the "Constructor Arguments ABI-encoded" section:cast abi-encode "constructor(<arg type 1>, <arg type 2>, ...)" "arg1Val" "arg2Val" # Example cast abi-encode "constructor(address,address,uint256)" "0x0166EA90E565476f13c6a0D25ED2C35599E58785" "0x0000000071727De22E5E9d8BAf0edAc6f37da032" 18
Make sure to save this output under
script/verify/<ContractName>/abiEncodedConstructorArgs.Tip: if your contract's constructor arguments have complicated types, you can get this value instead by
console.loging the output ofabi.encode(arg1, arg2, ...)in Solidity. -
For account factory contracts, you may also need to manually verify the proxy contract. To do this, you will first need to create an account using the account factory, and follow the above strategy. After verifying one account, all subsequent proxies should be automatically detected.
Tip: analyze the account creation function on the factory (e.g.
createAccountforSingleOwnerMSCAFactory) to deduce the constructor arguments for the proxy deployment when trying to verify it on a block explorer. -
Click verify and publish
-
-
The below contract deployment are based on abi and bytecode to ensure the same address across all EVM-compatible chains. The abi and bytecode for each of the contracts are available at the locations below. These files were generated from first deployments, will all fields except abi and bytecode removed.
| Contract Name | Location |
|---|---|
| ECDSAAcountFactory | deploy/metadata/ECDSAAccountFactory.json |
| SponsorPaymaster_Implementation | deploy/metadata/SponsorPaymaster_Implementation.json |
| SponsorPaymaster_Proxy | deploy/metadata/SponsorPaymaster_Proxy.json |
To ensure consistent deployment of contracts across chains, we store the cast send deployment commands for contracts in the script/cmd folder. To deploy these contracts, simply run the command inside the file corresponding to that contract on the command line.
To build the cast send deployment command for a contract:
-
Run the Solidity deployment script for the contract in the
script/*folder: -
Find the deployment result in broadcast/*
-
Looking for “transactions” -> ”transaction” → “data” in run-*.json
-
Copy the "data" in step 2 without "0x" prefix.
-
Add
cast send --rpc-url $RPC_URL --private-key $DEPLOYER_PRIVATE_KEY 0x4e59b44847b379578588920cA78FbF26c0B4956C 0x0000000000000000000000000000000000000000000000000000000000000000in front of the bytecode from previous step (no space)Refer to the note after these steps, you may not need the
0x0...prefix. The0x4e59b44847b379578588920cA78FbF26c0B4956Cis foundry's default create2 address deployer. See https://book.getfoundry.sh/reference/cast/cast-create2. -
Run the command in step5 in Terminal.
-
Save this command to script/cmd folder with
<script_name> -
Create the README under
broadcastfolder for the chain id. For example, seebroadcast/011_DeployTokenCallbackPlugin.s.sol/11155111/README.md
Note: The "data" field may be called "input". Additionally, if the input/data field already contains the leading 0000000000000000000000000000000000000000000000000000000000000000 prefix, simply copy this value over directly (no need to re-add the zeros prefix).
If you encountered this error after executing make test, try increasing memory resource for docker engine.
If you are using macbook with M1 or newer chips, try enabling Use Rosetta for x86/amd64 emulation on Apple Silicon under Features in development in Docker settings.
As an alternative to the above, if you are using macbook with M1 or newer chips, try adding the --platform=linux/amd64 flag to the build Make command. If you encounter this error while running make anvil, make sure to run make before.
When deploying contract deployment scripts from the /script folder on a local chain (started up using make anvil), you can remove the --verify flag if you are getting errors related to the API key, such as Missing etherscan key for chain 31337.
Run forge clean && forge build.