diff --git a/README.md b/README.md index b8d43b61..e7054c68 100644 --- a/README.md +++ b/README.md @@ -83,9 +83,12 @@ CHILD_GAS_SERVICE_ADDRESS= ROOT_CHAIN_NAME="ROOT" CHILD_CHAIN_NAME="CHILD" ROOT_IMX_ADDRESS= +INITIAL_IMX_CUMULATIVE_DEPOSIT_LIMIT="0" ``` where `{ROOT,CHILD}_{GATEWAY,GAS_SERVICE}_ADDRESS` refers to the gateway and gas service addresses used by Axelar. +`INITIAL_IMX_CUMULATIVE_DEPOSIT_LIMIT` refers to the cumulative amount of IMX that can be deposited. A value of `0` indicated unlimited. + We can just use dummy gateway/gas service addresses if we only want to test the deployment, and not bridging functionality. If wanting to use dummy addresses, any valid Ethereum address can be used here. 4. Run the deploy script. @@ -138,6 +141,7 @@ ROOT_GATEWAY_ADDRESS="0x013459EC3E8Aeced878C5C4bFfe126A366cd19E9" CHILD_GATEWAY_ADDRESS="0xc7B788E88BAaB770A6d4936cdcCcd5250E1bbAd8" ROOT_GAS_SERVICE_ADDRESS="0x28f8B50E1Be6152da35e923602a2641491E71Ed8" CHILD_GAS_SERVICE_ADDRESS="0xC573c722e21eD7fadD38A8f189818433e01Ae466" +INITIAL_IMX_CUMULATIVE_DEPOSIT_LIMIT="0" ENVIRONMENT="local" ``` (Note that `{ROOT,CHILD}_PRIVATE_KEY` can be any of the standard localhost private keys that get funded) diff --git a/script/DeployRootContracts.s.sol b/script/DeployRootContracts.s.sol index 8c04b8b5..0ef5c52b 100644 --- a/script/DeployRootContracts.s.sol +++ b/script/DeployRootContracts.s.sol @@ -37,7 +37,7 @@ contract DeployRootContracts is Script { RootERC20Bridge rootERC20BridgeImplementation = new RootERC20Bridge(); rootERC20BridgeImplementation.initialize( - address(1), address(1), "filler", address(1), address(1), address(1), "filler_child_name" + address(1), address(1), "filler", address(1), address(1), address(1), "filler_child_name", 1 ); TransparentUpgradeableProxy rootERC20BridgeProxy = new TransparentUpgradeableProxy( address(rootERC20BridgeImplementation), @@ -45,10 +45,9 @@ contract DeployRootContracts is Script { "" ); - // TODO add dummy initialize of implementation contracts! - RootAxelarBridgeAdaptor rootBridgeAdaptorImplementation = new RootAxelarBridgeAdaptor(rootGateway); rootBridgeAdaptorImplementation.initialize(address(rootERC20BridgeImplementation), "Filler", address(1)); + TransparentUpgradeableProxy rootBridgeAdaptorProxy = new TransparentUpgradeableProxy( address(rootBridgeAdaptorImplementation), address(proxyAdmin), diff --git a/script/InitializeRootContracts.s.sol b/script/InitializeRootContracts.s.sol index d29ee55c..625345d0 100644 --- a/script/InitializeRootContracts.s.sol +++ b/script/InitializeRootContracts.s.sol @@ -25,6 +25,7 @@ struct InitializeRootContractsParams { address rootWETHToken; string childChainName; address rootGasService; + uint256 initialIMXCumulativeDepositLimit; } contract InitializeRootContracts is Script { @@ -40,7 +41,8 @@ contract InitializeRootContracts is Script { rootIMXToken: vm.envAddress("ROOT_IMX_ADDRESS"), rootWETHToken: vm.envAddress("ROOT_WETH_ADDRESS"), childChainName: vm.envString("CHILD_CHAIN_NAME"), - rootGasService: vm.envAddress("ROOT_GAS_SERVICE_ADDRESS") + rootGasService: vm.envAddress("ROOT_GAS_SERVICE_ADDRESS"), + initialIMXCumulativeDepositLimit: vm.envUint("INITIAL_IMX_CUMULATIVE_DEPOSIT_LIMIT") }); string[] memory checksumInputs = Utils.getChecksumInputs(params.childBridgeAdaptor); @@ -59,7 +61,8 @@ contract InitializeRootContracts is Script { params.rootChainChildTokenTemplate, params.rootIMXToken, params.rootWETHToken, - params.childChainName + params.childChainName, + params.initialIMXCumulativeDepositLimit ); params.rootBridgeAdaptor.initialize( diff --git a/src/interfaces/root/IRootERC20Bridge.sol b/src/interfaces/root/IRootERC20Bridge.sol index e3fe1129..13bfb1e8 100644 --- a/src/interfaces/root/IRootERC20Bridge.sol +++ b/src/interfaces/root/IRootERC20Bridge.sol @@ -43,6 +43,10 @@ interface IRootERC20Bridge { } interface IRootERC20BridgeEvents { + /// @notice Emitted when the child chain bridge adaptor is updated. + event NewRootBridgeAdaptor(address oldRootBridgeAdaptor, address newRootBridgeAdaptor); + /// @notice Emitted when the IMX deposit limit is updated. + event NewImxDepositLimit(uint256 oldImxDepositLimit, uint256 newImxDepositLimit); /// @notice Emitted when a map token message is sent to the child chain. event L1TokenMapped(address indexed rootToken, address indexed childToken); /// @notice Emitted when an ERC20 deposit message is sent to the child chain. @@ -109,4 +113,8 @@ interface IRootERC20BridgeErrors { error InvalidSourceChain(); /// @notice Error when caller is not the root bridge adaptor but should be. error NotBridgeAdaptor(); + /// @notice Error when the total IMX deposit limit is exceeded + error ImxDepositLimitExceeded(); + /// @notice Error when the IMX deposit limit is set below the amount of IMX already deposited + error ImxDepositLimitTooLow(); } diff --git a/src/root/RootERC20Bridge.sol b/src/root/RootERC20Bridge.sol index 5ba1511c..bf72034d 100644 --- a/src/root/RootERC20Bridge.sol +++ b/src/root/RootERC20Bridge.sol @@ -36,6 +36,7 @@ contract RootERC20Bridge is /// @dev leave this as the first param for the integration tests mapping(address => address) public rootTokenToChildToken; + uint256 public constant NO_DEPOSIT_LIMIT = 0; bytes32 public constant MAP_TOKEN_SIG = keccak256("MAP_TOKEN"); bytes32 public constant DEPOSIT_SIG = keccak256("DEPOSIT"); bytes32 public constant WITHDRAW_SIG = keccak256("WITHDRAW"); @@ -56,6 +57,9 @@ contract RootERC20Bridge is address public rootWETHToken; /// @dev The name of the chain that this bridge is connected to. string public childChain; + /// @dev The maximum cumulative amount of IMX that can be deposited into the bridge. + /// @dev A limit of zero indicates unlimited. + uint256 public imxCumulativeDepositLimit; /** * @notice Initilization function for RootERC20Bridge. @@ -66,6 +70,7 @@ contract RootERC20Bridge is * @param newRootIMXToken Address of ERC20 IMX on the root chain. * @param newRootWETHToken Address of ERC20 WETH on the root chain. * @param newChildChain Name of child chain. + * @param newImxCumulativeDepositLimit The cumulative IMX deposit limit. * @dev Can only be called once. */ function initialize( @@ -75,7 +80,8 @@ contract RootERC20Bridge is address newChildTokenTemplate, address newRootIMXToken, address newRootWETHToken, - string memory newChildChain + string memory newChildChain, + uint256 newImxCumulativeDepositLimit ) public initializer { if ( newRootBridgeAdaptor == address(0) || newChildERC20Bridge == address(0) @@ -100,15 +106,41 @@ contract RootERC20Bridge is rootBridgeAdaptor = IRootERC20BridgeAdaptor(newRootBridgeAdaptor); childBridgeAdaptor = newChildBridgeAdaptor; childChain = newChildChain; + imxCumulativeDepositLimit = newImxCumulativeDepositLimit; } + /** + * @notice Updates the root bridge adaptor. + * @param newRootBridgeAdaptor Address of new root bridge adaptor. + * @dev Can only be called by owner. + */ function updateRootBridgeAdaptor(address newRootBridgeAdaptor) external onlyOwner { if (newRootBridgeAdaptor == address(0)) { revert ZeroAddress(); } + emit NewRootBridgeAdaptor(address(rootBridgeAdaptor), newRootBridgeAdaptor); rootBridgeAdaptor = IRootERC20BridgeAdaptor(newRootBridgeAdaptor); } + // TODO add updating of child bridge adaptor. Part of SMR-1908 + + /** + * @notice Updates the IMX deposit limit. + * @param newImxCumulativeDepositLimit The new cumulative IMX deposit limit. + * @dev Can only be called by owner. + * @dev The limit can decrease, but it can never decrease to below the contract's IMX balance. + */ + function updateImxCumulativeDepositLimit(uint256 newImxCumulativeDepositLimit) external onlyOwner { + if ( + newImxCumulativeDepositLimit != NO_DEPOSIT_LIMIT + && newImxCumulativeDepositLimit < IERC20Metadata(rootIMXToken).balanceOf(address(this)) + ) { + revert ImxDepositLimitTooLow(); + } + emit NewImxDepositLimit(imxCumulativeDepositLimit, newImxCumulativeDepositLimit); + imxCumulativeDepositLimit = newImxCumulativeDepositLimit; + } + /** * @dev method to receive the ETH back from the WETH contract when it is unwrapped */ @@ -267,6 +299,12 @@ contract RootERC20Bridge is if (amount == 0) { revert ZeroAmount(); } + if ( + address(rootToken) == rootIMXToken && imxCumulativeDepositLimit != NO_DEPOSIT_LIMIT + && IERC20Metadata(rootIMXToken).balanceOf(address(this)) + amount > imxCumulativeDepositLimit + ) { + revert ImxDepositLimitExceeded(); + } // ETH, WETH and IMX do not need to be mapped since it should have been mapped on initialization // ETH also cannot be transferred since it was received in the payable function call diff --git a/test/integration/child/withdrawals/ChildAxelarBridgeWithdraw.t.sol b/test/integration/child/withdrawals/ChildAxelarBridgeWithdraw.t.sol index c2406f45..e52abd11 100644 --- a/test/integration/child/withdrawals/ChildAxelarBridgeWithdraw.t.sol +++ b/test/integration/child/withdrawals/ChildAxelarBridgeWithdraw.t.sol @@ -138,359 +138,4 @@ contract ChildERC20BridgeWithdrawIntegrationTest is vm.expectRevert(NoGas.selector); childBridge.withdraw(childToken, withdrawAmount); } - - // 7a8dc26796a1e50e6e190b70259f58f6a4edd5b22280ceecc82b687b8e982869 - // 000000000000000000000000000000000000000000000000000000000000ad9c - // 0000000000000000000000007fa9385be102ac3eac297483dd6233d62b3e1496 - // 0000000000000000000000007fa9385be102ac3eac297483dd6233d62b3e1496 - // 000000000000000000000000000000000000000000000000000000174876e7ff - // 0x7FA9385bE102ac3EAc297483Dd6233D62b3e1496 - - // f20755ba - // 0000000000000000000000000000000000000000000000000000000000000040 - // 0000000000000000000000007fa9385be102ac3eac297483dd6233d62b3e1496 - // 00000000000000000000000000000000000000000000000000000000000000a0 - // 2cef46a936bdc5b7e6e8c71aa04560c41cf7d88bb26901a7e7f4936ff02accad - // 000000000000000000000000000000000000000000000000000000000000ad9c - // 0000000000000000000000007fa9385be102ac3eac297483dd6233d62b3e1496 - // 0000000000000000000000007fa9385be102ac3eac297483dd6233d62b3e1496 - // 000000000000000000000000000000000000000000000000000000174876e7ff - - // bytes memory payload = abi.encode(MAP_TOKEN_SIG, address(token), token.name(), token.symbol(), token.decimals()); - // vm.expectEmit(true, true, true, false, address(axelarAdaptor)); - // emit AxelarMessage(CHILD_CHAIN_NAME, Strings.toHexString(CHILD_BRIDGE_ADAPTOR), payload); - - // vm.expectEmit(true, true, false, false, address(rootBridge)); - // emit L1TokenMapped(address(token), childToken); - - // // Instead of using expectCalls, we could use expectEmit in combination with mock contracts emitting events. - // // expectCalls requires less boilerplate and is less dependant on mock code. - // vm.expectCall( - // address(axelarAdaptor), - // mapTokenFee, - // abi.encodeWithSelector(axelarAdaptor.sendMessage.selector, payload, address(this)) - // ); - - // // These are calls that the axelarAdaptor should make. - // vm.expectCall( - // address(axelarGasService), - // mapTokenFee, - // abi.encodeWithSelector( - // axelarGasService.payNativeGasForContractCall.selector, - // address(axelarAdaptor), - // CHILD_CHAIN_NAME, - // Strings.toHexString(CHILD_BRIDGE_ADAPTOR), - // payload, - // address(this) - // ) - // ); - - // vm.expectCall( - // address(mockAxelarGateway), - // 0, - // abi.encodeWithSelector( - // mockAxelarGateway.callContract.selector, - // CHILD_CHAIN_NAME, - // Strings.toHexString(CHILD_BRIDGE_ADAPTOR), - // payload - // ) - // ); - - // // Check that we pay mapTokenFee to the axelarGasService. - // uint256 thisPreBal = address(this).balance; - // uint256 axelarGasServicePreBal = address(axelarGasService).balance; - - // rootBridge.mapToken{value: mapTokenFee}(token); - - // // Should update ETH balances as gas payment for message. - // assertEq(address(this).balance, thisPreBal - mapTokenFee, "ETH balance not decreased"); - // assertEq(address(axelarGasService).balance, axelarGasServicePreBal + mapTokenFee, "ETH not paid to gas service"); - - // assertEq(rootBridge.rootTokenToChildToken(address(token)), childToken, "childToken not set"); - // } - - // // TODO split into multiple tests - // function test_depositETH() public { - // uint256 tokenAmount = 300; - // string memory childBridgeAdaptorString = Strings.toHexString(CHILD_BRIDGE_ADAPTOR); - - // (, bytes memory predictedPayload) = - // setupDeposit(NATIVE_ETH, rootBridge, mapTokenFee, depositFee, tokenAmount, false); - - // console2.logBytes(predictedPayload); - - // vm.expectEmit(address(axelarAdaptor)); - // emit AxelarMessage(CHILD_CHAIN_NAME, childBridgeAdaptorString, predictedPayload); - // vm.expectEmit(address(rootBridge)); - // emit NativeEthDeposit( - // address(NATIVE_ETH), rootBridge.childETHToken(), address(this), address(this), tokenAmount - // ); - - // vm.expectCall( - // address(axelarAdaptor), - // depositFee, - // abi.encodeWithSelector(axelarAdaptor.sendMessage.selector, predictedPayload, address(this)) - // ); - - // vm.expectCall( - // address(axelarGasService), - // depositFee, - // abi.encodeWithSelector( - // axelarGasService.payNativeGasForContractCall.selector, - // address(axelarAdaptor), - // CHILD_CHAIN_NAME, - // childBridgeAdaptorString, - // predictedPayload, - // address(this) - // ) - // ); - - // vm.expectCall( - // address(mockAxelarGateway), - // 0, - // abi.encodeWithSelector( - // mockAxelarGateway.callContract.selector, CHILD_CHAIN_NAME, childBridgeAdaptorString, predictedPayload - // ) - // ); - - // uint256 bridgePreBal = address(rootBridge).balance; - - // uint256 thisNativePreBal = address(this).balance; - // uint256 gasServiceNativePreBal = address(axelarGasService).balance; - - // rootBridge.depositETH{value: tokenAmount + depositFee}(tokenAmount); - - // // Check that tokens are transferred - // assertEq(bridgePreBal + tokenAmount, address(rootBridge).balance, "ETH not transferred to bridge"); - // // Check that native asset transferred to gas service - // assertEq(thisNativePreBal - (depositFee + tokenAmount), address(this).balance, "ETH not paid from user"); - // assertEq(gasServiceNativePreBal + depositFee, address(axelarGasService).balance, "ETH not paid to adaptor"); - // } - - // // TODO split into multiple tests - // function test_depositIMXToken() public { - // uint256 tokenAmount = 300; - // string memory childBridgeAdaptorString = Strings.toHexString(CHILD_BRIDGE_ADAPTOR); - - // (, bytes memory predictedPayload) = - // setupDeposit(IMX_TOKEN_ADDRESS, rootBridge, mapTokenFee, depositFee, tokenAmount, false); - - // vm.expectEmit(address(axelarAdaptor)); - // emit AxelarMessage(CHILD_CHAIN_NAME, childBridgeAdaptorString, predictedPayload); - // vm.expectEmit(address(rootBridge)); - // emit IMXDeposit(address(IMX_TOKEN_ADDRESS), address(this), address(this), tokenAmount); - - // vm.expectCall( - // address(axelarAdaptor), - // depositFee, - // abi.encodeWithSelector(axelarAdaptor.sendMessage.selector, predictedPayload, address(this)) - // ); - - // vm.expectCall( - // address(axelarGasService), - // depositFee, - // abi.encodeWithSelector( - // axelarGasService.payNativeGasForContractCall.selector, - // address(axelarAdaptor), - // CHILD_CHAIN_NAME, - // childBridgeAdaptorString, - // predictedPayload, - // address(this) - // ) - // ); - - // vm.expectCall( - // address(mockAxelarGateway), - // 0, - // abi.encodeWithSelector( - // mockAxelarGateway.callContract.selector, CHILD_CHAIN_NAME, childBridgeAdaptorString, predictedPayload - // ) - // ); - - // uint256 thisPreBal = imxToken.balanceOf(address(this)); - // uint256 bridgePreBal = imxToken.balanceOf(address(rootBridge)); - - // uint256 thisNativePreBal = address(this).balance; - // uint256 gasServiceNativePreBal = address(axelarGasService).balance; - - // rootBridge.deposit{value: depositFee}(IERC20Metadata(IMX_TOKEN_ADDRESS), tokenAmount); - - // // Check that tokens are transferred - // assertEq(thisPreBal - tokenAmount, imxToken.balanceOf(address(this)), "Tokens not transferred from user"); - // assertEq( - // bridgePreBal + tokenAmount, imxToken.balanceOf(address(rootBridge)), "Tokens not transferred to bridge" - // ); - // // Check that native asset transferred to gas service - // assertEq(thisNativePreBal - depositFee, address(this).balance, "ETH not paid from user"); - // assertEq(gasServiceNativePreBal + depositFee, address(axelarGasService).balance, "ETH not paid to adaptor"); - // } - - // // TODO split into multiple tests - // function test_depositWETH() public { - // uint256 tokenAmount = 300; - // string memory childBridgeAdaptorString = Strings.toHexString(CHILD_BRIDGE_ADAPTOR); - // (, bytes memory predictedPayload) = - // setupDeposit(WRAPPED_ETH, rootBridge, mapTokenFee, depositFee, tokenAmount, false); - - // vm.expectEmit(address(axelarAdaptor)); - // emit AxelarMessage(CHILD_CHAIN_NAME, childBridgeAdaptorString, predictedPayload); - // vm.expectEmit(address(rootBridge)); - // emit WETHDeposit(address(WRAPPED_ETH), rootBridge.childETHToken(), address(this), address(this), tokenAmount); - // vm.expectCall( - // address(axelarAdaptor), - // depositFee, - // abi.encodeWithSelector(axelarAdaptor.sendMessage.selector, predictedPayload, address(this)) - // ); - - // vm.expectCall( - // address(axelarGasService), - // depositFee, - // abi.encodeWithSelector( - // axelarGasService.payNativeGasForContractCall.selector, - // address(axelarAdaptor), - // CHILD_CHAIN_NAME, - // childBridgeAdaptorString, - // predictedPayload, - // address(this) - // ) - // ); - - // vm.expectCall( - // address(mockAxelarGateway), - // 0, - // abi.encodeWithSelector( - // mockAxelarGateway.callContract.selector, CHILD_CHAIN_NAME, childBridgeAdaptorString, predictedPayload - // ) - // ); - - // uint256 thisPreBal = IERC20Metadata(WRAPPED_ETH).balanceOf(address(this)); - // uint256 bridgePreBal = address(rootBridge).balance; - - // uint256 thisNativePreBal = address(this).balance; - // uint256 gasServiceNativePreBal = address(axelarGasService).balance; - - // rootBridge.deposit{value: depositFee}(IERC20Metadata(WRAPPED_ETH), tokenAmount); - - // // Check that tokens are transferred - // assertEq( - // thisPreBal - tokenAmount, - // IERC20Metadata(WRAPPED_ETH).balanceOf(address(this)), - // "Tokens not transferred from user" - // ); - // assertEq(bridgePreBal + tokenAmount, address(rootBridge).balance, "ETH not transferred to Bridge"); - // // Check that native asset transferred to gas service - // assertEq(thisNativePreBal - depositFee, address(this).balance, "ETH for fee not paid from user"); - // assertEq(gasServiceNativePreBal + depositFee, address(axelarGasService).balance, "ETH not paid to adaptor"); - // } - - // // TODO split into multiple tests - // function test_depositToken() public { - // uint256 tokenAmount = 300; - // string memory childBridgeAdaptorString = Strings.toHexString(CHILD_BRIDGE_ADAPTOR); - // (address childToken, bytes memory predictedPayload) = - // setupDeposit(address(token), rootBridge, mapTokenFee, depositFee, tokenAmount, true); - - // vm.expectEmit(address(axelarAdaptor)); - // emit AxelarMessage(CHILD_CHAIN_NAME, childBridgeAdaptorString, predictedPayload); - // vm.expectEmit(address(rootBridge)); - // emit ChildChainERC20Deposit(address(token), childToken, address(this), address(this), tokenAmount); - - // vm.expectCall( - // address(axelarAdaptor), - // depositFee, - // abi.encodeWithSelector(axelarAdaptor.sendMessage.selector, predictedPayload, address(this)) - // ); - - // vm.expectCall( - // address(axelarGasService), - // depositFee, - // abi.encodeWithSelector( - // axelarGasService.payNativeGasForContractCall.selector, - // address(axelarAdaptor), - // CHILD_CHAIN_NAME, - // childBridgeAdaptorString, - // predictedPayload, - // address(this) - // ) - // ); - - // vm.expectCall( - // address(mockAxelarGateway), - // 0, - // abi.encodeWithSelector( - // mockAxelarGateway.callContract.selector, CHILD_CHAIN_NAME, childBridgeAdaptorString, predictedPayload - // ) - // ); - - // uint256 thisPreBal = token.balanceOf(address(this)); - // uint256 bridgePreBal = token.balanceOf(address(rootBridge)); - - // uint256 thisNativePreBal = address(this).balance; - // uint256 gasServiceNativePreBal = address(axelarGasService).balance; - - // rootBridge.deposit{value: depositFee}(token, tokenAmount); - - // // Check that tokens are transferred - // assertEq(thisPreBal - tokenAmount, token.balanceOf(address(this)), "Tokens not transferred from user"); - // assertEq(bridgePreBal + tokenAmount, token.balanceOf(address(rootBridge)), "Tokens not transferred to bridge"); - // // Check that native asset transferred to gas service - // assertEq(thisNativePreBal - depositFee, address(this).balance, "ETH not paid from user"); - // assertEq(gasServiceNativePreBal + depositFee, address(axelarGasService).balance, "ETH not paid to adaptor"); - // } - - // // TODO split into multiple tests - // function test_depositTo() public { - // uint256 tokenAmount = 300; - // address recipient = address(9876); - // string memory childBridgeAdaptorString = Strings.toHexString(CHILD_BRIDGE_ADAPTOR); - // (address childToken, bytes memory predictedPayload) = - // setupDepositTo(address(token), rootBridge, mapTokenFee, depositFee, tokenAmount, recipient, true); - - // vm.expectEmit(address(axelarAdaptor)); - // emit AxelarMessage(CHILD_CHAIN_NAME, childBridgeAdaptorString, predictedPayload); - // vm.expectEmit(address(rootBridge)); - // emit ChildChainERC20Deposit(address(token), childToken, address(this), recipient, tokenAmount); - - // vm.expectCall( - // address(axelarAdaptor), - // depositFee, - // abi.encodeWithSelector(axelarAdaptor.sendMessage.selector, predictedPayload, address(this)) - // ); - - // vm.expectCall( - // address(axelarGasService), - // depositFee, - // abi.encodeWithSelector( - // axelarGasService.payNativeGasForContractCall.selector, - // address(axelarAdaptor), - // CHILD_CHAIN_NAME, - // childBridgeAdaptorString, - // predictedPayload, - // address(this) - // ) - // ); - - // vm.expectCall( - // address(mockAxelarGateway), - // 0, - // abi.encodeWithSelector( - // mockAxelarGateway.callContract.selector, CHILD_CHAIN_NAME, childBridgeAdaptorString, predictedPayload - // ) - // ); - - // uint256 thisPreBal = token.balanceOf(address(this)); - // uint256 bridgePreBal = token.balanceOf(address(rootBridge)); - - // uint256 thisNativePreBal = address(this).balance; - // uint256 gasServiceNativePreBal = address(axelarGasService).balance; - - // rootBridge.depositTo{value: depositFee}(token, recipient, tokenAmount); - - // // Check that tokens are transferred - // assertEq(thisPreBal - tokenAmount, token.balanceOf(address(this)), "Tokens not transferred from user"); - // assertEq(bridgePreBal + tokenAmount, token.balanceOf(address(rootBridge)), "Tokens not transferred to bridge"); - // // Check that native asset transferred to gas service - // assertEq(thisNativePreBal - depositFee, address(this).balance, "ETH not paid from user"); - // assertEq(gasServiceNativePreBal + depositFee, address(axelarGasService).balance, "ETH not paid to adaptor"); - // } } diff --git a/test/integration/root/RootERC20Bridge.t.sol b/test/integration/root/RootERC20Bridge.t.sol index e26e762b..9728d595 100644 --- a/test/integration/root/RootERC20Bridge.t.sol +++ b/test/integration/root/RootERC20Bridge.t.sol @@ -19,6 +19,7 @@ contract RootERC20BridgeIntegrationTest is Test, IRootERC20BridgeEvents, IRootAx address constant IMX_TOKEN_ADDRESS = address(0xccc); address constant NATIVE_ETH = address(0xeee); address constant WRAPPED_ETH = address(0xddd); + uint256 constant unlimitedDepositLimit = 0; uint256 constant mapTokenFee = 300; uint256 constant depositFee = 200; @@ -33,8 +34,9 @@ contract RootERC20BridgeIntegrationTest is Test, IRootERC20BridgeEvents, IRootAx function setUp() public { deployCodeTo("WETH.sol", abi.encode("Wrapped ETH", "WETH"), WRAPPED_ETH); - (imxToken, token, rootBridge, axelarAdaptor, mockAxelarGateway, axelarGasService) = - rootIntegrationSetup(CHILD_BRIDGE, CHILD_BRIDGE_ADAPTOR, CHILD_CHAIN_NAME, IMX_TOKEN_ADDRESS, WRAPPED_ETH); + (imxToken, token, rootBridge, axelarAdaptor, mockAxelarGateway, axelarGasService) = rootIntegrationSetup( + CHILD_BRIDGE, CHILD_BRIDGE_ADAPTOR, CHILD_CHAIN_NAME, IMX_TOKEN_ADDRESS, WRAPPED_ETH, unlimitedDepositLimit + ); } /** diff --git a/test/integration/root/withdrawals.t.sol/RootERC20BridgeWithdraw.t.sol b/test/integration/root/withdrawals.t.sol/RootERC20BridgeWithdraw.t.sol index 8e54d0ed..70b0d140 100644 --- a/test/integration/root/withdrawals.t.sol/RootERC20BridgeWithdraw.t.sol +++ b/test/integration/root/withdrawals.t.sol/RootERC20BridgeWithdraw.t.sol @@ -32,6 +32,7 @@ contract RootERC20BridgeWithdrawIntegrationTest is address constant IMX_TOKEN_ADDRESS = address(0xccc); address constant NATIVE_ETH = address(0xeee); address constant WRAPPED_ETH = address(0xddd); + uint256 constant UNLIMITED_IMX_DEPOSIT_LIMIT = 0; uint256 constant withdrawAmount = 0.5 ether; @@ -45,8 +46,14 @@ contract RootERC20BridgeWithdrawIntegrationTest is function setUp() public { deployCodeTo("WETH.sol", abi.encode("Wrapped ETH", "WETH"), WRAPPED_ETH); - (imxToken, token, rootBridge, axelarAdaptor, mockAxelarGateway, axelarGasService) = - rootIntegrationSetup(CHILD_BRIDGE, CHILD_BRIDGE_ADAPTOR, CHILD_CHAIN_NAME, IMX_TOKEN_ADDRESS, WRAPPED_ETH); + (imxToken, token, rootBridge, axelarAdaptor, mockAxelarGateway, axelarGasService) = rootIntegrationSetup( + CHILD_BRIDGE, + CHILD_BRIDGE_ADAPTOR, + CHILD_CHAIN_NAME, + IMX_TOKEN_ADDRESS, + WRAPPED_ETH, + UNLIMITED_IMX_DEPOSIT_LIMIT + ); // Need to first map the token. rootBridge.mapToken{value: 1}(token); diff --git a/test/unit/root/RootERC20Bridge.t.sol b/test/unit/root/RootERC20Bridge.t.sol index 07d316ba..e4c3de3e 100644 --- a/test/unit/root/RootERC20Bridge.t.sol +++ b/test/unit/root/RootERC20Bridge.t.sol @@ -27,6 +27,7 @@ contract RootERC20BridgeUnitTest is Test, IRootERC20BridgeEvents, IRootERC20Brid address constant WRAPPED_ETH = address(0xddd); uint256 constant mapTokenFee = 300; uint256 constant depositFee = 200; + uint256 constant UNLIMITED_IMX_DEPOSITS = 0; ERC20PresetMinterPauser public token; RootERC20Bridge public rootBridge; @@ -54,7 +55,8 @@ contract RootERC20BridgeUnitTest is Test, IRootERC20BridgeEvents, IRootERC20Brid address(token), IMX_TOKEN, WRAPPED_ETH, - CHILD_CHAIN_NAME + CHILD_CHAIN_NAME, + UNLIMITED_IMX_DEPOSITS ); } @@ -82,7 +84,8 @@ contract RootERC20BridgeUnitTest is Test, IRootERC20BridgeEvents, IRootERC20Brid address(token), IMX_TOKEN, WRAPPED_ETH, - CHILD_CHAIN_NAME + CHILD_CHAIN_NAME, + UNLIMITED_IMX_DEPOSITS ); } @@ -90,7 +93,14 @@ contract RootERC20BridgeUnitTest is Test, IRootERC20BridgeEvents, IRootERC20Brid RootERC20Bridge bridge = new RootERC20Bridge(); vm.expectRevert(ZeroAddress.selector); bridge.initialize( - address(0), address(1), CHILD_BRIDGE_ADAPTOR_STRING, address(1), address(1), address(1), CHILD_CHAIN_NAME + address(0), + address(1), + CHILD_BRIDGE_ADAPTOR_STRING, + address(1), + address(1), + address(1), + CHILD_CHAIN_NAME, + UNLIMITED_IMX_DEPOSITS ); } @@ -98,21 +108,37 @@ contract RootERC20BridgeUnitTest is Test, IRootERC20BridgeEvents, IRootERC20Brid RootERC20Bridge bridge = new RootERC20Bridge(); vm.expectRevert(ZeroAddress.selector); bridge.initialize( - address(1), address(0), CHILD_BRIDGE_ADAPTOR_STRING, address(1), address(1), address(1), CHILD_CHAIN_NAME + address(1), + address(0), + CHILD_BRIDGE_ADAPTOR_STRING, + address(1), + address(1), + address(1), + CHILD_CHAIN_NAME, + UNLIMITED_IMX_DEPOSITS ); } function test_RevertIf_InitializeWithEmptyChildAdapter() public { RootERC20Bridge bridge = new RootERC20Bridge(); vm.expectRevert(InvalidChildERC20BridgeAdaptor.selector); - bridge.initialize(address(1), address(1), "", address(1), address(1), address(1), CHILD_CHAIN_NAME); + bridge.initialize( + address(1), address(1), "", address(1), address(1), address(1), CHILD_CHAIN_NAME, UNLIMITED_IMX_DEPOSITS + ); } function test_RevertIf_InitializeWithAZeroAddressTokenTemplate() public { RootERC20Bridge bridge = new RootERC20Bridge(); vm.expectRevert(ZeroAddress.selector); bridge.initialize( - address(1), address(1), CHILD_BRIDGE_ADAPTOR_STRING, address(0), address(1), address(1), CHILD_CHAIN_NAME + address(1), + address(1), + CHILD_BRIDGE_ADAPTOR_STRING, + address(0), + address(1), + address(1), + CHILD_CHAIN_NAME, + UNLIMITED_IMX_DEPOSITS ); } @@ -120,7 +146,14 @@ contract RootERC20BridgeUnitTest is Test, IRootERC20BridgeEvents, IRootERC20Brid RootERC20Bridge bridge = new RootERC20Bridge(); vm.expectRevert(ZeroAddress.selector); bridge.initialize( - address(1), address(1), CHILD_BRIDGE_ADAPTOR_STRING, address(1), address(0), address(1), CHILD_CHAIN_NAME + address(1), + address(1), + CHILD_BRIDGE_ADAPTOR_STRING, + address(1), + address(0), + address(1), + CHILD_CHAIN_NAME, + UNLIMITED_IMX_DEPOSITS ); } @@ -128,20 +161,59 @@ contract RootERC20BridgeUnitTest is Test, IRootERC20BridgeEvents, IRootERC20Brid RootERC20Bridge bridge = new RootERC20Bridge(); vm.expectRevert(ZeroAddress.selector); bridge.initialize( - address(1), address(1), CHILD_BRIDGE_ADAPTOR_STRING, address(1), address(1), address(0), CHILD_CHAIN_NAME + address(1), + address(1), + CHILD_BRIDGE_ADAPTOR_STRING, + address(1), + address(1), + address(0), + CHILD_CHAIN_NAME, + UNLIMITED_IMX_DEPOSITS ); } function test_RevertIf_InitializeWithAZeroAddressAll() public { RootERC20Bridge bridge = new RootERC20Bridge(); vm.expectRevert(ZeroAddress.selector); - bridge.initialize(address(0), address(0), "", address(0), address(0), address(0), CHILD_CHAIN_NAME); + bridge.initialize( + address(0), address(0), "", address(0), address(0), address(0), CHILD_CHAIN_NAME, UNLIMITED_IMX_DEPOSITS + ); } function test_RevertIf_InitializeWithEmptyChildName() public { RootERC20Bridge bridge = new RootERC20Bridge(); vm.expectRevert(InvalidChildChain.selector); - bridge.initialize(address(1), address(1), CHILD_BRIDGE_ADAPTOR_STRING, address(1), address(1), address(1), ""); + bridge.initialize( + address(1), + address(1), + CHILD_BRIDGE_ADAPTOR_STRING, + address(1), + address(1), + address(1), + "", + UNLIMITED_IMX_DEPOSITS + ); + } + + /** + * UPDATE IMX CUMULATIVE DEPOSIT LIMIT + */ + function test_RevertsIf_IMXDepositLimitTooLow() public { + uint256 imxCumulativeDepositLimit = 700; + uint256 depositAmount = imxCumulativeDepositLimit + 1; + + rootBridge.updateImxCumulativeDepositLimit(imxCumulativeDepositLimit); + + setupDeposit(IMX_TOKEN, rootBridge, mapTokenFee, depositFee, depositAmount, false); + + IERC20Metadata(IMX_TOKEN).approve(address(rootBridge), type(uint256).max); + + rootBridge.updateImxCumulativeDepositLimit(depositAmount); + + rootBridge.deposit{value: depositFee}(IERC20Metadata(IMX_TOKEN), depositAmount); + + vm.expectRevert(ImxDepositLimitTooLow.selector); + rootBridge.updateImxCumulativeDepositLimit(imxCumulativeDepositLimit); } /** @@ -221,7 +293,7 @@ contract RootERC20BridgeUnitTest is Test, IRootERC20BridgeEvents, IRootERC20Brid rootBridge.mapToken{value: 300}(IERC20Metadata(NATIVE_ETH)); } - function test_updateRootBridgeAdaptor() public { + function test_updateRootBridgeAdaptor_UpdatesRootBridgeAdaptor() public { address newAdaptorAddress = address(0x11111); assertEq(address(rootBridge.rootBridgeAdaptor()), address(mockAxelarAdaptor), "bridgeAdaptor not set"); @@ -229,6 +301,15 @@ contract RootERC20BridgeUnitTest is Test, IRootERC20BridgeEvents, IRootERC20Brid assertEq(address(rootBridge.rootBridgeAdaptor()), newAdaptorAddress, "bridgeAdaptor not updated"); } + function test_updateRootBridgeAdaptor_EmitsNewRootBridgeAdaptorEvent() public { + address newAdaptorAddress = address(0x11111); + + vm.expectEmit(); + emit NewRootBridgeAdaptor(address(rootBridge.rootBridgeAdaptor()), newAdaptorAddress); + + rootBridge.updateRootBridgeAdaptor(newAdaptorAddress); + } + function test_RevertIf_updateRootBridgeAdaptorCalledByNonOwner() public { vm.prank(address(0xf00f00)); vm.expectRevert("Ownable: caller is not the owner"); @@ -429,6 +510,74 @@ contract RootERC20BridgeUnitTest is Test, IRootERC20BridgeEvents, IRootERC20Brid * DEPOSIT TOKEN */ + function test_RevertsIf_IMXDepositLimitExceeded() public { + uint256 imxCumulativeDepositLimit = 700; + + uint256 amount = 300; + setupDeposit(IMX_TOKEN, rootBridge, mapTokenFee, depositFee, amount, false); + + IERC20Metadata(IMX_TOKEN).approve(address(rootBridge), type(uint256).max); + + rootBridge.updateImxCumulativeDepositLimit(imxCumulativeDepositLimit); + + // Valid + rootBridge.deposit{value: depositFee}(IERC20Metadata(IMX_TOKEN), amount); + + setupDeposit(IMX_TOKEN, rootBridge, mapTokenFee, depositFee, amount, false); + // Valid + rootBridge.deposit{value: depositFee}(IERC20Metadata(IMX_TOKEN), amount); + + setupDeposit(IMX_TOKEN, rootBridge, mapTokenFee, depositFee, amount, false); + // Invalid + vm.expectRevert(ImxDepositLimitExceeded.selector); + rootBridge.deposit{value: depositFee}(IERC20Metadata(IMX_TOKEN), amount); + } + + function test_deposit_whenSettingImxDepositLimitToUnlimited() public { + uint256 imxCumulativeDepositLimit = 700; + + uint256 amount = 300; + setupDeposit(IMX_TOKEN, rootBridge, mapTokenFee, depositFee, amount, false); + + IERC20Metadata(IMX_TOKEN).approve(address(rootBridge), type(uint256).max); + + rootBridge.updateImxCumulativeDepositLimit(imxCumulativeDepositLimit); + + // Valid + rootBridge.deposit{value: depositFee}(IERC20Metadata(IMX_TOKEN), amount); + + setupDeposit(IMX_TOKEN, rootBridge, mapTokenFee, depositFee, amount, false); + // Valid + rootBridge.deposit{value: depositFee}(IERC20Metadata(IMX_TOKEN), amount); + + setupDeposit(IMX_TOKEN, rootBridge, mapTokenFee, depositFee, amount, false); + // Invalid + vm.expectRevert(ImxDepositLimitExceeded.selector); + rootBridge.deposit{value: depositFee}(IERC20Metadata(IMX_TOKEN), amount); + + rootBridge.updateImxCumulativeDepositLimit(UNLIMITED_IMX_DEPOSITS); + + uint256 bigDepositAmount = 999999999999 ether; + setupDeposit(IMX_TOKEN, rootBridge, mapTokenFee, depositFee, bigDepositAmount, false); + + uint256 thisPreBal = IERC20Metadata(IMX_TOKEN).balanceOf(address(this)); + uint256 bridgePreBal = IERC20Metadata(IMX_TOKEN).balanceOf(address(rootBridge)); + + rootBridge.deposit{value: depositFee}(IERC20Metadata(IMX_TOKEN), bigDepositAmount); + + // Check that tokens are transferred + assertEq( + thisPreBal - bigDepositAmount, + IERC20Metadata(IMX_TOKEN).balanceOf(address(this)), + "Tokens not transferred from user" + ); + assertEq( + bridgePreBal + bigDepositAmount, + IERC20Metadata(IMX_TOKEN).balanceOf(address(rootBridge)), + "Tokens not transferred to bridge" + ); + } + function test_depositCallsSendMessage() public { uint256 amount = 100; (, bytes memory predictedPayload) = diff --git a/test/unit/root/withdrawals/RootERC20BridgeWithdraw.t.sol b/test/unit/root/withdrawals/RootERC20BridgeWithdraw.t.sol index bc1c1cba..5ea3c0e4 100644 --- a/test/unit/root/withdrawals/RootERC20BridgeWithdraw.t.sol +++ b/test/unit/root/withdrawals/RootERC20BridgeWithdraw.t.sol @@ -21,6 +21,7 @@ contract RootERC20BridgeWithdrawUnitTest is Test, IRootERC20BridgeEvents, IRootE address constant WRAPPED_ETH = address(0xddd); uint256 constant mapTokenFee = 300; uint256 constant withdrawAmount = 0.5 ether; + uint256 constant UNLIMITED_IMX_DEPOSIT_LIMIT = 0; ERC20PresetMinterPauser public token; RootERC20Bridge public rootBridge; @@ -49,7 +50,8 @@ contract RootERC20BridgeWithdrawUnitTest is Test, IRootERC20BridgeEvents, IRootE address(token), IMX_TOKEN, WRAPPED_ETH, - CHILD_CHAIN_NAME + CHILD_CHAIN_NAME, + UNLIMITED_IMX_DEPOSIT_LIMIT ); } diff --git a/test/utils.t.sol b/test/utils.t.sol index 9e9b86ce..fb3b0c2e 100644 --- a/test/utils.t.sol +++ b/test/utils.t.sol @@ -59,7 +59,8 @@ contract Utils is Test { address childBridgeAdaptor, string memory childBridgeName, address imxTokenAddress, - address wethTokenAddress + address wethTokenAddress, + uint256 imxCumulativeDepositLimit ) public returns ( @@ -95,7 +96,8 @@ contract Utils is Test { address(token), imxTokenAddress, wethTokenAddress, - "CHILD" + "CHILD", + imxCumulativeDepositLimit ); axelarAdaptor.initialize(address(rootBridge), childBridgeName, address(axelarGasService));