ERC2771Forwarder.t.sol 5.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165
  1. // SPDX-License-Identifier: MIT
  2. pragma solidity ^0.8.20;
  3. import {Test} from "forge-std/Test.sol";
  4. import {ERC2771Forwarder} from "@openzeppelin/contracts/metatx/ERC2771Forwarder.sol";
  5. import {CallReceiverMockTrustingForwarder, CallReceiverMock} from "@openzeppelin/contracts/mocks/CallReceiverMock.sol";
  6. struct ForwardRequest {
  7. address from;
  8. address to;
  9. uint256 value;
  10. uint256 gas;
  11. uint256 nonce;
  12. uint48 deadline;
  13. bytes data;
  14. }
  15. contract ERC2771ForwarderMock is ERC2771Forwarder {
  16. constructor(string memory name) ERC2771Forwarder(name) {}
  17. function structHash(ForwardRequest calldata request) external view returns (bytes32) {
  18. return
  19. _hashTypedDataV4(
  20. keccak256(
  21. abi.encode(
  22. _FORWARD_REQUEST_TYPEHASH,
  23. request.from,
  24. request.to,
  25. request.value,
  26. request.gas,
  27. request.nonce,
  28. request.deadline,
  29. keccak256(request.data)
  30. )
  31. )
  32. );
  33. }
  34. }
  35. contract ERC2771ForwarderTest is Test {
  36. ERC2771ForwarderMock internal _erc2771Forwarder;
  37. CallReceiverMockTrustingForwarder internal _receiver;
  38. uint256 internal _signerPrivateKey;
  39. uint256 internal _relayerPrivateKey;
  40. address internal _signer;
  41. address internal _relayer;
  42. uint256 internal constant _MAX_ETHER = 10_000_000; // To avoid overflow
  43. function setUp() public {
  44. _erc2771Forwarder = new ERC2771ForwarderMock("ERC2771Forwarder");
  45. _receiver = new CallReceiverMockTrustingForwarder(address(_erc2771Forwarder));
  46. _signerPrivateKey = 0xA11CE;
  47. _relayerPrivateKey = 0xB0B;
  48. _signer = vm.addr(_signerPrivateKey);
  49. _relayer = vm.addr(_relayerPrivateKey);
  50. }
  51. function _forgeRequestData(
  52. uint256 value,
  53. uint256 nonce,
  54. uint48 deadline,
  55. bytes memory data
  56. ) private view returns (ERC2771Forwarder.ForwardRequestData memory) {
  57. ForwardRequest memory request = ForwardRequest({
  58. from: _signer,
  59. to: address(_receiver),
  60. value: value,
  61. gas: 30000,
  62. nonce: nonce,
  63. deadline: deadline,
  64. data: data
  65. });
  66. bytes32 digest = _erc2771Forwarder.structHash(request);
  67. (uint8 v, bytes32 r, bytes32 s) = vm.sign(_signerPrivateKey, digest);
  68. bytes memory signature = abi.encodePacked(r, s, v);
  69. return
  70. ERC2771Forwarder.ForwardRequestData({
  71. from: request.from,
  72. to: request.to,
  73. value: request.value,
  74. gas: request.gas,
  75. deadline: request.deadline,
  76. data: request.data,
  77. signature: signature
  78. });
  79. }
  80. function testExecuteAvoidsETHStuck(uint256 initialBalance, uint256 value, bool targetReverts) public {
  81. initialBalance = bound(initialBalance, 0, _MAX_ETHER);
  82. value = bound(value, 0, _MAX_ETHER);
  83. vm.deal(address(_erc2771Forwarder), initialBalance);
  84. uint256 nonce = _erc2771Forwarder.nonces(_signer);
  85. vm.deal(address(this), value);
  86. ERC2771Forwarder.ForwardRequestData memory requestData = _forgeRequestData({
  87. value: value,
  88. nonce: nonce,
  89. deadline: uint48(block.timestamp + 1),
  90. data: targetReverts
  91. ? abi.encodeCall(CallReceiverMock.mockFunctionRevertsNoReason, ())
  92. : abi.encodeCall(CallReceiverMock.mockFunction, ())
  93. });
  94. if (targetReverts) {
  95. vm.expectRevert();
  96. }
  97. _erc2771Forwarder.execute{value: value}(requestData);
  98. assertEq(address(_erc2771Forwarder).balance, initialBalance);
  99. }
  100. function testExecuteBatchAvoidsETHStuck(uint256 initialBalance, uint256 batchSize, uint256 value) public {
  101. batchSize = bound(batchSize, 1, 10);
  102. initialBalance = bound(initialBalance, 0, _MAX_ETHER);
  103. value = bound(value, 0, _MAX_ETHER);
  104. vm.deal(address(_erc2771Forwarder), initialBalance);
  105. uint256 nonce = _erc2771Forwarder.nonces(_signer);
  106. ERC2771Forwarder.ForwardRequestData[] memory batchRequestDatas = new ERC2771Forwarder.ForwardRequestData[](
  107. batchSize
  108. );
  109. uint256 expectedRefund;
  110. for (uint256 i = 0; i < batchSize; ++i) {
  111. bytes memory data;
  112. bool succeed = uint256(keccak256(abi.encodePacked(initialBalance, i))) % 2 == 0;
  113. if (succeed) {
  114. data = abi.encodeCall(CallReceiverMock.mockFunction, ());
  115. } else {
  116. expectedRefund += value;
  117. data = abi.encodeCall(CallReceiverMock.mockFunctionRevertsNoReason, ());
  118. }
  119. batchRequestDatas[i] = _forgeRequestData({
  120. value: value,
  121. nonce: nonce + i,
  122. deadline: uint48(block.timestamp + 1),
  123. data: data
  124. });
  125. }
  126. address payable refundReceiver = payable(address(0xebe));
  127. uint256 totalValue = value * batchSize;
  128. vm.deal(address(this), totalValue);
  129. _erc2771Forwarder.executeBatch{value: totalValue}(batchRequestDatas, refundReceiver);
  130. assertEq(address(_erc2771Forwarder).balance, initialBalance);
  131. assertEq(refundReceiver.balance, expectedRefund);
  132. }
  133. }