Setup
This guide provides step-by-step instructions to integrate the ZK Email Email Transaction Builder system into your existing contract using Foundry.
Install Dependencies
Install the necessary NPM packages in your project directory:
npm install @openzeppelin/contracts@5.0.0 @openzeppelin/contracts-upgradeable@5.0.0 @zk-email/contracts@6.1.5 @zk-email/ether-email-auth-contracts@0.0.2-preview
Packages Installed:
@zk-email/contracts@6.1.5
: Core contracts for ZK Email functionalities.@zk-email/ether-email-auth-contracts@0.0.2-preview
: Contracts for email authentication and verification.@openzeppelin/contracts@5.0.0
: Standard library of reusable smart contracts.@openzeppelin/contracts-upgradeable@5.0.0
: Upgradeable version of OpenZeppelin contracts.
In your Solidity contract file, import the required contracts and libraries:
// Import ZK Email contracts
import "@zk-email/ether-email-auth-contracts/src/EmailAuth.sol";
// Import OpenZeppelin contracts
import "@openzeppelin/contracts/utils/Create2.sol";
import {ERC1967Proxy} from "@openzeppelin/contracts/proxy/ERC1967/ERC1967Proxy.sol";
Add the following remappings to your remappings.txt
file or create the file if it doesn't exist:
@openzeppelin/=node_modules/@openzeppelin/
@zk-email/=node_modules/@zk-email/
If you are using a workspace or monorepo, you may need to update the node_modules
path to your root node_modules
directory.
Smart Contract Setup
Contract Variables
Add variables to store the addresses of the essential contracts:
address public verifierAddr;
address public dkimAddr;
address public emailAuthImplementationAddr;
Set the addresses of the Verifier, DKIM Registry, and EmailAuth implementation in the constructor:
constructor(
address _verifierAddr,
address _dkimAddr,
address _emailAuthImplementationAddr
) {
verifierAddr = _verifierAddr;
dkimAddr = _dkimAddr;
emailAuthImplementationAddr = _emailAuthImplementationAddr;
}
Provide functions to access these addresses:
function verifier() public view virtual returns (address) {
return verifierAddr;
}
function dkim() public view virtual returns (address) {
return dkimAddr;
}
function emailAuthImplementation() public view virtual returns (address) {
return emailAuthImplementationAddr;
}
Email Authentication
Implement a function to compute the deterministic address for the EmailAuth
contract:
function computeEmailAuthAddress(
address owner,
bytes32 accountSalt
) public view returns (address) {
return
Create2.computeAddress(
accountSalt,
keccak256(
abi.encodePacked(
type(ERC1967Proxy).creationCode,
abi.encode(
emailAuthImplementation(),
abi.encodeCall(
EmailAuth.initialize,
(owner, accountSalt, address(this))
)
)
)
)
);
}
Deploy the EmailAuth
proxy contract when needed:
function deployEmailAuthProxy(
address owner,
bytes32 accountSalt
) internal returns (address) {
ERC1967Proxy proxy = new ERC1967Proxy{salt: accountSalt}(
emailAuthImplementation(),
abi.encodeCall(
EmailAuth.initialize,
(owner, accountSalt, address(this))
)
);
return address(proxy);
}
Command Template Definition
Implement the commandTemplates
function to specify command structures:
function commandTemplates() public pure returns (string[][] memory) {
string[][] memory templates = new string[][](NUMBER_OF_TEMPLATES);
// Template 0: "Approve transaction {uint}"
templates[0] = new string[](3);
templates[0][0] = "Approve";
templates[0][1] = "transaction";
templates[0][2] = "{uint}";
// Template 1: "Set threshold to {uint} percent"
templates[1] = new string[](5);
templates[1][0] = "Set";
templates[1][1] = "threshold";
templates[1][2] = "to";
templates[1][3] = "{uint}";
templates[1][4] = "percent";
// Add more templates as needed...
return templates;
}
Create a function to generate unique IDs for each template:
function computeTemplateId(uint templateIdx) public pure returns (uint) {
return uint256(keccak256(abi.encode("IDENTIFIER", templateIdx)));
}
Command Execution Implementation
Process authenticated email commands in your contract:
function executeEmailCommand(
EmailAuthMsg memory emailAuthMsg,
address owner,
uint templateIdx
) public {
// Compute addresses and template IDs
address emailAuthAddr = computeEmailAuthAddress(
owner,
emailAuthMsg.proof.accountSalt
);
uint templateId = computeTemplateId(templateIdx);
require(templateId == emailAuthMsg.templateId, "Invalid template ID");
EmailAuth emailAuth;
if (emailAuthAddr.code.length == 0) {
// Deploy and initialize EmailAuth proxy
require(
emailAuthMsg.proof.isCodeExist == true,
"isCodeExist must be true for the first email"
);
address proxyAddress = deployEmailAuthProxy(
owner,
emailAuthMsg.proof.accountSalt
);
require(
proxyAddress == emailAuthAddr,
"Proxy address mismatch"
);
emailAuth = EmailAuth(proxyAddress);
emailAuth.initDKIMRegistry(dkim());
emailAuth.initVerifier(verifier());
// Insert command templates
string[][] memory templates = commandTemplates();
for (uint idx = 0; idx < templates.length; idx++) {
emailAuth.insertCommandTemplate(
computeTemplateId(idx),
templates[idx]
);
}
} else {
// Use existing EmailAuth contract
emailAuth = EmailAuth(payable(address(emailAuthAddr)));
require(
emailAuth.controller() == address(this),
"Invalid controller"
);
}
// Authenticate the email
emailAuth.authEmail(emailAuthMsg);
// Execute the action based on the command
_executeAction(emailAuthAddr, emailAuthMsg.commandParams, templateIdx);
}
Define actions corresponding to each command template:
function _executeAction(
address emailAuthAddr,
bytes[] memory commandParams,
uint templateIdx
) private {
if (templateIdx == 0) {
uint transactionId = abi.decode(commandParams[0], (uint));
_approveTransaction(transactionId);
} else if (templateIdx == 1) {
uint threshold = abi.decode(commandParams[0], (uint));
_setThreshold(threshold);
} else {
revert("Invalid template index");
}
}
Deployment
Deploy the following contracts and note their addresses:
- Verifier Contract
- DKIM Registry Contract
- EmailAuth Implementation Contract
Deploy your main contract with the addresses of the deployed contracts:
constructor(
address _verifierAddr,
address _dkimAddr,
address _emailAuthImplementationAddr
) {
verifierAddr = _verifierAddr;
dkimAddr = _dkimAddr;
emailAuthImplementationAddr = _emailAuthImplementationAddr;
}