diff --git a/.github/workflows/integration-test.yml b/.github/workflows/integration-test.yml index f278142d..f58da075 100644 --- a/.github/workflows/integration-test.yml +++ b/.github/workflows/integration-test.yml @@ -163,8 +163,10 @@ jobs: export ZKSYNC_CHAIN_ID=${{vars.ZKSYNC_CHAIN_ID}} export STARKNET_CLAIM_PAYMENT_SELECTOR=${{vars.STARKNET_CLAIM_PAYMENT_SELECTOR}} export STARKNET_CLAIM_PAYMENT_BATCH_SELECTOR=${{vars.STARKNET_CLAIM_PAYMENT_BATCH_SELECTOR}} + export STARKNET_CLAIM_PAYMENT_ERC20_SELECTOR=${{vars.STARKNET_CLAIM_PAYMENT_ERC20_SELECTOR}} export ZKSYNC_CLAIM_PAYMENT_SELECTOR=${{vars.ZKSYNC_CLAIM_PAYMENT_SELECTOR}} export ZKSYNC_CLAIM_PAYMENT_BATCH_SELECTOR=${{vars.ZKSYNC_CLAIM_PAYMENT_BATCH_SELECTOR}} + export ZKSYNC_CLAIM_PAYMENT_ERC20_SELECTOR=${{vars.ZKSYNC_CLAIM_PAYMENT_ERC20_SELECTOR}} export SKIP_VERIFY=true . ./contracts/ethereum/deploy.sh diff --git a/Makefile b/Makefile index 23038f0c..8289efbc 100644 --- a/Makefile +++ b/Makefile @@ -42,6 +42,9 @@ ethereum-upgrade: ethereum-build ethereum-set-escrow: @. ./contracts/ethereum/.env && . ./contracts/ethereum/set_starknet_escrow.sh +ethereum-deploy-erc20: ethereum-build + @. ./contracts/ethereum/.env && . ./contracts/ethereum/deploy_erc20.sh + ### STARKNET ### @@ -75,6 +78,9 @@ starknet-pause: starknet-unpause: @. ./contracts/starknet/.env && . ./contracts/starknet/scripts/change_pause_state.sh unpause +starknet-deploy-erc20: starknet-build + @. ./contracts/starknet/.env && . ./contracts/starknet/scripts/deploy_erc20.sh + ### ZKSYNC ### @@ -113,6 +119,10 @@ zksync-test-integration: . ./contracts/zksync/test/transfer.sh && \ . ./contracts/zksync/test/claim_payment.sh +zksync-deploy-erc20: + @make zksync-build && \ + . ./contracts/zksync/deploy_erc20.sh + # zksync-upgrade: WIP @@ -141,15 +151,15 @@ ethereum-and-starknet-deploy: deploy-all: @. ./contracts/ethereum/.env && . ./contracts/starknet/.env && . ./contracts/zksync/.env && \ make ethereum-build && \ - . ./contracts/ethereum/deploy.sh && \ make starknet-build && \ + make zksync-build && \ + . ./contracts/ethereum/deploy.sh && \ . ./contracts/starknet/scripts/deploy.sh && \ . ./contracts/ethereum/set_starknet_escrow.sh && \ - . ./contracts/utils/display_info.sh && \ - make zksync-build && \ . ./contracts/zksync/deploy.sh && \ - . ./contracts/ethereum/set_zksync_escrow.sh - + . ./contracts/ethereum/set_zksync_escrow.sh && \ + . ./contracts/utils/display_info.sh + test: make starknet-test make ethereum-test diff --git a/contracts/ethereum/.env.example b/contracts/ethereum/.env.example index fb4e644c..aff8645b 100644 --- a/contracts/ethereum/.env.example +++ b/contracts/ethereum/.env.example @@ -10,8 +10,10 @@ ZKSYNC_DIAMOND_PROXY_ADDRESS=<0x9A6DE0f62Aa270A8bCB1e2610078650D539B1Ef9> # Sepo STARKNET_CLAIM_PAYMENT_SELECTOR=<0x03636c566f6409560d55d5f6d1eb4ee163b096b4698c503e69e210be79de2afa> #hex value of starknet's claim_payment selector STARKNET_CLAIM_PAYMENT_BATCH_SELECTOR=<0x0354a01e49fe07e43306a97ed84dbd5de8238c7d8ff616caa3444630cfc559e6> #hex value of starknet's claim_payment_batch selector +STARKNET_CLAIM_PAYMENT_ERC20_SELECTOR=<0x0091ec2842317cd03601c3f46ee8ebc9b1dc6cdbc96cb7b0873cc6360538d754> #hex value of starknet's claim_payment_erc20 selector ZKSYNC_CLAIM_PAYMENT_SELECTOR=<0xa5168739> #hex value of ZKSync's claim_payment selctor ZKSYNC_CLAIM_PAYMENT_BATCH_SELECTOR=<0x156be1ae> #hex value of ZKSync's claim_payment_batch selctor +ZKSYNC_CLAIM_PAYMENT_ERC20_SELECTOR=<0xb9738dd6> #hex value of ZKSync's claim_payment_erc20 selctor STARKNET_CHAIN_ID=<0x534e5f5345504f4c4941|0x534e5f4d41494e> #Sepolia | Mainnet ZKSYNC_CHAIN_ID=<300|324> # Sepolia | Mainnet diff --git a/contracts/ethereum/.env.test b/contracts/ethereum/.env.test index 42e5dd41..c0c649a1 100644 --- a/contracts/ethereum/.env.test +++ b/contracts/ethereum/.env.test @@ -6,7 +6,10 @@ MM_ETHEREUM_ADDRESS=0xf39Fd6e51aad88F6F4ce6aB8827279cffFb92266 SKIP_VERIFY=true STARKNET_CLAIM_PAYMENT_SELECTOR=0x03636c566f6409560d55d5f6d1eb4ee163b096b4698c503e69e210be79de2afa STARKNET_CLAIM_PAYMENT_BATCH_SELECTOR=0x0354a01e49fe07e43306a97ed84dbd5de8238c7d8ff616caa3444630cfc559e6 +STARKNET_CLAIM_PAYMENT_ERC20_SELECTOR=0x0091ec2842317cd03601c3f46ee8ebc9b1dc6cdbc96cb7b0873cc6360538d754 ZKSYNC_CLAIM_PAYMENT_SELECTOR=0xa5168739 ZKSYNC_CLAIM_PAYMENT_BATCH_SELECTOR=0x156be1ae +ZKSYNC_CLAIM_PAYMENT_ERC20_SELECTOR=0xb9738dd6 + STARKNET_CHAIN_ID=0x534e5f5345504f4c4941 ZKSYNC_CHAIN_ID=300 diff --git a/contracts/ethereum/deploy_erc20.sh b/contracts/ethereum/deploy_erc20.sh new file mode 100755 index 00000000..a2add319 --- /dev/null +++ b/contracts/ethereum/deploy_erc20.sh @@ -0,0 +1,25 @@ +#!/bin/bash +. contracts/utils/colors.sh #for ANSI colors + +cd contracts/ethereum + +printf "${GREEN}\n=> [ETH] Deploying ERC20 ${COLOR_RESET}\n" + + +export ETHEREUM_PRIVATE_KEY=$ETHEREUM_PRIVATE_KEY + +RESULT_LOG=$(forge script ./script/Deploy_ERC20.s.sol --rpc-url $ETHEREUM_RPC --broadcast ${SKIP_VERIFY:---verify}) +# echo "$RESULT_LOG" #uncomment this line for debugging in detail + +# Getting result addresses +ERC20_ADDRESS=$(echo "$RESULT_LOG" | grep -Eo '0: address ([^\n]+)' | awk '{print $NF}') + +if [ -z "$ERC20_ADDRESS" ]; then + printf "\n${RED}ERROR:${COLOR_RESET}\n" + echo "ERC20_ADDRESS Variable is empty. Aborting execution.\n" + exit 1 +fi + +printf "${GREEN}\n=> [ETH] Deployed ERC20 address: $ERC20_ADDRESS ${COLOR_RESET}\n" + +cd ../.. #to reset working directory diff --git a/contracts/ethereum/lib/openzeppelin/contracts/token/ERC20/ERC20.sol b/contracts/ethereum/lib/openzeppelin/contracts/token/ERC20/ERC20.sol new file mode 100644 index 00000000..a7598a66 --- /dev/null +++ b/contracts/ethereum/lib/openzeppelin/contracts/token/ERC20/ERC20.sol @@ -0,0 +1,312 @@ +// SPDX-License-Identifier: MIT +// OpenZeppelin Contracts (last updated v5.0.0) (token/ERC20/ERC20.sol) + +pragma solidity ^0.8.20; + +import {IERC20} from "./IERC20.sol"; +import {IERC20Metadata} from "./extensions/IERC20Metadata.sol"; +import {Context} from "../../utils/Context.sol"; +import {IERC20Errors} from "../../interfaces/draft-IERC6093.sol"; + +/** + * @dev Implementation of the {IERC20} interface. + * + * This implementation is agnostic to the way tokens are created. This means + * that a supply mechanism has to be added in a derived contract using {_mint}. + * + * TIP: For a detailed writeup see our guide + * https://forum.openzeppelin.com/t/how-to-implement-erc20-supply-mechanisms/226[How + * to implement supply mechanisms]. + * + * The default value of {decimals} is 18. To change this, you should override + * this function so it returns a different value. + * + * We have followed general OpenZeppelin Contracts guidelines: functions revert + * instead returning `false` on failure. This behavior is nonetheless + * conventional and does not conflict with the expectations of ERC-20 + * applications. + */ +abstract contract ERC20 is Context, IERC20, IERC20Metadata, IERC20Errors { + mapping(address account => uint256) private _balances; + + mapping(address account => mapping(address spender => uint256)) private _allowances; + + uint256 private _totalSupply; + + string private _name; + string private _symbol; + + /** + * @dev Sets the values for {name} and {symbol}. + * + * All two of these values are immutable: they can only be set once during + * construction. + */ + constructor(string memory name_, string memory symbol_) { + _name = name_; + _symbol = symbol_; + } + + /** + * @dev Returns the name of the token. + */ + function name() public view virtual returns (string memory) { + return _name; + } + + /** + * @dev Returns the symbol of the token, usually a shorter version of the + * name. + */ + function symbol() public view virtual returns (string memory) { + return _symbol; + } + + /** + * @dev Returns the number of decimals used to get its user representation. + * For example, if `decimals` equals `2`, a balance of `505` tokens should + * be displayed to a user as `5.05` (`505 / 10 ** 2`). + * + * Tokens usually opt for a value of 18, imitating the relationship between + * Ether and Wei. This is the default value returned by this function, unless + * it's overridden. + * + * NOTE: This information is only used for _display_ purposes: it in + * no way affects any of the arithmetic of the contract, including + * {IERC20-balanceOf} and {IERC20-transfer}. + */ + function decimals() public view virtual returns (uint8) { + return 18; + } + + /** + * @dev See {IERC20-totalSupply}. + */ + function totalSupply() public view virtual returns (uint256) { + return _totalSupply; + } + + /** + * @dev See {IERC20-balanceOf}. + */ + function balanceOf(address account) public view virtual returns (uint256) { + return _balances[account]; + } + + /** + * @dev See {IERC20-transfer}. + * + * Requirements: + * + * - `to` cannot be the zero address. + * - the caller must have a balance of at least `value`. + */ + function transfer(address to, uint256 value) public virtual returns (bool) { + address owner = _msgSender(); + _transfer(owner, to, value); + return true; + } + + /** + * @dev See {IERC20-allowance}. + */ + function allowance(address owner, address spender) public view virtual returns (uint256) { + return _allowances[owner][spender]; + } + + /** + * @dev See {IERC20-approve}. + * + * NOTE: If `value` is the maximum `uint256`, the allowance is not updated on + * `transferFrom`. This is semantically equivalent to an infinite approval. + * + * Requirements: + * + * - `spender` cannot be the zero address. + */ + function approve(address spender, uint256 value) public virtual returns (bool) { + address owner = _msgSender(); + _approve(owner, spender, value); + return true; + } + + /** + * @dev See {IERC20-transferFrom}. + * + * Skips emitting an {Approval} event indicating an allowance update. This is not + * required by the ERC. See {xref-ERC20-_approve-address-address-uint256-bool-}[_approve]. + * + * NOTE: Does not update the allowance if the current allowance + * is the maximum `uint256`. + * + * Requirements: + * + * - `from` and `to` cannot be the zero address. + * - `from` must have a balance of at least `value`. + * - the caller must have allowance for ``from``'s tokens of at least + * `value`. + */ + function transferFrom(address from, address to, uint256 value) public virtual returns (bool) { + address spender = _msgSender(); + _spendAllowance(from, spender, value); + _transfer(from, to, value); + return true; + } + + /** + * @dev Moves a `value` amount of tokens from `from` to `to`. + * + * This internal function is equivalent to {transfer}, and can be used to + * e.g. implement automatic token fees, slashing mechanisms, etc. + * + * Emits a {Transfer} event. + * + * NOTE: This function is not virtual, {_update} should be overridden instead. + */ + function _transfer(address from, address to, uint256 value) internal { + if (from == address(0)) { + revert ERC20InvalidSender(address(0)); + } + if (to == address(0)) { + revert ERC20InvalidReceiver(address(0)); + } + _update(from, to, value); + } + + /** + * @dev Transfers a `value` amount of tokens from `from` to `to`, or alternatively mints (or burns) if `from` + * (or `to`) is the zero address. All customizations to transfers, mints, and burns should be done by overriding + * this function. + * + * Emits a {Transfer} event. + */ + function _update(address from, address to, uint256 value) internal virtual { + if (from == address(0)) { + // Overflow check required: The rest of the code assumes that totalSupply never overflows + _totalSupply += value; + } else { + uint256 fromBalance = _balances[from]; + if (fromBalance < value) { + revert ERC20InsufficientBalance(from, fromBalance, value); + } + unchecked { + // Overflow not possible: value <= fromBalance <= totalSupply. + _balances[from] = fromBalance - value; + } + } + + if (to == address(0)) { + unchecked { + // Overflow not possible: value <= totalSupply or value <= fromBalance <= totalSupply. + _totalSupply -= value; + } + } else { + unchecked { + // Overflow not possible: balance + value is at most totalSupply, which we know fits into a uint256. + _balances[to] += value; + } + } + + emit Transfer(from, to, value); + } + + /** + * @dev Creates a `value` amount of tokens and assigns them to `account`, by transferring it from address(0). + * Relies on the `_update` mechanism + * + * Emits a {Transfer} event with `from` set to the zero address. + * + * NOTE: This function is not virtual, {_update} should be overridden instead. + */ + function _mint(address account, uint256 value) internal { + if (account == address(0)) { + revert ERC20InvalidReceiver(address(0)); + } + _update(address(0), account, value); + } + + /** + * @dev Destroys a `value` amount of tokens from `account`, lowering the total supply. + * Relies on the `_update` mechanism. + * + * Emits a {Transfer} event with `to` set to the zero address. + * + * NOTE: This function is not virtual, {_update} should be overridden instead + */ + function _burn(address account, uint256 value) internal { + if (account == address(0)) { + revert ERC20InvalidSender(address(0)); + } + _update(account, address(0), value); + } + + /** + * @dev Sets `value` as the allowance of `spender` over the `owner` s tokens. + * + * This internal function is equivalent to `approve`, and can be used to + * e.g. set automatic allowances for certain subsystems, etc. + * + * Emits an {Approval} event. + * + * Requirements: + * + * - `owner` cannot be the zero address. + * - `spender` cannot be the zero address. + * + * Overrides to this logic should be done to the variant with an additional `bool emitEvent` argument. + */ + function _approve(address owner, address spender, uint256 value) internal { + _approve(owner, spender, value, true); + } + + /** + * @dev Variant of {_approve} with an optional flag to enable or disable the {Approval} event. + * + * By default (when calling {_approve}) the flag is set to true. On the other hand, approval changes made by + * `_spendAllowance` during the `transferFrom` operation set the flag to false. This saves gas by not emitting any + * `Approval` event during `transferFrom` operations. + * + * Anyone who wishes to continue emitting `Approval` events on the`transferFrom` operation can force the flag to + * true using the following override: + * + * ```solidity + * function _approve(address owner, address spender, uint256 value, bool) internal virtual override { + * super._approve(owner, spender, value, true); + * } + * ``` + * + * Requirements are the same as {_approve}. + */ + function _approve(address owner, address spender, uint256 value, bool emitEvent) internal virtual { + if (owner == address(0)) { + revert ERC20InvalidApprover(address(0)); + } + if (spender == address(0)) { + revert ERC20InvalidSpender(address(0)); + } + _allowances[owner][spender] = value; + if (emitEvent) { + emit Approval(owner, spender, value); + } + } + + /** + * @dev Updates `owner` s allowance for `spender` based on spent `value`. + * + * Does not update the allowance value in case of infinite allowance. + * Revert if not enough allowance is available. + * + * Does not emit an {Approval} event. + */ + function _spendAllowance(address owner, address spender, uint256 value) internal virtual { + uint256 currentAllowance = allowance(owner, spender); + if (currentAllowance != type(uint256).max) { + if (currentAllowance < value) { + revert ERC20InsufficientAllowance(spender, currentAllowance, value); + } + unchecked { + _approve(owner, spender, currentAllowance - value, false); + } + } + } +} \ No newline at end of file diff --git a/contracts/ethereum/script/Deploy.s.sol b/contracts/ethereum/script/Deploy.s.sol index 4aa4958e..494f1b3a 100644 --- a/contracts/ethereum/script/Deploy.s.sol +++ b/contracts/ethereum/script/Deploy.s.sol @@ -13,10 +13,12 @@ contract Deploy is Script { address STARKNET_MESSAGING_ADDRESS = vm.envAddress("STARKNET_MESSAGING_ADDRESS"); uint256 STARKNET_CLAIM_PAYMENT_SELECTOR = vm.envUint("STARKNET_CLAIM_PAYMENT_SELECTOR"); uint256 STARKNET_CLAIM_PAYMENT_BATCH_SELECTOR = vm.envUint("STARKNET_CLAIM_PAYMENT_BATCH_SELECTOR"); + uint256 STARKNET_CLAIM_PAYMENT_ERC20_SELECTOR = vm.envUint("STARKNET_CLAIM_PAYMENT_ERC20_SELECTOR"); address MM_ETHEREUM_WALLET_ADDRESS = vm.envAddress("MM_ETHEREUM_WALLET_ADDRESS"); address ZKSYNC_DIAMOND_PROXY_ADDRESS = vm.envAddress("ZKSYNC_DIAMOND_PROXY_ADDRESS"); bytes4 ZKSYNC_CLAIM_PAYMENT_SELECTOR = bytes4(vm.envBytes("ZKSYNC_CLAIM_PAYMENT_SELECTOR")); bytes4 ZKSYNC_CLAIM_PAYMENT_BATCH_SELECTOR = bytes4(vm.envBytes("ZKSYNC_CLAIM_PAYMENT_BATCH_SELECTOR")); + bytes4 ZKSYNC_CLAIM_PAYMENT_ERC20_SELECTOR = bytes4(vm.envBytes("ZKSYNC_CLAIM_PAYMENT_ERC20_SELECTOR")); uint128 STARKNET_CHAIN_ID = uint128(vm.envUint("STARKNET_CHAIN_ID")); uint128 ZKSYNC_CHAIN_ID = uint128(vm.envUint("ZKSYNC_CHAIN_ID")); @@ -26,11 +28,13 @@ contract Deploy is Script { PaymentRegistry(address(proxy)).initialize( STARKNET_MESSAGING_ADDRESS, STARKNET_CLAIM_PAYMENT_SELECTOR, - STARKNET_CLAIM_PAYMENT_BATCH_SELECTOR, + STARKNET_CLAIM_PAYMENT_BATCH_SELECTOR, + STARKNET_CLAIM_PAYMENT_ERC20_SELECTOR, MM_ETHEREUM_WALLET_ADDRESS, ZKSYNC_DIAMOND_PROXY_ADDRESS, ZKSYNC_CLAIM_PAYMENT_SELECTOR, ZKSYNC_CLAIM_PAYMENT_BATCH_SELECTOR, + ZKSYNC_CLAIM_PAYMENT_ERC20_SELECTOR, STARKNET_CHAIN_ID, ZKSYNC_CHAIN_ID ); diff --git a/contracts/ethereum/script/Deploy_ERC20.s.sol b/contracts/ethereum/script/Deploy_ERC20.s.sol new file mode 100644 index 00000000..4eb69ccf --- /dev/null +++ b/contracts/ethereum/script/Deploy_ERC20.s.sol @@ -0,0 +1,18 @@ +// SPDX-License-Identifier: Apache-2.0 +pragma solidity ^0.8.20; + +import {Script} from "forge-std/Script.sol"; +import {UriCoin} from "../src/ERC20_L1.sol"; + +contract Deploy is Script { + function run() external returns (address) { + uint256 deployerPrivateKey = vm.envUint("ETHEREUM_PRIVATE_KEY"); + vm.startBroadcast(deployerPrivateKey); + + UriCoin uriCoin = new UriCoin(); + + vm.stopBroadcast(); + + return (address(uriCoin)); + } +} diff --git a/contracts/ethereum/src/ERC20_L1.sol b/contracts/ethereum/src/ERC20_L1.sol new file mode 100644 index 00000000..a0673684 --- /dev/null +++ b/contracts/ethereum/src/ERC20_L1.sol @@ -0,0 +1,10 @@ +// SPDX-License-Identifier: Unlicense +pragma solidity ^0.8.0; + +import "@openzeppelin/contracts/token/ERC20/ERC20.sol"; + +contract UriCoin is ERC20 { + constructor() ERC20("UriCoin", "Uri") { + _mint(0xda963fA72caC2A3aC01c642062fba3C099993D56, 1000000); //TODO unhardcode recipient + initial_supply + } +} diff --git a/contracts/ethereum/src/PaymentRegistry.sol b/contracts/ethereum/src/PaymentRegistry.sol index 1f7302e1..718456b0 100644 --- a/contracts/ethereum/src/PaymentRegistry.sol +++ b/contracts/ethereum/src/PaymentRegistry.sol @@ -6,19 +6,30 @@ import {OwnableUpgradeable} from "@openzeppelin/contracts-upgradeable/access/Own import {Initializable} from "@openzeppelin/contracts-upgradeable/proxy/utils/Initializable.sol"; import {UUPSUpgradeable} from "@openzeppelin/contracts-upgradeable/proxy/utils/UUPSUpgradeable.sol"; import {IZkSync} from "@matterlabs/interfaces/IZkSync.sol"; +import "@openzeppelin/contracts/token/ERC20/utils/SafeERC20.sol"; + contract PaymentRegistry is Initializable, OwnableUpgradeable, UUPSUpgradeable { + using SafeERC20 for IERC20; + event Transfer(uint256 indexed orderId, address srcAddress, address destAddress, uint256 amount, uint128 chainId); + event TransferERC20(uint256 indexed orderId, address srcAddress, address destAddress, uint256 amount, uint128 chainId, address erc20Address); + event ClaimPayment(uint256 indexed orderId, address destAddress, uint256 amount, uint128 chainId); + event ClaimPaymentERC20(uint256 indexed orderId, address destAddress, uint256 amount, uint128 chainId, address erc20Address); event ClaimPaymentBatch(uint256[] orderIds, address[] destAddresses, uint256[] amounts, uint128 chainId); event ModifiedZKSyncEscrowAddress(address newEscrowAddress); event ModifiedStarknetEscrowAddress(uint256 newEscrowAddress); + event ModifiedStarknetClaimPaymentSelector(uint256 newEscrowClaimPaymentSelector); event ModifiedStarknetClaimPaymentBatchSelector(uint256 newEscrowClaimPaymentBatchSelector); + event ModifiedStarknetClaimPaymentERC20Selector(uint256 newEscrowClaimPaymentERC20Selector); + event ModifiedZKSyncClaimPaymentSelector(bytes4 newZKSyncEscrowClaimPaymentSelector); event ModifiedZKSyncClaimPaymentBatchSelector(bytes4 newZKSyncEscrowClaimPaymentBatchSelector); + event ModifiedZKSyncClaimPaymentERC20Selector(bytes4 newZKSyncEscrowClaimPaymentERC20Selector); mapping(bytes32 => bool) public transfers; address public marketMaker; @@ -26,8 +37,10 @@ contract PaymentRegistry is Initializable, OwnableUpgradeable, UUPSUpgradeable { address public ZKSyncEscrowAddress; uint256 public StarknetEscrowClaimPaymentSelector; uint256 public StarknetEscrowClaimPaymentBatchSelector; + uint256 public StarknetEscrowClaimPaymentERC20Selector; bytes4 public ZKSyncEscrowClaimPaymentSelector; bytes4 public ZKSyncEscrowClaimPaymentBatchSelector; + bytes4 public ZKSyncEscrowClaimPaymentERC20Selector; IZkSync private _ZKSyncDiamondProxy; IStarknetMessaging private _snMessaging; @@ -45,10 +58,12 @@ contract PaymentRegistry is Initializable, OwnableUpgradeable, UUPSUpgradeable { address snMessaging, uint256 StarknetEscrowClaimPaymentSelector_, uint256 StarknetEscrowClaimPaymentBatchSelector_, + uint256 StarknetEscrowClaimPaymentERC20Selector_, address marketMaker_, address ZKSyncDiamondProxyAddress, bytes4 ZKSyncEscrowClaimPaymentSelector_, bytes4 ZKSyncEscrowClaimPaymentBatchSelector_, + bytes4 ZKSyncEscrowClaimPaymentERC20Selector_, uint128 StarknetChainId_, uint128 ZKSyncChainId_) public initializer { __Ownable_init(msg.sender); @@ -59,14 +74,106 @@ contract PaymentRegistry is Initializable, OwnableUpgradeable, UUPSUpgradeable { StarknetEscrowClaimPaymentSelector = StarknetEscrowClaimPaymentSelector_; StarknetEscrowClaimPaymentBatchSelector = StarknetEscrowClaimPaymentBatchSelector_; + StarknetEscrowClaimPaymentERC20Selector = StarknetEscrowClaimPaymentERC20Selector_; ZKSyncEscrowClaimPaymentSelector = ZKSyncEscrowClaimPaymentSelector_; ZKSyncEscrowClaimPaymentBatchSelector = ZKSyncEscrowClaimPaymentBatchSelector_; + ZKSyncEscrowClaimPaymentERC20Selector = ZKSyncEscrowClaimPaymentERC20Selector_; StarknetChainId = StarknetChainId_; ZKSyncChainId = ZKSyncChainId_; marketMaker = marketMaker_; } + + function transferERC20(uint256 orderId, address destAddress, uint128 chainId, address l1_erc20_address, uint256 amount) external onlyOwnerOrMM { + // Decide if MM fees are paid in ERC20 or in ETH. + // // If paid in ETH: + // // It is easy for MM to calculate how much fee is desirable for him to bridge the tokens + // // But An extra tx is needed containing this gas. + // // it is more expensive + + // // If paid in ERC20: + // // MM can get paid in a random coin, price may vary, more difficult for MM to determine how much ERC20 tokens are necesarry. + // // But MM only subscribes to desired ERC20. For example if he only bridges USDT, he may be willing to take USDT as fee for bridge. + // // No extra tx is needed to pay this gas fee, MM will simply transfer less ERC20 tokens than what he recieved. + // // It is cheaper + + // // Allowance: + // // This unlimited allowance should be set in a separate function. MM will allow to brdige x or y ERC20. + // // // we could even find new uses for this allowance. PaymentRegistry could be more intertwined with MM. Maybe automatically doing transfers in its name. + + require(amount > 0, "Invalid amount, should be higher than 0."); + + // these 2 checks are made and reverted accordingly by SafeTransfer + // but if made now, if reverted, user doesnt spend the gas of calculating keccak + // I think appropriate users should not pay for shitty users's mustakes + // require(IERC20(l1_erc20_address).balanceOf(msg.sender) >= amount, "MM has insufficient balance"); + // require(IERC20(l1_erc20_address).allowance(msg.sender, address(this)) >= amount, "PaymentRegistry has insufficient allowance"); + //TODO check if there is a way to increment allowance directly from here, i think there is not. + + bytes32 index = keccak256(abi.encodePacked(orderId, destAddress, amount, chainId, l1_erc20_address)); //added erc20Address + + require(transfers[index] == false, "Transfer already processed."); + transfers[index] = true; //now this transfer is in progress + + IERC20(l1_erc20_address).safeTransferFrom(msg.sender, destAddress, amount); //this reverts if failed + emit TransferERC20(orderId, msg.sender, destAddress, amount, chainId, l1_erc20_address); + } + + function claimPaymentZKSyncERC20( + uint256 orderId, + address destAddress, + uint256 amount, + uint256 gasLimit, + uint256 gasPerPubdataByteLimit, + address l1_erc20_address + ) external payable onlyOwnerOrMM { + _verifyTransferExistsZKSyncERC20(orderId, destAddress, amount, l1_erc20_address); + + bytes memory messageToL2 = abi.encodeWithSelector( + ZKSyncEscrowClaimPaymentERC20Selector, + orderId, + destAddress, + amount, + l1_erc20_address + ); + + _ZKSyncDiamondProxy.requestL2Transaction{value: msg.value}( + ZKSyncEscrowAddress, //L2 contract called + 0, //msg.value + messageToL2, //msg.calldata + gasLimit, + gasPerPubdataByteLimit, + new bytes[](0), //factory dependencies + msg.sender //refund recipient + ); + + emit ClaimPaymentERC20(orderId, destAddress, amount, ZKSyncChainId, l1_erc20_address); + } + + function claimPaymentStarknetERC20( + uint256 orderId, + address destAddress, + uint256 amount, + address l1_erc20_address + ) external payable onlyOwnerOrMM { + _verifyTransferExistsStarknetERC20(orderId, destAddress, amount, l1_erc20_address); + + uint256[] memory payload = new uint256[](6); //this is not an array of u128 because sendMessageToL2 takes an array of uint256 + payload[0] = uint128(orderId); // low + payload[1] = uint128(orderId >> 128); // high + payload[2] = uint256(uint160(destAddress)); + payload[3] = uint128(amount); // low + payload[4] = uint128(amount >> 128); // high + payload[5] = uint256(uint160(l1_erc20_address)); + + _snMessaging.sendMessageToL2{value: msg.value}( + StarknetEscrowAddress, + StarknetEscrowClaimPaymentERC20Selector, //TODO set erc20 variable + payload); + + emit ClaimPaymentERC20(orderId, destAddress, amount, StarknetChainId, l1_erc20_address); + } function transfer(uint256 orderId, address destAddress, uint128 chainId) external payable onlyOwnerOrMM { require(msg.value > 0, "Invalid amount, should be higher than 0."); @@ -136,11 +243,6 @@ contract PaymentRegistry is Initializable, OwnableUpgradeable, UUPSUpgradeable { emit ClaimPaymentBatch(orderIds, destAddresses, amounts, StarknetChainId); } - function _verifyTransferExistsStarknet(uint256 orderId, address destAddress, uint256 amount) internal view { - bytes32 index = keccak256(abi.encodePacked(orderId, destAddress, amount, StarknetChainId)); - require(transfers[index] == true, "Transfer not found."); - } - function claimPaymentZKSync( uint256 orderId, address destAddress, uint256 amount, uint256 gasLimit, @@ -202,6 +304,21 @@ contract PaymentRegistry is Initializable, OwnableUpgradeable, UUPSUpgradeable { emit ClaimPaymentBatch(orderIds, destAddresses, amounts, ZKSyncChainId); } + function _verifyTransferExistsStarknetERC20(uint256 orderId, address destAddress, uint256 amount, address l1_erc20_address) internal view { + bytes32 index = keccak256(abi.encodePacked(orderId, destAddress, amount, StarknetChainId, l1_erc20_address)); + require(transfers[index] == true, "Transfer not found."); //if this is claimed twice, Escrow will know + } + + function _verifyTransferExistsStarknet(uint256 orderId, address destAddress, uint256 amount) internal view { + bytes32 index = keccak256(abi.encodePacked(orderId, destAddress, amount, StarknetChainId)); + require(transfers[index] == true, "Transfer not found."); //if this is claimed twice, Escrow will know + } + + function _verifyTransferExistsZKSyncERC20(uint256 orderId, address destAddress, uint256 amount, address l1_erc20_address) internal view { + bytes32 index = keccak256(abi.encodePacked(orderId, destAddress, amount, ZKSyncChainId, l1_erc20_address)); + require(transfers[index] == true, "Transfer not found."); //if this is claimed twice, Escrow will know + } + function _verifyTransferExistsZKSync(uint256 orderId, address destAddress, uint256 amount) internal view { bytes32 index = keccak256(abi.encodePacked(orderId, destAddress, amount, ZKSyncChainId)); require(transfers[index] == true, "Transfer not found."); //if this is claimed twice, Escrow will know @@ -227,6 +344,11 @@ contract PaymentRegistry is Initializable, OwnableUpgradeable, UUPSUpgradeable { emit ModifiedStarknetClaimPaymentBatchSelector(StarknetEscrowClaimPaymentBatchSelector); } + function setStarknetClaimPaymentERC20Selector(uint256 NewStarknetEscrowClaimPaymentERC20Selector) external onlyOwner { + StarknetEscrowClaimPaymentERC20Selector = NewStarknetEscrowClaimPaymentERC20Selector; + emit ModifiedStarknetClaimPaymentERC20Selector(StarknetEscrowClaimPaymentERC20Selector); + } + function setZKSyncEscrowClaimPaymentSelector(bytes4 NewZKSyncEscrowClaimPaymentSelector) external onlyOwner { ZKSyncEscrowClaimPaymentSelector = NewZKSyncEscrowClaimPaymentSelector; emit ModifiedZKSyncClaimPaymentSelector(ZKSyncEscrowClaimPaymentSelector); @@ -237,6 +359,11 @@ contract PaymentRegistry is Initializable, OwnableUpgradeable, UUPSUpgradeable { emit ModifiedZKSyncClaimPaymentBatchSelector(ZKSyncEscrowClaimPaymentBatchSelector); } + function setZKSyncEscrowClaimPaymentERC20Selector(bytes4 NewZKSyncEscrowClaimPaymentERC20Selector) external onlyOwner { + ZKSyncEscrowClaimPaymentERC20Selector = NewZKSyncEscrowClaimPaymentERC20Selector; + emit ModifiedZKSyncClaimPaymentERC20Selector(ZKSyncEscrowClaimPaymentERC20Selector); + } + //// MM ACL: diff --git a/contracts/ethereum/test/ACL.t.sol b/contracts/ethereum/test/ACL.t.sol index 62d4e665..3f82ae4d 100644 --- a/contracts/ethereum/test/ACL.t.sol +++ b/contracts/ethereum/test/ACL.t.sol @@ -16,9 +16,11 @@ contract TransferTest is Test { address STARKNET_MESSAGING_ADDRESS = 0xde29d060D45901Fb19ED6C6e959EB22d8626708e; uint256 STARKNET_CLAIM_PAYMENT_SELECTOR = 0x15511cc3694f64379908437d6d64458dc76d02482052bfb8a5b33a72c054c77; uint256 STARKNET_CLAIM_PAYMENT_BATCH_SELECTOR = 0x0354a01e49fe07e43306a97ed84dbd5de8238c7d8ff616caa3444630cfc559e6; + uint256 STARKNET_CLAIM_PAYMENT_ERC20_SELECTOR = 0x0091ec2842317cd03601c3f46ee8ebc9b1dc6cdbc96cb7b0873cc6360538d754; address ZKSYNC_DIAMOND_PROXY_ADDRESS = 0x2eD8eF54a16bBF721a318bd5a5C0F39Be70eaa65; bytes4 ZKSYNC_CLAIM_PAYMENT_SELECTOR = 0xa5168739; bytes4 ZKSYNC_CLAIM_PAYMENT_BATCH_SELECTOR = 0x156be1ae; + bytes4 ZKSYNC_CLAIM_PAYMENT_ERC20_SELECTOR = 0xb9738dd6; uint128 STARKNET_CHAIN_ID = 0x534e5f5345504f4c4941; uint128 ZKSYNC_CHAIN_ID = 300; @@ -29,7 +31,7 @@ contract TransferTest is Test { yab = new PaymentRegistry(); proxy = new ERC1967Proxy(address(yab), ""); yab_caller = PaymentRegistry(address(proxy)); - yab_caller.initialize(STARKNET_MESSAGING_ADDRESS, STARKNET_CLAIM_PAYMENT_SELECTOR, STARKNET_CLAIM_PAYMENT_BATCH_SELECTOR, MM_ETHEREUM_WALLET_ADDRESS, ZKSYNC_DIAMOND_PROXY_ADDRESS, ZKSYNC_CLAIM_PAYMENT_SELECTOR, ZKSYNC_CLAIM_PAYMENT_BATCH_SELECTOR, STARKNET_CHAIN_ID, ZKSYNC_CHAIN_ID); + yab_caller.initialize(STARKNET_MESSAGING_ADDRESS, STARKNET_CLAIM_PAYMENT_SELECTOR, STARKNET_CLAIM_PAYMENT_BATCH_SELECTOR, STARKNET_CLAIM_PAYMENT_ERC20_SELECTOR, MM_ETHEREUM_WALLET_ADDRESS, ZKSYNC_DIAMOND_PROXY_ADDRESS, ZKSYNC_CLAIM_PAYMENT_SELECTOR, ZKSYNC_CLAIM_PAYMENT_BATCH_SELECTOR, ZKSYNC_CLAIM_PAYMENT_ERC20_SELECTOR, STARKNET_CHAIN_ID, ZKSYNC_CHAIN_ID); vm.stopPrank(); } diff --git a/contracts/ethereum/test/Transfer_Claim_SN.t.sol b/contracts/ethereum/test/Transfer_Claim_SN.t.sol index 20bab404..13b9bd57 100644 --- a/contracts/ethereum/test/Transfer_Claim_SN.t.sol +++ b/contracts/ethereum/test/Transfer_Claim_SN.t.sol @@ -18,9 +18,11 @@ contract TransferTest is Test { address STARKNET_MESSAGING_ADDRESS = 0xde29d060D45901Fb19ED6C6e959EB22d8626708e; uint256 STARKNET_CLAIM_PAYMENT_SELECTOR = 0x15511cc3694f64379908437d6d64458dc76d02482052bfb8a5b33a72c054c77; uint256 STARKNET_CLAIM_PAYMENT_BATCH_SELECTOR = 0x0354a01e49fe07e43306a97ed84dbd5de8238c7d8ff616caa3444630cfc559e6; + uint256 STARKNET_CLAIM_PAYMENT_ERC20_SELECTOR = 0x0091ec2842317cd03601c3f46ee8ebc9b1dc6cdbc96cb7b0873cc6360538d754; address ZKSYNC_DIAMOND_PROXY_ADDRESS = 0x2eD8eF54a16bBF721a318bd5a5C0F39Be70eaa65; bytes4 ZKSYNC_CLAIM_PAYMENT_SELECTOR = 0xa5168739; bytes4 ZKSYNC_CLAIM_PAYMENT_BATCH_SELECTOR = 0x156be1ae; + bytes4 ZKSYNC_CLAIM_PAYMENT_ERC20_SELECTOR = 0xb9738dd6; uint128 STARKNET_CHAIN_ID = 0x534e5f5345504f4c4941; uint128 ZKSYNC_CHAIN_ID = 300; @@ -31,7 +33,7 @@ contract TransferTest is Test { yab = new PaymentRegistry(); proxy = new ERC1967Proxy(address(yab), ""); yab_caller = PaymentRegistry(address(proxy)); - yab_caller.initialize(STARKNET_MESSAGING_ADDRESS, STARKNET_CLAIM_PAYMENT_SELECTOR, STARKNET_CLAIM_PAYMENT_BATCH_SELECTOR, MM_ETHEREUM_WALLET_ADDRESS, ZKSYNC_DIAMOND_PROXY_ADDRESS, ZKSYNC_CLAIM_PAYMENT_SELECTOR, ZKSYNC_CLAIM_PAYMENT_BATCH_SELECTOR, STARKNET_CHAIN_ID, ZKSYNC_CHAIN_ID); + yab_caller.initialize(STARKNET_MESSAGING_ADDRESS, STARKNET_CLAIM_PAYMENT_SELECTOR, STARKNET_CLAIM_PAYMENT_BATCH_SELECTOR, STARKNET_CLAIM_PAYMENT_ERC20_SELECTOR, MM_ETHEREUM_WALLET_ADDRESS, ZKSYNC_DIAMOND_PROXY_ADDRESS, ZKSYNC_CLAIM_PAYMENT_SELECTOR, ZKSYNC_CLAIM_PAYMENT_BATCH_SELECTOR, ZKSYNC_CLAIM_PAYMENT_ERC20_SELECTOR, STARKNET_CHAIN_ID, ZKSYNC_CHAIN_ID); // Mock calls to Starknet Messaging contract vm.mockCall( STARKNET_MESSAGING_ADDRESS, diff --git a/contracts/ethereum/test/Transfer_Claim_ZKSync.t.sol b/contracts/ethereum/test/Transfer_Claim_ZKSync.t.sol index 0d412eb5..4dcc8373 100644 --- a/contracts/ethereum/test/Transfer_Claim_ZKSync.t.sol +++ b/contracts/ethereum/test/Transfer_Claim_ZKSync.t.sol @@ -17,9 +17,11 @@ contract TransferTest is Test { address STARKNET_MESSAGING_ADDRESS = 0xde29d060D45901Fb19ED6C6e959EB22d8626708e; uint256 STARKNET_CLAIM_PAYMENT_SELECTOR = 0x15511cc3694f64379908437d6d64458dc76d02482052bfb8a5b33a72c054c77; uint256 STARKNET_CLAIM_PAYMENT_BATCH_SELECTOR = 0x0354a01e49fe07e43306a97ed84dbd5de8238c7d8ff616caa3444630cfc559e6; + uint256 STARKNET_CLAIM_PAYMENT_ERC20_SELECTOR = 0x0091ec2842317cd03601c3f46ee8ebc9b1dc6cdbc96cb7b0873cc6360538d754; address ZKSYNC_DIAMOND_PROXY_ADDRESS = 0x2eD8eF54a16bBF721a318bd5a5C0F39Be70eaa65; bytes4 ZKSYNC_CLAIM_PAYMENT_SELECTOR = 0xa5168739; bytes4 ZKSYNC_CLAIM_PAYMENT_BATCH_SELECTOR = 0x156be1ae; + bytes4 ZKSYNC_CLAIM_PAYMENT_ERC20_SELECTOR = 0xb9738dd6; uint128 STARKNET_CHAIN_ID = 0x534e5f5345504f4c4941; uint128 ZKSYNC_CHAIN_ID = 300; @@ -30,7 +32,7 @@ contract TransferTest is Test { yab = new PaymentRegistry(); proxy = new ERC1967Proxy(address(yab), ""); yab_caller = PaymentRegistry(address(proxy)); - yab_caller.initialize(STARKNET_MESSAGING_ADDRESS, STARKNET_CLAIM_PAYMENT_SELECTOR, STARKNET_CLAIM_PAYMENT_BATCH_SELECTOR, MM_ETHEREUM_WALLET_ADDRESS, ZKSYNC_DIAMOND_PROXY_ADDRESS, ZKSYNC_CLAIM_PAYMENT_SELECTOR, ZKSYNC_CLAIM_PAYMENT_BATCH_SELECTOR, STARKNET_CHAIN_ID, ZKSYNC_CHAIN_ID); + yab_caller.initialize(STARKNET_MESSAGING_ADDRESS, STARKNET_CLAIM_PAYMENT_SELECTOR, STARKNET_CLAIM_PAYMENT_BATCH_SELECTOR, STARKNET_CLAIM_PAYMENT_ERC20_SELECTOR, MM_ETHEREUM_WALLET_ADDRESS, ZKSYNC_DIAMOND_PROXY_ADDRESS, ZKSYNC_CLAIM_PAYMENT_SELECTOR, ZKSYNC_CLAIM_PAYMENT_BATCH_SELECTOR, ZKSYNC_CLAIM_PAYMENT_ERC20_SELECTOR, STARKNET_CHAIN_ID, ZKSYNC_CHAIN_ID); //Mock calls to ZKSync Mailbox contract vm.mockCall( ZKSYNC_DIAMOND_PROXY_ADDRESS, diff --git a/contracts/starknet/scripts/deploy_erc20.sh b/contracts/starknet/scripts/deploy_erc20.sh new file mode 100755 index 00000000..6dd2597a --- /dev/null +++ b/contracts/starknet/scripts/deploy_erc20.sh @@ -0,0 +1,62 @@ +#!/bin/bash +. contracts/utils/colors.sh #for ANSI colors + +export STARKNET_RPC=$STARKNET_RPC + +if [ -z "$STARKNET_ACCOUNT" ]; then + printf "\n${RED}ERROR:${COLOR_RESET}" + echo "STARKNET_ACCOUNT Variable is empty. Aborting execution.\n" + exit 1 +fi +if [ -z "$STARKNET_KEYSTORE" ] && [ -z "$STARKNET_PRIVATE_KEY" ]; then + printf "\n${RED}ERROR:${COLOR_RESET}" + echo "STARKNET_KEYSTORE and STARKNET_PRIVATE_KEY Variables are empty. Aborting execution.\n" + exit 1 +fi +if [ -z "$MM_STARKNET_WALLET_ADDRESS" ]; then + printf "\n${RED}ERROR:${COLOR_RESET}" + echo "MM_STARKNET_WALLET_ADDRESS Variable is empty. Aborting execution.\n" + exit 1 +fi + + +printf "${GREEN}\n=> [SN] Declaring ERC20${COLOR_RESET}\n" +ERC20_CLASS_HASH=$(starkli declare \ + --account $STARKNET_ACCOUNT \ + $(if [ -n "$STARKNET_KEYSTORE" ]; then echo "--keystore $STARKNET_KEYSTORE"; fi) \ + $(if [ -n "$STARKNET_PRIVATE_KEY" ]; then echo "--private-key $STARKNET_PRIVATE_KEY"; fi) \ + --watch contracts/starknet/target/dev/yab_ERC20.contract_class.json) + + +if [ -z "$ERC20_CLASS_HASH" ]; then + printf "\n${RED}ERROR:${COLOR_RESET}\n" + echo "ERC20_CLASS_HASH Variable is empty. Aborting execution.\n" + exit 1 +fi + +printf "${GREEN}\n=> [SN] ERC20 Declared${COLOR_RESET}\n" + +printf "${CYAN}[SN] ERC20 ClassHash: $ERC20_CLASS_HASH${COLOR_RESET}\n" +# maybe print initial whale +# printf "${PINK}[ETH] Market Maker ETH Wallet: $MM_ETHEREUM_WALLET_ADDRESS${COLOR_RESET}\n" +NAME=0x555249434f494e #'URICOIN' +SYMBOL=0x555249 #'URI' +INITIAL_SUPPLY_low=0x0f4240 # 1000000 +INITIAL_SUPPLY_high=0x0 #because INITIAL_SUPPLY is a uint256 and requires 2 params +RECIPIENT=0x078557823d56a27dd29881285ae58efba18a9da536df0a0c674564e4185e7629 #Braavos account 1, user, contract address format is OK + +printf "${GREEN}\n=> [SN] Deploying ERC20${COLOR_RESET}\n" +ERC20_CONTRACT_ADDRESS=$(starkli deploy --max-fee-raw 31367442226306\ + --account $STARKNET_ACCOUNT \ + $(if [ -n "$STARKNET_KEYSTORE" ]; then echo "--keystore $STARKNET_KEYSTORE"; fi) \ + $(if [ -n "$STARKNET_PRIVATE_KEY" ]; then echo "--private-key $STARKNET_PRIVATE_KEY"; fi) \ + --watch $ERC20_CLASS_HASH \ + $NAME \ + $SYMBOL \ + $INITIAL_SUPPLY_low $INITIAL_SUPPLY_high \ + $RECIPIENT) +echo $ERC20_CONTRACT_ADDRESS + +printf "${GREEN}\n=> [SN] ERC20 Deployed${COLOR_RESET}\n" + +printf "${CYAN}[SN] ERC20 Address: $ERC20_CONTRACT_ADDRESS${COLOR_RESET}\n" diff --git a/contracts/starknet/src/ERC20.cairo b/contracts/starknet/src/ERC20.cairo index 0a76f9b9..6f4234d3 100644 --- a/contracts/starknet/src/ERC20.cairo +++ b/contracts/starknet/src/ERC20.cairo @@ -48,6 +48,20 @@ mod ERC20 { const MINT_TO_ZERO: felt252 = 'ERC20: mint to 0'; } +//hardcoded one + // #[constructor] + // fn constructor( + // ref self: ContractState, + // // name: felt252, + // // symbol: felt252, + // // initial_supply: u256, + // recipient: ContractAddress + // ) { + // // self.initializer(name, symbol); + // self.initializer('URICOIN', 'URI'); + // self._mint(recipient, 1000000); + // } + #[constructor] fn constructor( ref self: ContractState, diff --git a/contracts/starknet/src/escrow.cairo b/contracts/starknet/src/escrow.cairo index 60cdf1c4..c750be5d 100644 --- a/contracts/starknet/src/escrow.cairo +++ b/contracts/starknet/src/escrow.cairo @@ -7,22 +7,35 @@ struct Order { fee: u256 } +#[derive(Copy, Drop, Serde, starknet::Store)] +struct OrderERC20 { + recipient_address: EthAddress, + amount_l2: u256, + amount_l1: u256, + fee: u256, + l2_erc20_address: ContractAddress, + l1_erc20_address: EthAddress +} + #[starknet::interface] trait IEscrow { fn get_order(self: @ContractState, order_id: u256) -> Order; + fn get_order_erc20(self: @ContractState, order_id: u256) -> OrderERC20; fn set_order(ref self: ContractState, order: Order) -> u256; + fn set_order_erc20(ref self: ContractState, order_erc20: OrderERC20) -> u256; fn get_order_pending(self: @ContractState, order_id: u256) -> bool; fn get_order_fee(self: @ContractState, order_id: u256) -> u256; + fn get_order_erc20_fee(self: @ContractState, order_id: u256) -> u256; fn get_eth_transfer_contract(self: @ContractState) -> EthAddress; - fn get_mm_ethereum_contract(self: @ContractState) -> EthAddress; + fn get_mm_ethereum_wallet(self: @ContractState) -> EthAddress; fn get_mm_starknet_contract(self: @ContractState) -> ContractAddress; fn set_eth_transfer_contract(ref self: ContractState, new_contract: EthAddress); - fn set_mm_ethereum_contract(ref self: ContractState, new_contract: EthAddress); + fn set_mm_ethereum_wallet(ref self: ContractState, new_contract: EthAddress); fn set_mm_starknet_contract(ref self: ContractState, new_contract: ContractAddress); fn pause(ref self: ContractState); @@ -32,7 +45,7 @@ trait IEscrow { #[starknet::contract] mod Escrow { use core::traits::Into; -use super::{IEscrow, Order}; + use super::{IEscrow, Order, OrderERC20}; use openzeppelin::{ access::ownable::OwnableComponent, @@ -77,7 +90,9 @@ use super::{IEscrow, Order}; #[derive(Drop, starknet::Event)] enum Event { ClaimPayment: ClaimPayment, + ClaimPaymentERC20: ClaimPaymentERC20, SetOrder: SetOrder, + SetOrderERC20: SetOrderERC20, #[flat] OwnableEvent: OwnableComponent::Event, #[flat] @@ -94,6 +109,17 @@ use super::{IEscrow, Order}; fee: u256 } + #[derive(Drop, starknet::Event)] + struct SetOrderERC20 { + order_id: u256, + recipient_address: EthAddress, + amount_l2: u256, + amount_l1: u256, + fee: u256, + l2_erc20_address: ContractAddress, + l1_erc20_address: EthAddress + } + #[derive(Drop, starknet::Event)] struct ClaimPayment { order_id: u256, @@ -101,14 +127,24 @@ use super::{IEscrow, Order}; amount: u256, } + #[derive(Drop, starknet::Event)] + struct ClaimPaymentERC20 { + order_id: u256, + address: ContractAddress, + amount_l2: u256, + fee: u256, + l2_erc20_address: ContractAddress + } + #[storage] struct Storage { current_order_id: u256, orders: LegacyMap::, + orders_erc20: LegacyMap::, orders_pending: LegacyMap::, orders_senders: LegacyMap::, orders_timestamps: LegacyMap::, - eth_transfer_contract: EthAddress, // our transfer contract in L1 + eth_transfer_contract: EthAddress, // our transfer (PaymentRegistry) contract in L1 mm_ethereum_wallet: EthAddress, mm_starknet_wallet: ContractAddress, native_token_eth_starknet: ContractAddress, @@ -152,9 +188,13 @@ use super::{IEscrow, Order}; self.orders.read(order_id) } + fn get_order_erc20(self: @ContractState, order_id: u256) -> OrderERC20 { + self.orders_erc20.read(order_id) + } + fn set_order(ref self: ContractState, order: Order) -> u256 { self.pausable.assert_not_paused(); - assert(order.amount > 0, 'Amount must be greater than 0'); + assert(order.amount > 0, 'Amount must > 0'); let payment_amount = order.amount + order.fee; let dispatcher = IERC20Dispatcher { contract_address: self.native_token_eth_starknet.read() }; @@ -183,6 +223,54 @@ use super::{IEscrow, Order}; order_id } + + // Recieves in order.fee the total fee for MM + // in order.amount_l2, the total tokens he will give to MM in L2 + // in order.amount_l1, the total tokens he will receive from MM in L1 + // this way, the user is able to bridge tokens cross-erc20, giving, for example, WETH and recieving USDC + // the extra computational costs of this is neglegible: only 1 extra param and 1 extra uint256 stored per ERC20 order in L2, and NO EXTRA COSTS in L1 + fn set_order_erc20(ref self: ContractState, order_erc20: OrderERC20) -> u256 { + self.pausable.assert_not_paused(); + assert(order_erc20.amount_l2 > 0, 'Amount_l2 must > 0'); //in ERC20 + assert(order_erc20.amount_l1 > 0, 'Amount_l1 must > 0'); //in ERC20 + assert(order_erc20.fee > 0, 'Fee must > 0'); //in ETH + + // Fee (ETH): + let eth_dispatcher = IERC20Dispatcher { contract_address: self.native_token_eth_starknet.read() }; + assert(eth_dispatcher.allowance(get_caller_address(), get_contract_address()) >= order_erc20.fee, 'Need allowance for fee'); + assert(eth_dispatcher.balanceOf(get_caller_address()) >= order_erc20.fee, 'Need balance for fee'); + + // Amount (ERC20): + let erc20_dispatcher = IERC20Dispatcher { contract_address: order_erc20.l2_erc20_address }; + assert(erc20_dispatcher.allowance(get_caller_address(), get_contract_address()) >= order_erc20.amount_l2, 'Need allowance for amount_l2'); + assert(erc20_dispatcher.balanceOf(get_caller_address()) >= order_erc20.amount_l2, 'Need balance for amount_l2'); + + let mut order_id = self.current_order_id.read(); + self.orders_erc20.write(order_id, order_erc20); + self.orders_pending.write(order_id, true); + self.orders_senders.write(order_id, get_caller_address()); + self.orders_timestamps.write(order_id, get_block_timestamp()); + self.current_order_id.write(order_id + 1); + + eth_dispatcher.transferFrom(get_caller_address(), get_contract_address(), order_erc20.fee); + erc20_dispatcher.transferFrom(get_caller_address(), get_contract_address(), order_erc20.amount_l2); + + self + .emit( + SetOrderERC20 { + order_id, + recipient_address: order_erc20.recipient_address, + amount_l2: order_erc20.amount_l2, + amount_l1: order_erc20.amount_l1, + fee: order_erc20.fee, + l2_erc20_address: order_erc20.l2_erc20_address, + l1_erc20_address: order_erc20.l1_erc20_address + } + ); + + order_id + } + fn get_order_pending(self: @ContractState, order_id: u256) -> bool { self.orders_pending.read(order_id) } @@ -192,11 +280,16 @@ use super::{IEscrow, Order}; order.fee } + fn get_order_erc20_fee(self: @ContractState, order_id: u256) -> u256 { + let order_erc20: OrderERC20 = self.orders_erc20.read(order_id); + order_erc20.fee + } + fn get_eth_transfer_contract(self: @ContractState) -> EthAddress { self.eth_transfer_contract.read() } - fn get_mm_ethereum_contract(self: @ContractState) -> EthAddress { + fn get_mm_ethereum_wallet(self: @ContractState) -> EthAddress { self.mm_ethereum_wallet.read() } @@ -210,7 +303,7 @@ use super::{IEscrow, Order}; self.eth_transfer_contract.write(new_contract); } - fn set_mm_ethereum_contract(ref self: ContractState, new_contract: EthAddress) { + fn set_mm_ethereum_wallet(ref self: ContractState, new_contract: EthAddress) { self.pausable.assert_not_paused(); self.ownable.assert_only_owner(); self.mm_ethereum_wallet.write(new_contract); @@ -248,6 +341,22 @@ use super::{IEscrow, Order}; _claim_payment(ref self, from_address, order_id, recipient_address, amount); } + #[l1_handler] + fn claim_payment_erc20( + ref self: ContractState, + from_address: felt252, + order_id: u256, + recipient_address: EthAddress, + amount_l1: u256, + l1_erc20_address: EthAddress + ) { + self.pausable.assert_not_paused(); + let eth_transfer_contract_felt: felt252 = self.eth_transfer_contract.read().into(); + assert(from_address == eth_transfer_contract_felt, 'Only PAYMENT_REGISTRY_CONTRACT'); + + _claim_payment_erc20(ref self, from_address, order_id, recipient_address, amount_l1, l1_erc20_address); + } + #[l1_handler] fn claim_payment_batch( ref self: ContractState, @@ -278,20 +387,48 @@ use super::{IEscrow, Order}; order_id: u256, recipient_address: EthAddress, amount: u256 - ) { - assert(self.orders_pending.read(order_id), 'Order withdrew or nonexistent'); + ) { + assert(self.orders_pending.read(order_id), 'Order withdrew or nonexistent'); + + let order = self.orders.read(order_id); + assert(order.recipient_address == recipient_address, 'recipient_address not match L1'); + assert(order.amount == amount, 'amount not match L1'); + + self.orders_pending.write(order_id, false); + let payment_amount = order.amount + order.fee; + + // TODO: In batch, it might be best to transfer all at once + IERC20Dispatcher { contract_address: self.native_token_eth_starknet.read() } + .transfer(self.mm_starknet_wallet.read(), payment_amount); - let order = self.orders.read(order_id); - assert(order.recipient_address == recipient_address, 'recipient_address not match L1'); - assert(order.amount == amount, 'amount not match L1'); + self.emit(ClaimPayment { order_id, address: self.mm_starknet_wallet.read(), amount }); + } - self.orders_pending.write(order_id, false); - let payment_amount = order.amount + order.fee; + fn _claim_payment_erc20( + ref self: ContractState, + from_address: felt252, + order_id: u256, + recipient_address: EthAddress, + amount_l1: u256, + l1_erc20_address: EthAddress + ) { + assert(self.orders_pending.read(order_id), 'Order withdrew or nonexistent'); + + let order_erc20 = self.orders_erc20.read(order_id); + assert(order_erc20.recipient_address == recipient_address, 'recipient_address not match L1'); + assert(order_erc20.amount_l1 == amount_l1, 'amount_l1 not match L1'); + assert(order_erc20.l1_erc20_address == l1_erc20_address, 'l1_erc20_address not match L1'); - // TODO: Might be best to transfer all at once - IERC20Dispatcher { contract_address: self.native_token_eth_starknet.read() } - .transfer(self.mm_starknet_wallet.read(), payment_amount); + self.orders_pending.write(order_id, false); - self.emit(ClaimPayment { order_id, address: self.mm_starknet_wallet.read(), amount }); - } + //Fee (ETH): + IERC20Dispatcher { contract_address: self.native_token_eth_starknet.read() } + .transfer(self.mm_starknet_wallet.read(), order_erc20.fee); + + //Amount (ERC20): + IERC20Dispatcher { contract_address: order_erc20.l2_erc20_address } + .transfer(self.mm_starknet_wallet.read(), order_erc20.amount_l2); + + self.emit(ClaimPaymentERC20 { order_id, address: self.mm_starknet_wallet.read(), amount_l2: order_erc20.amount_l2, fee: order_erc20.fee, l2_erc20_address: order_erc20.l2_erc20_address } ); + } } diff --git a/contracts/starknet/src/lib.cairo b/contracts/starknet/src/lib.cairo index 2e0fd740..93ad58a3 100644 --- a/contracts/starknet/src/lib.cairo +++ b/contracts/starknet/src/lib.cairo @@ -19,6 +19,7 @@ mod tests { mod test_escrow_upgrade; mod test_escrow_ownable; mod test_escrow_claim; + mod test_escrow_erc20; mod utils { mod constants; diff --git a/contracts/starknet/src/tests/test_escrow_allowance.cairo b/contracts/starknet/src/tests/test_escrow_allowance.cairo index 7b9e0b37..64a73ce6 100644 --- a/contracts/starknet/src/tests/test_escrow_allowance.cairo +++ b/contracts/starknet/src/tests/test_escrow_allowance.cairo @@ -147,7 +147,7 @@ mod Escrow { stop_prank(CheatTarget::One(escrow.contract_address)); // check balance - assert(eth_token.balanceOf(escrow.contract_address) == 500, 'set_order: wrong balance '); + assert(eth_token.balanceOf(escrow.contract_address) == 500, 'set_order: wrong balance'); assert(eth_token.balanceOf(MM_STARKNET()) == 0, 'set_order: wrong balance'); } diff --git a/contracts/starknet/src/tests/test_escrow_erc20.cairo b/contracts/starknet/src/tests/test_escrow_erc20.cairo new file mode 100644 index 00000000..ff1410b2 --- /dev/null +++ b/contracts/starknet/src/tests/test_escrow_erc20.cairo @@ -0,0 +1,287 @@ +mod Escrow { + use core::array::ArrayTrait; + use core::to_byte_array::FormatAsByteArray; + use core::serde::Serde; + use core::traits::Into; + use starknet::{EthAddress, ContractAddress}; + use integer::BoundedInt; + + use snforge_std::{declare, ContractClassTrait, L1Handler, L1HandlerTrait}; + use snforge_std::{CheatTarget, start_prank, stop_prank, start_warp, stop_warp}; + + use yab::mocks::mock_Escrow_changed_functions::{IEscrow_mock_changed_functionsDispatcher, IEscrow_mock_changed_functionsDispatcherTrait}; + use yab::mocks::mock_pausableEscrow::{IEscrow_mockPausableDispatcher, IEscrow_mockPausableDispatcherTrait}; + use yab::interfaces::IERC20::{IERC20Dispatcher, IERC20DispatcherTrait}; + use yab::escrow::{IEscrowDispatcher, IEscrowDispatcherTrait, Order, OrderERC20}; + use yab::interfaces::IEVMFactsRegistry::{ + IEVMFactsRegistryDispatcher, IEVMFactsRegistryDispatcherTrait + }; + + use yab::tests::utils::{ + constants::EscrowConstants::{ + USER, OWNER, MM_STARKNET, MM_ETHEREUM, ETH_TRANSFER_CONTRACT, ETH_USER, ETH_USER_2, ETH_USER_3, L1_ERC20_ADDRESS, L1_ERC20_ADDRESS_2 + }, + }; + + use openzeppelin::{ + upgrades::{ + UpgradeableComponent, + interface::{IUpgradeable, IUpgradeableDispatcher, IUpgradeableDispatcherTrait} + }, + }; + + fn setup_with_erc20() -> (IEscrowDispatcher, IERC20Dispatcher, IERC20Dispatcher) { + setup_general_with_erc20(BoundedInt::max(), BoundedInt::max()) + } + + fn setup_general_with_erc20(balance: u256, approved: u256) -> (IEscrowDispatcher, IERC20Dispatcher, IERC20Dispatcher){ + let (eth_token, uri_token) = deploy_erc20_2('ETH', '$ETH', BoundedInt::max(), OWNER(), 'UriCoin', '$Uri', BoundedInt::max(), USER()); + + let escrow = deploy_escrow( + OWNER(), + ETH_TRANSFER_CONTRACT(), + MM_ETHEREUM(), + MM_STARKNET(), + eth_token.contract_address + ); + + start_prank(CheatTarget::One(eth_token.contract_address), OWNER()); + eth_token.transfer(USER(), balance); + stop_prank(CheatTarget::One(eth_token.contract_address)); + + start_prank(CheatTarget::One(eth_token.contract_address), USER()); + eth_token.approve(escrow.contract_address, approved); + stop_prank(CheatTarget::One(eth_token.contract_address)); + + start_prank(CheatTarget::One(uri_token.contract_address), USER()); + uri_token.approve(escrow.contract_address, approved); + stop_prank(CheatTarget::One(uri_token.contract_address)); + + (escrow, eth_token, uri_token) + } + + fn deploy_escrow( + escrow_owner: ContractAddress, + eth_transfer_contract: EthAddress, + mm_ethereum_contract: EthAddress, + mm_starknet_contract: ContractAddress, + native_token_eth_starknet: ContractAddress + ) -> IEscrowDispatcher { + let escrow = declare('Escrow'); + let mut calldata: Array = ArrayTrait::new(); + calldata.append(escrow_owner.into()); + calldata.append(eth_transfer_contract.into()); + calldata.append(mm_ethereum_contract.into()); + calldata.append(mm_starknet_contract.into()); + calldata.append(native_token_eth_starknet.into()); + let address = escrow.deploy(@calldata).unwrap(); + return IEscrowDispatcher { contract_address: address }; + } + + fn deploy_erc20_2( //without this, the declare('ERC20') line is called twice, causing the execution to crash + name: felt252, symbol: felt252, initial_supply: u256, recipent: ContractAddress, name_2: felt252, symbol_2: felt252, initial_supply_2: u256, recipent_2: ContractAddress + ) -> (IERC20Dispatcher, IERC20Dispatcher) { + let erc20 = declare('ERC20'); + let mut calldata = array![name, symbol]; + Serde::serialize(@initial_supply, ref calldata); + calldata.append(recipent.into()); + let address = erc20.deploy(@calldata).unwrap(); + + let mut calldata_2 = array![name_2, symbol_2]; + Serde::serialize(@initial_supply_2, ref calldata_2); + calldata_2.append(recipent_2.into()); + let address_2 = erc20.deploy(@calldata_2).unwrap(); + + return (IERC20Dispatcher { contract_address: address }, IERC20Dispatcher { contract_address: address_2 }); + } + + #[test] + fn test_claim_payment_erc20() { + let (escrow, eth_token, uri_token) = setup_with_erc20(); + + // check balance + assert(eth_token.balanceOf(escrow.contract_address) == 0, 'init: wrong balance 1'); + assert(eth_token.balanceOf(MM_STARKNET()) == 0, 'init: wrong balance 2'); + + assert(uri_token.balanceOf(escrow.contract_address) == 0, 'init: wrong balance 3'); + assert(uri_token.balanceOf(USER()) == BoundedInt::max(), 'init: wrong balance 4'); + assert(uri_token.balanceOf(MM_STARKNET()) == 0, 'init: wrong balance 5'); + + start_prank(CheatTarget::One(escrow.contract_address), USER()); + let order_erc20 = OrderERC20 { recipient_address: ETH_USER(), amount_l2: 200, amount_l1: 100, fee: 10, l2_erc20_address: uri_token.contract_address, l1_erc20_address: L1_ERC20_ADDRESS() }; + let order_id = escrow.set_order_erc20(order_erc20); + stop_prank(CheatTarget::One(escrow.contract_address)); + + // check balance + assert(eth_token.balanceOf(escrow.contract_address) == 10, 'set_order: wrong balance '); + assert(eth_token.balanceOf(MM_STARKNET()) == 0, 'set_order: wrong balance'); + + assert(uri_token.balanceOf(escrow.contract_address) == 200, 'init: wrong balance 3'); + assert(uri_token.balanceOf(USER()) == BoundedInt::max()-200, 'init: wrong balance 4'); + assert(uri_token.balanceOf(MM_STARKNET()) == 0, 'init: wrong balance 5'); + + // check Order + assert(order_id == 0, 'wrong order_id'); + let order_save = escrow.get_order_erc20(order_id); + assert(order_erc20.recipient_address == order_save.recipient_address, 'wrong recipient_address'); + assert(order_erc20.amount_l2 == order_save.amount_l2, 'wrong amount_l2'); + assert(order_erc20.amount_l1 == order_save.amount_l1, 'wrong amount_l1'); + assert(order_erc20.fee == order_save.fee, 'wrong fee'); + assert(order_erc20.l2_erc20_address == order_save.l2_erc20_address, 'wrong l2_erc20_address'); + assert(order_erc20.l1_erc20_address == order_save.l1_erc20_address, 'wrong l1_erc20_address'); + assert(escrow.get_order_pending(order_id), 'wrong order used'); + + let mut l1_handler = L1HandlerTrait::new( + contract_address: escrow.contract_address, + function_name: 'claim_payment_erc20' + ); + + let mut payload_buffer: Array = ArrayTrait::new(); + Serde::serialize(@order_id, ref payload_buffer); + Serde::serialize(@order_erc20.recipient_address, ref payload_buffer); + Serde::serialize(@order_erc20.amount_l1, ref payload_buffer); + Serde::serialize(@order_erc20.l1_erc20_address, ref payload_buffer); + + l1_handler.from_address = ETH_TRANSFER_CONTRACT().into(); + l1_handler.payload = payload_buffer.span(); + + l1_handler.execute().expect('Failed to execute l1_handler'); + + // check Order + assert(!escrow.get_order_pending(order_id), 'wrong order used'); + // check balance + assert(eth_token.balanceOf(escrow.contract_address) == 0, 'withdraw: wrong balance'); + assert(eth_token.balanceOf(MM_STARKNET()) == 10, 'withdraw: wrong balance'); + + assert(uri_token.balanceOf(escrow.contract_address) == 0, 'init: wrong balance'); + assert(uri_token.balanceOf(MM_STARKNET()) == 200, 'init: wrong balance'); + } + + #[test] + fn test_fail_claim_erc20_wrong_erc20() { + let (escrow, eth_token, uri_token) = setup_with_erc20(); + + //set order + start_prank(CheatTarget::One(escrow.contract_address), USER()); + let order_erc20 = OrderERC20 { recipient_address: ETH_USER(), amount_l2: 200, amount_l1: 100, fee: 10, l2_erc20_address: uri_token.contract_address, l1_erc20_address: L1_ERC20_ADDRESS() }; + let order_id = escrow.set_order_erc20(order_erc20); + stop_prank(CheatTarget::One(escrow.contract_address)); + + + let mut l1_handler = L1HandlerTrait::new( + contract_address: escrow.contract_address, + function_name: 'claim_payment_erc20' + ); + + //claim payment + let mut payload_buffer: Array = ArrayTrait::new(); + Serde::serialize(@order_id, ref payload_buffer); + Serde::serialize(@order_erc20.recipient_address, ref payload_buffer); + Serde::serialize(@order_erc20.amount_l1, ref payload_buffer); + Serde::serialize(@L1_ERC20_ADDRESS_2(), ref payload_buffer); //wrong, sent to l2_erc20_address instead of l1_erc20_address + + l1_handler.from_address = ETH_TRANSFER_CONTRACT().into(); + l1_handler.payload = payload_buffer.span(); + + // same as "Should Panic" but for the L1 handler function + match l1_handler.execute() { + Result::Ok(_) => panic_with_felt252('shouldve panicked'), + Result::Err(RevertedTransaction) => { + assert(*RevertedTransaction.panic_data.at(0) == 'l1_erc20_address not match L1', *RevertedTransaction.panic_data.at(0)); + } + } + } + + #[test] + fn test_fail_claim_erc20_wrong_amount_l2() { + let (escrow, eth_token, uri_token) = setup_with_erc20(); + + //set order + start_prank(CheatTarget::One(escrow.contract_address), USER()); + let order_erc20 = OrderERC20 { recipient_address: ETH_USER(), amount_l2: 200, amount_l1: 100, fee: 10, l2_erc20_address: uri_token.contract_address, l1_erc20_address: L1_ERC20_ADDRESS() }; + let order_id = escrow.set_order_erc20(order_erc20); + stop_prank(CheatTarget::One(escrow.contract_address)); + + + let mut l1_handler = L1HandlerTrait::new( + contract_address: escrow.contract_address, + function_name: 'claim_payment_erc20' + ); + + //claim payment + let mut payload_buffer: Array = ArrayTrait::new(); + Serde::serialize(@order_id, ref payload_buffer); + Serde::serialize(@order_erc20.recipient_address, ref payload_buffer); + Serde::serialize(@order_erc20.amount_l2, ref payload_buffer); //wrong, sent amount_l2 instead of amount_l1 + Serde::serialize(@order_erc20.l1_erc20_address, ref payload_buffer); + + l1_handler.from_address = ETH_TRANSFER_CONTRACT().into(); + l1_handler.payload = payload_buffer.span(); + + // same as "Should Panic" but for the L1 handler function + match l1_handler.execute() { + Result::Ok(_) => panic_with_felt252('shouldve panicked'), + Result::Err(RevertedTransaction) => { + assert(*RevertedTransaction.panic_data.at(0) == 'amount_l1 not match L1', *RevertedTransaction.panic_data.at(0)); + } + } + } + + #[test] + fn test_fail_claim_erc20_wrong_recipient_address() { + let (escrow, eth_token, uri_token) = setup_with_erc20(); + + //set order + start_prank(CheatTarget::One(escrow.contract_address), USER()); + let order_erc20 = OrderERC20 { recipient_address: ETH_USER(), amount_l2: 200, amount_l1: 100, fee: 10, l2_erc20_address: uri_token.contract_address, l1_erc20_address: L1_ERC20_ADDRESS() }; + let order_id = escrow.set_order_erc20(order_erc20); + stop_prank(CheatTarget::One(escrow.contract_address)); + + + let mut l1_handler = L1HandlerTrait::new( + contract_address: escrow.contract_address, + function_name: 'claim_payment_erc20' + ); + + //claim payment + let mut payload_buffer: Array = ArrayTrait::new(); + Serde::serialize(@order_id, ref payload_buffer); + Serde::serialize(@ETH_USER_2(), ref payload_buffer); //wrong recipient address + Serde::serialize(@order_erc20.amount_l1, ref payload_buffer); + Serde::serialize(@order_erc20.l1_erc20_address, ref payload_buffer); + + l1_handler.from_address = ETH_TRANSFER_CONTRACT().into(); + l1_handler.payload = payload_buffer.span(); + + // same as "Should Panic" but for the L1 handler function + match l1_handler.execute() { + Result::Ok(_) => panic_with_felt252('shouldve panicked'), + Result::Err(RevertedTransaction) => { + assert(*RevertedTransaction.panic_data.at(0) == 'recipient_address not match L1', *RevertedTransaction.panic_data.at(0)); + } + } + } + + #[test] + fn test_fail_random_eth_user_calls_l1_handler() { + let (escrow, _, _) = setup_with_erc20(); + let data: Array = array![1, MM_ETHEREUM().into(), 3, L1_ERC20_ADDRESS().into(), 5]; + let mut payload_buffer: Array = ArrayTrait::new(); + data.serialize(ref payload_buffer); + let mut l1_handler = L1HandlerTrait::new( + contract_address: escrow.contract_address, + function_name: 'claim_payment_erc20', + ); + l1_handler.from_address = ETH_USER().into(); + + l1_handler.payload = payload_buffer.span(); + + // same as "Should Panic" but for the L1 handler function + match l1_handler.execute() { + Result::Ok(_) => panic_with_felt252('shouldve panicked'), + Result::Err(RevertedTransaction) => { + assert(*RevertedTransaction.panic_data.at(0) == 'Only PAYMENT_REGISTRY_CONTRACT', *RevertedTransaction.panic_data.at(0)); + } + } + } +} diff --git a/contracts/starknet/src/tests/test_escrow_ownable.cairo b/contracts/starknet/src/tests/test_escrow_ownable.cairo index ef1a9f91..77285646 100644 --- a/contracts/starknet/src/tests/test_escrow_ownable.cairo +++ b/contracts/starknet/src/tests/test_escrow_ownable.cairo @@ -118,14 +118,14 @@ mod Escrow { #[should_panic(expected: ('Caller is not the owner',))] fn test_fail_set_mm_ethereum_contract() { let (escrow, _) = setup(); - escrow.set_mm_ethereum_contract(MM_ETHEREUM()); + escrow.set_mm_ethereum_wallet(MM_ETHEREUM()); } #[test] fn test_set_mm_ethereum_contract() { let (escrow, _) = setup(); start_prank(CheatTarget::One(escrow.contract_address), OWNER()); - escrow.set_mm_ethereum_contract(MM_ETHEREUM()); + escrow.set_mm_ethereum_wallet(MM_ETHEREUM()); } #[test] diff --git a/contracts/starknet/src/tests/test_escrow_pause.cairo b/contracts/starknet/src/tests/test_escrow_pause.cairo index b16d7219..cf05ec40 100644 --- a/contracts/starknet/src/tests/test_escrow_pause.cairo +++ b/contracts/starknet/src/tests/test_escrow_pause.cairo @@ -18,7 +18,7 @@ mod Escrow { use yab::tests::utils::{ constants::EscrowConstants::{ - USER, OWNER, MM_STARKNET, MM_ETHEREUM, ETH_TRANSFER_CONTRACT, ETH_USER + USER, OWNER, MM_STARKNET, MM_ETHEREUM, ETH_TRANSFER_CONTRACT, ETH_USER, L1_ERC20_ADDRESS }, }; @@ -268,4 +268,33 @@ mod Escrow { } } } + + #[test] + fn test_fail_call_erc20_l1_handler_while_paused() { + let (escrow, _) = setup(); + let pausable = IPausableDispatcher { contract_address: escrow.contract_address }; + + start_prank(CheatTarget::One(escrow.contract_address), OWNER()); + escrow.pause(); + stop_prank(CheatTarget::One(escrow.contract_address)); + + let data: Array = array![1, MM_ETHEREUM().into(), 3, L1_ERC20_ADDRESS().into(), 5]; + let mut payload_buffer: Array = ArrayTrait::new(); + data.serialize(ref payload_buffer); + let mut l1_handler = L1HandlerTrait::new( + contract_address: escrow.contract_address, + function_name: 'claim_payment_erc20', + ); + l1_handler.from_address = ETH_USER().into(); + + l1_handler.payload = payload_buffer.span(); + + // same as "Should Panic" but for the L1 handler function + match l1_handler.execute() { + Result::Ok(_) => panic_with_felt252('shouldve panicked'), + Result::Err(RevertedTransaction) => { + assert(*RevertedTransaction.panic_data.at(0) == 'Pausable: paused', *RevertedTransaction.panic_data.at(0)); + } + } + } } diff --git a/contracts/starknet/src/tests/utils/constants.cairo b/contracts/starknet/src/tests/utils/constants.cairo index 7e9c638a..c354c310 100644 --- a/contracts/starknet/src/tests/utils/constants.cairo +++ b/contracts/starknet/src/tests/utils/constants.cairo @@ -32,4 +32,12 @@ mod EscrowConstants { fn ETH_USER_3() -> EthAddress { 101.try_into().unwrap() } + + fn L1_ERC20_ADDRESS() -> EthAddress { + 111.try_into().unwrap() + } + + fn L1_ERC20_ADDRESS_2() -> EthAddress { + 112.try_into().unwrap() + } } diff --git a/contracts/zksync/contracts/ERC20_L2.sol b/contracts/zksync/contracts/ERC20_L2.sol new file mode 100644 index 00000000..085a2e39 --- /dev/null +++ b/contracts/zksync/contracts/ERC20_L2.sol @@ -0,0 +1,16 @@ +// SPDX-License-Identifier: Unlicense +pragma solidity ^0.8.0; + +import "@openzeppelin/contracts/token/ERC20/ERC20.sol"; + +// contract UriCoin is ERC20 { +// constructor() ERC20("UriCoin", "Uri") { +// _mint(0xB321099cf86D9BB913b891441B014c03a6CcFc54, 1000000); //hardcoded recipient + initial_supply +// } +// } + +contract UriCoin is ERC20 { + constructor(address initial_whale, uint256 initial_supply) ERC20("UriCoin", "Uri") { + _mint(initial_whale, initial_supply); //hardcoded recipient + initial_supply + } +} \ No newline at end of file diff --git a/contracts/zksync/contracts/escrow.sol b/contracts/zksync/contracts/escrow.sol index 00e4a4c3..2ac789bf 100644 --- a/contracts/zksync/contracts/escrow.sol +++ b/contracts/zksync/contracts/escrow.sol @@ -7,26 +7,41 @@ import "@openzeppelin/contracts/token/ERC20/ERC20.sol"; import {OwnableUpgradeable} from "@openzeppelin/contracts-upgradeable/access/OwnableUpgradeable.sol"; import {PausableUpgradeable} from "@openzeppelin/contracts-upgradeable/security/PausableUpgradeable.sol"; +import "@openzeppelin/contracts/token/ERC20/utils/SafeERC20.sol"; // import {Ownable2StepUpgradeable} from "@openzeppelin/contracts-upgradeable/access/Ownable2StepUpgradeable.sol"; // import {UUPSUpgradeable} from "@openzeppelin/contracts-upgradeable/proxy/utils/UUPSUpgradeable.sol"; // TODO make upgradeable contract Escrow is Initializable, OwnableUpgradeable, PausableUpgradeable { //}, UUPSUpgradeable { + using SafeERC20 for IERC20; + struct Order { address recipient_address; uint256 amount; uint256 fee; } + struct OrderERC20 { + address recipient_address; + uint256 amount_l2; + uint256 amount_l1; + uint256 fee; + address l2_erc20_address; + address l1_erc20_address; + } + event SetOrder(uint256 order_id, address recipient_address, uint256 amount, uint256 fee); + event SetOrderERC20(uint256 order_id, address recipient_address, uint256 amount_l2, uint256 amount_l1, uint256 fee, address l2_erc20_address, address l1_erc20_address); - event ClaimPayment(uint256 order_id, address claimerAddress, uint256 amount); + event ClaimPayment(uint256 order_id, address claimer_address, uint256 amount); + event ClaimPaymentERC20(uint256 order_id, address claimer_address, uint256 amount_l2, uint256 fee, address l2_erc20_address); //storage uint256 private _current_order_id; mapping(uint256 => Order) private _orders; + mapping(uint256 => OrderERC20) private _orders_erc20; mapping(uint256 => bool) private _orders_pending; mapping(uint256 => address) private _orders_senders; mapping(uint256 => uint256) private _orders_timestamps; @@ -47,10 +62,65 @@ contract Escrow is Initializable, OwnableUpgradeable, PausableUpgradeable { //}, // FUNCTIONS : + // Recieves in msg.value the total fee for MM + // in amount_l2, the total tokens he will give to MM in L2 + // in amount_l1, the total tokens he will receive from MM in L1 + // this way, the user is able to bridge tokens cross-erc20, giving, for example, WETH and recieving USDC + // the extra computational costs of this is neglegible: only 1 extra param and 1 extra uint256 stored per ERC20 order in L2, and NO EXTRA COSTS in L1 + function set_order_erc20(address recipient_address, uint256 amount_l2, uint256 amount_l1, address l2_erc20_address, address l1_erc20_address) public payable whenNotPaused returns (uint256) { + require(msg.value > 0, 'some ETH must be sent as MM fees'); + require(amount_l2 > 0, 'some tokens must be sent to MM in L2'); + require(amount_l1 > 0, 'some tokens must be sent to MM in L1'); + + //the following needs allowance, which is not set automatically + require(IERC20(l2_erc20_address).balanceOf(msg.sender) >= amount_l2, "User has insuficient funds"); + require(IERC20(l2_erc20_address).allowance(msg.sender, address(this)) >= amount_l2, "Escrow has insuficient allowance"); + IERC20(l2_erc20_address).safeTransferFrom(msg.sender, address(this), amount_l2); //will revert if failed + + OrderERC20 memory new_order = OrderERC20({recipient_address: recipient_address, amount_l2: amount_l2, amount_l1: amount_l1, fee: msg.value, l2_erc20_address: l2_erc20_address, l1_erc20_address: l1_erc20_address}); + _orders_erc20[_current_order_id] = new_order; + _orders_pending[_current_order_id] = true; + _orders_senders[_current_order_id] = msg.sender; + _orders_timestamps[_current_order_id] = block.timestamp; + _current_order_id++; //this here to follow CEI pattern + + emit SetOrderERC20(_current_order_id-1, recipient_address, amount_l2, amount_l1, msg.value, l2_erc20_address, l1_erc20_address); + return _current_order_id-1; + } + + // l1 handler + function claim_payment_erc20( + uint256 order_id, + address recipient_address, + uint256 amount_l1, + address l1_erc20_address + ) public whenNotPaused { + require(msg.sender == ethereum_payment_registry, 'Only PAYMENT_REGISTRY can call'); + require(_orders_pending[order_id], 'Order claimed or nonexistent'); + + OrderERC20 memory current_order = _orders_erc20[order_id]; //TODO check if order is memory or calldata + require(current_order.recipient_address == recipient_address, 'recipient_address not match L1'); + require(current_order.amount_l1 == amount_l1, 'amount_l1 not match L1'); + require(current_order.l1_erc20_address == l1_erc20_address, 'l1_erc20_address not match L1'); + + _orders_pending[order_id] = false; + + //transfer amount + fee in ETH: + IERC20(current_order.l2_erc20_address).safeTransfer(mm_zksync_wallet, current_order.amount_l2); //will revert if failed + (bool success,) = payable(address(uint160(mm_zksync_wallet))).call{value: current_order.fee}(""); + require(success, "Fee transfer failed."); + + emit ClaimPaymentERC20(order_id, mm_zksync_wallet, current_order.amount_l2, current_order.fee, current_order.l2_erc20_address); + } + function get_order(uint256 order_id) public view returns (Order memory) { return _orders[order_id]; } + function get_order_erc20(uint256 order_id) public view returns (OrderERC20 memory) { + return _orders_erc20[order_id]; + } + //Function recieves in msg.value the total value, and in fee the user specifies what portion of that msg.value is fee for MM function set_order(address recipient_address, uint256 fee) public payable whenNotPaused returns (uint256) { require(msg.value > 0, 'some ETH must be sent'); diff --git a/contracts/zksync/deploy/deploy_erc20.ts b/contracts/zksync/deploy/deploy_erc20.ts new file mode 100644 index 00000000..1d17ef92 --- /dev/null +++ b/contracts/zksync/deploy/deploy_erc20.ts @@ -0,0 +1,16 @@ +import { deployContractWithProxy } from "./utils"; +import { deployContract } from "./utils"; +// import {ERC1967Proxy} from "@openzeppelin/contracts/proxy/ERC1967/ERC1967Proxy.sol"; +import * as dotenv from 'dotenv'; +import { utils } from "zksync-ethers"; + + +// It will deploy a Escrow contract to selected network +export default async function () { + const escrowArtifactName = "UriCoin"; + + const escrowConstructorArguments = []; + const escrow = await deployContract(escrowArtifactName, escrowConstructorArguments); + + console.log("Deploy erc20 result:", escrow); +} diff --git a/contracts/zksync/deploy_erc20.sh b/contracts/zksync/deploy_erc20.sh new file mode 100755 index 00000000..4e317c3b --- /dev/null +++ b/contracts/zksync/deploy_erc20.sh @@ -0,0 +1,17 @@ +#!/bin/bash +. contracts/utils/colors.sh #for ANSI colors + +cd ./contracts/zksync/ + +ERC20_CONTRACT_ADDRESS=$(yarn deploy-erc20 | grep "Contract address:" | egrep -i -o '0x[a-zA-Z0-9]{40}') + +if [ -z "$ERC20_CONTRACT_ADDRESS" ]; then + printf "\n${RED}ERROR:${COLOR_RESET}\n" + echo "ERC20_CONTRACT_ADDRESS Variable is empty. Aborting execution.\n" + exit 1 +fi + +printf "${CYAN}[ZKSync] ERC20 Address: $ERC20_CONTRACT_ADDRESS${COLOR_RESET}\n" + +cd ../.. + diff --git a/contracts/zksync/package.json b/contracts/zksync/package.json index 5648a349..ad31b109 100644 --- a/contracts/zksync/package.json +++ b/contracts/zksync/package.json @@ -3,11 +3,12 @@ "private": true, "packageManager": "yarn@1.22.21", "scripts": { + "deploy-erc20": "hardhat deploy-zksync --network zkSyncSepoliaTestnet --script deploy_erc20.ts", "deploy": "hardhat deploy-zksync --network zkSyncSepoliaTestnet --script deploy.ts", "deploy-devnet": "hardhat deploy-zksync --network dockerizedNode --script deploy.ts", "compile": "hardhat compile", "clean": "hardhat clean", - "test": "hardhat test --network dockerizedNode --show-stack-traces", + "test": "hardhat test --network dockerizedNode --show-stack-traces test/main.test.ts test/erc20.test.ts", "test-in-memory": "hardhat test --network inMemoryNode --show-stack-traces" }, "devDependencies": { diff --git a/contracts/zksync/test/erc20.test.ts b/contracts/zksync/test/erc20.test.ts new file mode 100644 index 00000000..502d32d2 --- /dev/null +++ b/contracts/zksync/test/erc20.test.ts @@ -0,0 +1,180 @@ +import { expect } from 'chai'; +import { deployAndInit } from './utils'; +import { Contract, Fragment, Wallet, Provider, AddressLike } from 'ethers'; +import { getWallet, deployContract, LOCAL_RICH_WALLETS, getProvider } from '../deploy/utils'; + +let provider: Provider; +let escrow: Contract; +let erc20_l2: Contract; +let erc20_l1: Contract; +let paymentRegistry: Wallet = getWallet(LOCAL_RICH_WALLETS[3].privateKey); //its Wallet data type because I will mock calls from this addr + +let deployer: Wallet = getWallet(LOCAL_RICH_WALLETS[0].privateKey); +let user_zk: Wallet = getWallet(LOCAL_RICH_WALLETS[1].privateKey); +let user_zk2: Wallet = getWallet(LOCAL_RICH_WALLETS[2].privateKey); +let user_eth: Wallet = getWallet(LOCAL_RICH_WALLETS[1].privateKey); +let user_eth2: Wallet = getWallet(LOCAL_RICH_WALLETS[2].privateKey); + + + +const fee = 1; //TODO check, maybe make fuzz +const value = 10; //TODO check, maybe make fuzz +const amount_l2 = 20; +const amount_l1 = 10; + +const erc20_l1_address = '0xA0b86991c6218b36c1d19D4a2e9Eb0cE3606eB48'; +let erc20_l2_address: AddressLike; +let escrow_address: AddressLike; + +const initial_erc20_balance = 1000000; + +beforeEach( async () => { + escrow = await deployAndInit(); + escrow_address = await escrow.getAddress(); + + erc20_l2 = await deployContract("UriCoin", [user_zk.address, initial_erc20_balance], { wallet: deployer }); + erc20_l2_address = await erc20_l2.getAddress(); + + provider = getProvider(); +}); + + +describe('ERC20 basic tests', function () { + it("Should airdrop correctly", async () => { + const balanceOf = await erc20_l2.balanceOf(user_zk.address); + expect(balanceOf).to.eq(initial_erc20_balance) + }); + + it("Should transfer correctly", async () => { + const transferTx = await erc20_l2.connect(user_zk).transfer(user_zk2.address, value); + await transferTx.wait(); + + const balanceOf = await erc20_l2.balanceOf(user_zk2.address); + expect(balanceOf).to.eq(value) + }); + + it("Should not transfer more than balance", async () => { + expect(erc20_l2.connect(user_zk).transfer(user_zk2.address, initial_erc20_balance+1)).to.be.revertedWith("ERC20: transfer amount exceeds balance"); + }); + + it("Should not transfer without allowance", async () => { + expect(erc20_l2.connect(user_zk2).transferFrom(user_zk.address, user_zk2.address, value)).to.be.revertedWith("ERC20: transfer amount exceeds allowance"); + }); + + it("Should approve correctly", async () => { + const approveTx = await erc20_l2.connect(user_zk).approve(user_zk2.address, value); + await approveTx.wait(); + + const allowance = await erc20_l2.allowance(user_zk.address, user_zk2.address); + expect(allowance).to.eq(value) + }); + + it("Should transferFrom correctly", async () => { + const approveTx = await erc20_l2.connect(user_zk).approve(user_zk2.address, value); + await approveTx.wait(); + + const transferFromTx = await erc20_l2.connect(user_zk2).transferFrom(user_zk.address, user_zk2.address, value); + await transferFromTx.wait(); + + const balanceOf = await erc20_l2.balanceOf(user_zk2.address); + expect(balanceOf).to.eq(value) + }); +}) + + +describe('ERC20 Set Order tests', function () { + it("Should emit correct Event", async () => { + const approveTx = await erc20_l2.connect(user_zk).approve(escrow_address, amount_l2); + await approveTx.wait(); + + const setOrderTx = await escrow.connect(user_zk).set_order_erc20(user_eth.address, amount_l2, amount_l1, erc20_l2_address, erc20_l1_address, {value: fee}); + + await expect(setOrderTx) + .to.emit(escrow, "SetOrderERC20").withArgs(0, user_eth.address, amount_l2, amount_l1, fee, erc20_l2_address, erc20_l1_address); + }); + it("Should get the order setted", async () => { + const approveTx = await erc20_l2.connect(user_zk).approve(escrow_address, amount_l2); + await approveTx.wait(); + + const setOrderTx = await escrow.connect(user_zk).set_order_erc20(user_eth.address, amount_l2, amount_l1, erc20_l2_address, erc20_l1_address, {value: fee}); + await setOrderTx.wait(); + + const newOrder = await escrow.get_order_erc20(0); + + expect(newOrder[0]).to.eq(user_eth.address); //recipient_address + expect(Number(newOrder[1])).to.eq(amount_l2); //amount_l2 + expect(Number(newOrder[2])).to.eq(amount_l1); //amount_l1 + expect(Number(newOrder[3])).to.eq(fee); //fee + expect(newOrder[4]).to.eq(erc20_l2_address); //erc20_l2_address + expect(newOrder[5]).to.eq(erc20_l1_address); //erc20_l1_address + }) + it("Should get the pending order", async () => { + const approveTx = await erc20_l2.connect(user_zk).approve(escrow_address, amount_l2); + await approveTx.wait(); + + const setOrderTx = await escrow.connect(user_zk).set_order_erc20(user_eth.address, amount_l2, amount_l1, erc20_l2_address, erc20_l1_address, {value: fee}); + await setOrderTx.wait(); + + expect(await escrow.is_order_pending(0)).to.equal(true); + }) + it("Should not get the pending order", async () => { + expect(await escrow.is_order_pending(0)).to.equal(false); + }) +}); + +describe('ERC20 Claim Payment tests', function () { + it("Should allow PaymentRegistry to claim payment", async () => { + let mm_init_balance = await erc20_l2.balanceOf(escrow.mm_zksync_wallet()); + + const approveTx = await erc20_l2.connect(user_zk).approve(escrow_address, amount_l2); + await approveTx.wait(); + + const setOrderTx = await escrow.connect(user_zk).set_order_erc20(user_eth.address, amount_l2, amount_l1, erc20_l2_address, erc20_l1_address, {value: fee}); + await setOrderTx.wait(); + + //claims the payment of a transfer of amount_l1 made to erc20_l1 + const claimPaymentTx = await escrow.connect(paymentRegistry).claim_payment_erc20(0, user_eth, amount_l1, erc20_l1_address); //unit testing the Escrow, not the PaymentRegistry's transfer/claim_payment ACL + await claimPaymentTx.wait(); + + let mm_final_balance = await erc20_l2.balanceOf(escrow.mm_zksync_wallet()); + + //MM balance should increase by amount_l2 + expect(mm_final_balance - mm_init_balance).to.equals(amount_l2); + }); + + it("Should not allow PaymentRegistry to claim unexisting payment", async () => { + expect(escrow.connect(paymentRegistry).claim_payment_erc20(0, user_eth, amount_l1, erc20_l1_address)).to.be.revertedWith("Order claimed or nonexistent"); + }); + + it("Should not allow random user to call claim payment", async () => { + expect(escrow.connect(user_zk).claim_payment_erc20(0, user_eth, amount_l1, erc20_l1_address)).to.be.revertedWith("Only PAYMENT_REGISTRY can call"); + }); + + it("Should not allow PaymentRegistry to claim claimed payment", async () => { + const approveTx = await erc20_l2.connect(user_zk).approve(escrow_address, amount_l2); + await approveTx.wait(); + + const setOrderTx = await escrow.connect(user_zk).set_order_erc20(user_eth.address, amount_l2, amount_l1, erc20_l2_address, erc20_l1_address, {value: fee}); + await setOrderTx.wait(); + + //claims the payment of a transfer of amount_l1 made to erc20_l1 + const claimPaymentTx = await escrow.connect(paymentRegistry).claim_payment_erc20(0, user_eth, amount_l1, erc20_l1_address); //unit testing the Escrow, not the PaymentRegistry's transfer/claim_payment ACL + await claimPaymentTx.wait(); + + expect(escrow.connect(user_zk).claim_payment_erc20(0, user_eth, amount_l1, erc20_l1_address)).to.be.revertedWith("Only PAYMENT_REGISTRY can call"); + }); + it("Should not get the pending order after being claimed", async () => { + const approveTx = await erc20_l2.connect(user_zk).approve(escrow_address, amount_l2); + await approveTx.wait(); + + const setOrderTx = await escrow.connect(user_zk).set_order_erc20(user_eth.address, amount_l2, amount_l1, erc20_l2_address, erc20_l1_address, {value: fee}); + await setOrderTx.wait(); + + //claims the payment of a transfer of amount_l1 made to erc20_l1 + const claimPaymentTx = await escrow.connect(paymentRegistry).claim_payment_erc20(0, user_eth, amount_l1, erc20_l1_address); //unit testing the Escrow, not the PaymentRegistry's transfer/claim_payment ACL + await claimPaymentTx.wait(); + + expect(await escrow.is_order_pending(0)).to.equal(false); + }); +}) + diff --git a/contracts/zksync/test/main.test.ts b/contracts/zksync/test/main.test.ts index 2f4007f1..02417e27 100644 --- a/contracts/zksync/test/main.test.ts +++ b/contracts/zksync/test/main.test.ts @@ -165,6 +165,25 @@ describe('Claim Payment tests', function () { it("Should not allow random user to call claim payment", async () => { expect(escrow.connect(user_zk).claim_payment(0, user_eth, value)).to.be.revertedWith("Only PAYMENT_REGISTRY can call"); }); + it("Should not allow PaymentRegistry to claim claimed payment", async () => { + const setOrderTx = await escrow.connect(user_zk).set_order(user_eth, fee, {value}); + await setOrderTx.wait(); + + const claimPaymentTx = await escrow.connect(paymentRegistry).claim_payment(0, user_eth, value-fee); + await claimPaymentTx.wait(); + + expect(escrow.connect(paymentRegistry).claim_payment(0, user_eth, value)).to.be.revertedWith("Order claimed or nonexistent"); + }); + + it("Should not get the pending order after being claimed", async () => { + const setOrderTx = await escrow.connect(user_zk).set_order(user_eth, fee, {value}); + await setOrderTx.wait(); + + const claimPaymentTx = await escrow.connect(paymentRegistry).claim_payment(0, user_eth, value-fee); + await claimPaymentTx.wait(); + + expect(await escrow.is_order_pending(0)).to.equal(false); + }); }) @@ -198,4 +217,3 @@ describe('Claim payment batch tests', function () { await expect(escrow.connect(user_eth).claim_payment_batch([0], [user_eth.address], [value-fee])).to.be.revertedWith("Only PAYMENT_REGISTRY can call"); }); }); -