Skip to content

Commit

Permalink
refactor: support for custom maxDecryptionDelay
Browse files Browse the repository at this point in the history
  • Loading branch information
PacificYield committed Dec 20, 2024
1 parent 269cedf commit ead6d77
Show file tree
Hide file tree
Showing 8 changed files with 55 additions and 23 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -6,5 +6,5 @@ import { MockZamaFHEVMConfig } from "fhevm/config/ZamaFHEVMConfig.sol";
import { MockZamaGatewayConfig } from "fhevm/config/ZamaGatewayConfig.sol";

contract TestConfidentialERC20Wrapped is MockZamaFHEVMConfig, MockZamaGatewayConfig, ConfidentialERC20Wrapped {
constructor(address erc20_) ConfidentialERC20Wrapped(erc20_) {}
constructor(address erc20_, uint256 maxDecryptionDelay_) ConfidentialERC20Wrapped(erc20_, maxDecryptionDelay_) {}
}
5 changes: 3 additions & 2 deletions contracts/test/token/ERC20/TestConfidentialWETH.sol
Original file line number Diff line number Diff line change
Expand Up @@ -5,5 +5,6 @@ import { ConfidentialWETH } from "../../../token/ERC20/ConfidentialWETH.sol";
import { MockZamaFHEVMConfig } from "fhevm/config/ZamaFHEVMConfig.sol";
import { MockZamaGatewayConfig } from "fhevm/config/ZamaGatewayConfig.sol";

/* solhint-disable no-empty-blocks*/
contract TestConfidentialWETH is MockZamaFHEVMConfig, MockZamaGatewayConfig, ConfidentialWETH {}
contract TestConfidentialWETH is MockZamaFHEVMConfig, MockZamaGatewayConfig, ConfidentialWETH {
constructor(uint256 maxDecryptionDelay_) ConfidentialWETH(maxDecryptionDelay_) {}
}
32 changes: 23 additions & 9 deletions contracts/token/ERC20/ConfidentialERC20Wrapped.sol
Original file line number Diff line number Diff line change
Expand Up @@ -21,6 +21,9 @@ import { ConfidentialERC20 } from "./ConfidentialERC20.sol";
abstract contract ConfidentialERC20Wrapped is ConfidentialERC20, IConfidentialERC20Wrapped, GatewayCaller {
using SafeERC20 for IERC20Metadata;

/// @notice Returned if the maximum decryption delay is higher than 1 day.
error MaxDecryptionDelayTooHigh();

/// @notice ERC20 token that is wrapped.
IERC20Metadata public immutable ERC20_TOKEN;

Expand All @@ -31,22 +34,33 @@ abstract contract ConfidentialERC20Wrapped is ConfidentialERC20, IConfidentialER
mapping(uint256 requestId => UnwrapRequest unwrapRequest) public unwrapRequests;

/**
* @notice Deposit/withdraw ERC20 tokens using encrypted ERC20 tokens.
* @param erc20_ Address of the ERC20 token to wrap/unwrap.
* @dev The name/symbol are autogenerated.
* For instance,
* "Wrapped Ether" --> "Encrypted Wrapped Ether"
* "WETH" --> "eWETH".
* @notice Deposit/withdraw ERC20 tokens using confidential ERC20 tokens.
* @param erc20_ Address of the ERC20 token to wrap/unwrap.
* @dev The name/symbol are autogenerated.
* For instance,
* "Wrapped Ether" --> "Confidential Wrapped Ether"
* "WETH" --> "WETHc".
* @param maxDecryptionDelay_ Maximum delay for the Gateway to decrypt.
* @dev Do not use a small value in production to avoid security issues if the response
* cannot be processed because the block time is higher than the delay.
* The current implementation expects the Gateway to always return a decrypted
* value within the delay specified, as long as it is sufficient enough.
*/
constructor(
address erc20_
address erc20_,
uint256 maxDecryptionDelay_
)
ConfidentialERC20(
string(abi.encodePacked("Encrypted ", IERC20Metadata(erc20_).name())),
string(abi.encodePacked("e", IERC20Metadata(erc20_).symbol()))
string(abi.encodePacked("Confidential ", IERC20Metadata(erc20_).name())),
string(abi.encodePacked(IERC20Metadata(erc20_).symbol(), "c"))
)
{
ERC20_TOKEN = IERC20Metadata(erc20_);

/// @dev The maximum delay is set to 1 day.
if (maxDecryptionDelay_ > 1 days) {
revert MaxDecryptionDelayTooHigh();
}
}

/**
Expand Down
23 changes: 18 additions & 5 deletions contracts/token/ERC20/ConfidentialWETH.sol
Original file line number Diff line number Diff line change
Expand Up @@ -17,19 +17,32 @@ abstract contract ConfidentialWETH is ConfidentialERC20, IConfidentialERC20Wrapp
/// @notice Returned if ETH transfer fails.
error ETHTransferFail();

/// @notice Returned if the maximum decryption delay is higher than 1 day.
error MaxDecryptionDelayTooHigh();

/// @notice Tracks whether the account can move funds.
mapping(address account => bool isRestricted) public isAccountRestricted;

/// @notice Tracks the unwrap request to a unique request id.
mapping(uint256 requestId => UnwrapRequest unwrapRequest) public unwrapRequests;

/**
* @notice Deposit/withdraw ethers (or native tokens).
* @dev The name/symbol are autogenerated.
* @notice Deposit/withdraw ethers (or native tokens).
* @dev The name/symbol are autogenerated.
* @param maxDecryptionDelay_ Maximum delay for the Gateway to decrypt.
* @dev Do not use a small value in production to avoid security issues if the response
* cannot be processed because the block time is higher than the delay.
* The current implementation expects the Gateway to always return a decrypted
* value within the delay specified, as long as it is sufficient enough.
*/
constructor()
ConfidentialERC20(string(abi.encodePacked("Encrypted Wrapped Ether")), string(abi.encodePacked("eWETH")))
{}
constructor(
uint256 maxDecryptionDelay_
) ConfidentialERC20(string(abi.encodePacked("Confidential Wrapped Ether")), string(abi.encodePacked("WETHc"))) {
/// @dev The maximum delay is set to 1 day.
if (maxDecryptionDelay_ > 1 days) {
revert MaxDecryptionDelayTooHigh();
}
}

/**
* @notice Fallback function calls wrap().
Expand Down
4 changes: 3 additions & 1 deletion test/confidentialERC20/ConfidentialERC20Wrapped.fixture.ts
Original file line number Diff line number Diff line change
Expand Up @@ -9,6 +9,8 @@ export async function deployERC20AndConfidentialERC20WrappedFixture(
symbol: string,
decimals: number,
): Promise<[ERC20Mintable, TestConfidentialERC20Wrapped]> {
// @dev We use 5 minutes for the maximum decryption delay (from the Gateway).
const maxDecryptionDelay = 60 * 5;
const contractFactoryERC20Mintable = await ethers.getContractFactory("ERC20Mintable");
const contractERC20 = await contractFactoryERC20Mintable
.connect(signers.alice)
Expand All @@ -18,7 +20,7 @@ export async function deployERC20AndConfidentialERC20WrappedFixture(
const contractFactory = await ethers.getContractFactory("TestConfidentialERC20Wrapped");
const contractConfidentialERC20Wrapped = await contractFactory
.connect(signers.alice)
.deploy(contractERC20.getAddress());
.deploy(contractERC20.getAddress(), maxDecryptionDelay);
await contractConfidentialERC20Wrapped.waitForDeployment();

return [contractERC20, contractConfidentialERC20Wrapped];
Expand Down
4 changes: 2 additions & 2 deletions test/confidentialERC20/ConfidentialERC20Wrapped.test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -29,8 +29,8 @@ describe("ConfidentialERC20Wrapped using ERC20 with 6 decimals", function () {
});

it("name/symbol are automatically set", async function () {
expect(await this.confidentialERC20Wrapped.name()).to.eq("Encrypted Naraggara");
expect(await this.confidentialERC20Wrapped.symbol()).to.eq("eNARA");
expect(await this.confidentialERC20Wrapped.name()).to.eq("Confidential Naraggara");
expect(await this.confidentialERC20Wrapped.symbol()).to.eq("NARAc");
});

it("can wrap", async function () {
Expand Down
4 changes: 3 additions & 1 deletion test/confidentialERC20/ConfidentialWETH.fixture.ts
Original file line number Diff line number Diff line change
Expand Up @@ -4,8 +4,10 @@ import type { TestConfidentialWETH } from "../../types";
import { Signers } from "../signers";

export async function deployConfidentialWETHFixture(signers: Signers): Promise<TestConfidentialWETH> {
// @dev We use 5 minutes for the maximum decryption delay (from the Gateway).
const maxDecryptionDelay = 60 * 5;
const contractFactory = await ethers.getContractFactory("TestConfidentialWETH");
const confidentialWETH = await contractFactory.connect(signers.alice).deploy();
const confidentialWETH = await contractFactory.connect(signers.alice).deploy(maxDecryptionDelay);
await confidentialWETH.waitForDeployment();

return confidentialWETH;
Expand Down
4 changes: 2 additions & 2 deletions test/confidentialERC20/ConfidentialWETH.test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -22,8 +22,8 @@ describe("ConfidentialWETH", function () {
});

it("name/symbol are automatically set, totalSupply = 0", async function () {
expect(await this.confidentialWETH.name()).to.eq("Encrypted Wrapped Ether");
expect(await this.confidentialWETH.symbol()).to.eq("eWETH");
expect(await this.confidentialWETH.name()).to.eq("Confidential Wrapped Ether");
expect(await this.confidentialWETH.symbol()).to.eq("WETHc");
expect(await this.confidentialWETH.totalSupply()).to.eq("0");
});

Expand Down

0 comments on commit ead6d77

Please sign in to comment.