Skip to content

Commit c986aa5

Browse files
committed
feat: update fill logic
1 parent ff6e78f commit c986aa5

File tree

3 files changed

+105
-39
lines changed

3 files changed

+105
-39
lines changed

.gas-snapshot

+15-10
Original file line numberDiff line numberDiff line change
@@ -1,13 +1,18 @@
1-
OrdersTest:test_initiate_ERC20() (gas: 81255)
2-
OrdersTest:test_initiate_ETH() (gas: 44801)
3-
OrdersTest:test_initiate_both() (gas: 118573)
4-
OrdersTest:test_initiate_multiERC20() (gas: 688314)
5-
OrdersTest:test_initiate_multiETH() (gas: 75288)
6-
OrdersTest:test_initiate_underflowETH() (gas: 63455)
7-
OrdersTest:test_onlyBuilder() (gas: 12793)
8-
OrdersTest:test_orderExpired() (gas: 27993)
9-
OrdersTest:test_sweep_ERC20() (gas: 60250)
10-
OrdersTest:test_sweep_ETH() (gas: 81788)
1+
OrdersTest:test_fill_ERC20() (gas: 70364)
2+
OrdersTest:test_fill_ETH() (gas: 68414)
3+
OrdersTest:test_fill_both() (gas: 166580)
4+
OrdersTest:test_fill_multiETH() (gas: 131926)
5+
OrdersTest:test_fill_underflowETH() (gas: 115281)
6+
OrdersTest:test_initiate_ERC20() (gas: 81435)
7+
OrdersTest:test_initiate_ETH() (gas: 44949)
8+
OrdersTest:test_initiate_both() (gas: 118677)
9+
OrdersTest:test_initiate_multiERC20() (gas: 688417)
10+
OrdersTest:test_initiate_multiETH() (gas: 75304)
11+
OrdersTest:test_onlyBuilder() (gas: 12815)
12+
OrdersTest:test_orderExpired() (gas: 27956)
13+
OrdersTest:test_sweepERC20() (gas: 60402)
14+
OrdersTest:test_sweepETH() (gas: 81940)
15+
OrdersTest:test_underflowETH() (gas: 63528)
1116
PassageTest:test_configureEnter() (gas: 82311)
1217
PassageTest:test_disallowedEnter() (gas: 17916)
1318
PassageTest:test_enter() (gas: 25563)

src/Orders.sol

+26-24
Original file line numberDiff line numberDiff line change
@@ -22,36 +22,37 @@ struct Output {
2222
uint256 amount;
2323
/// @dev The address to receive the output tokens
2424
address recipient;
25-
/// @dev The destination chain for this output
25+
/// @dev When emitted on the origin chain, the destination chain for the Output.
26+
/// When emitted on the destination chain, the origin chain for the Order containing the Output.
2627
uint32 chainId;
2728
}
2829

2930
/// @notice Contract capable of processing fulfillment of intent-based Orders.
3031
abstract contract OrderDestination {
31-
/// @notice Emitted when an Order's Output is sent to the recipient.
32-
/// @dev There may be multiple Outputs per Order.
33-
/// @param originChainId - The chainId on which the Order was initiated.
34-
/// @param recipient - The recipient of the token.
35-
/// @param token - The address of the token transferred to the recipient. address(0) corresponds to native Ether.
36-
/// @param amount - The amount of the token transferred to the recipient.
37-
event OutputFilled(uint256 indexed originChainId, address indexed recipient, address indexed token, uint256 amount);
38-
39-
/// @notice Send the Output(s) of an Order to fulfill it.
40-
/// The user calls `initiate` on a rollup; the Builder calls `fill` on the target chain for each Output.
41-
/// @custom:emits OutputFilled
42-
/// @param originChainId - The chainId on which the Order was initiated.
43-
/// @param recipient - The recipient of the token.
44-
/// @param token - The address of the token to be transferred to the recipient.
45-
/// address(0) corresponds to native Ether.
46-
/// @param amount - The amount of the token to be transferred to the recipient.
47-
function fill(uint256 originChainId, address recipient, address token, uint256 amount) external payable {
48-
if (token == address(0)) {
49-
require(amount == msg.value);
50-
payable(recipient).transfer(msg.value);
51-
} else {
52-
IERC20(token).transferFrom(msg.sender, recipient, amount);
32+
/// @notice Emitted when Order Outputs are sent to their recipients.
33+
/// @param outputs - The Outputs transferred to their recipients
34+
/// @dev NOTE that here, Output.chainId denotes the *origin* chainId.
35+
event Filled(Output[] outputs);
36+
37+
/// @notice Send the Output(s) of any number of Orders.
38+
/// The user calls `initiate` on a rollup; the Builder calls `fill` on the target chain aggregating Outputs.
39+
/// @param outputs - The Outputs to be transferred.
40+
/// @dev NOTE that here, Output.chainId denotes the *origin* chainId.
41+
/// @custom:emits Filled
42+
function fill(Output[] memory outputs) external payable {
43+
// transfer outputs
44+
uint256 value = msg.value;
45+
for (uint256 i; i < outputs.length; i++) {
46+
if (outputs[i].token == address(0)) {
47+
// this line should underflow if there's an attempt to spend more ETH than is attached to the transaction
48+
value -= outputs[i].amount;
49+
payable(outputs[i].recipient).transfer(outputs[i].amount);
50+
} else {
51+
IERC20(outputs[i].token).transferFrom(msg.sender, outputs[i].recipient, outputs[i].amount);
52+
}
5353
}
54-
emit OutputFilled(originChainId, recipient, token, amount);
54+
// emit
55+
emit Filled(outputs);
5556
}
5657
}
5758

@@ -64,6 +65,7 @@ abstract contract OrderOrigin {
6465
error OnlyBuilder();
6566

6667
/// @notice Emitted when an Order is submitted for fulfillment.
68+
/// @dev NOTE that here, Output.chainId denotes the *destination* chainId.
6769
event Order(uint256 deadline, Input[] inputs, Output[] outputs);
6870

6971
/// @notice Emitted when tokens or native Ether is swept from the contract.

test/Orders.t.sol

+64-5
Original file line numberDiff line numberDiff line change
@@ -19,7 +19,7 @@ contract OrdersTest is Test {
1919
uint256 amount = 200;
2020
uint256 deadline = block.timestamp;
2121

22-
event OutputFilled(uint256 indexed originChainId, address indexed recipient, address indexed token, uint256 amount);
22+
event Filled(Output[] outputs);
2323

2424
event Order(uint256 deadline, Input[] inputs, Output[] outputs);
2525

@@ -122,13 +122,13 @@ contract OrdersTest is Test {
122122
assertEq(address(target).balance, amount * 3);
123123
}
124124

125-
function test_initiate_underflowETH() public {
125+
function test_underflowETH() public {
126126
// change first input to ETH
127127
inputs[0].token = address(0);
128128
// add second ETH input
129129
inputs.push(Input(address(0), 1));
130130

131-
// total ETH inputs should be `amount` + 1; function should underflow only sending `amount`
131+
// total ETH inputs should be amount + 1; function should underflow only sending amount
132132
vm.expectRevert();
133133
target.initiate{value: amount}(deadline, inputs, outputs);
134134
}
@@ -140,7 +140,7 @@ contract OrdersTest is Test {
140140
target.initiate(deadline, inputs, outputs);
141141
}
142142

143-
function test_sweep_ETH() public {
143+
function test_sweepETH() public {
144144
// set self as Builder
145145
vm.coinbase(address(this));
146146

@@ -158,7 +158,7 @@ contract OrdersTest is Test {
158158
assertEq(recipient.balance, amount);
159159
}
160160

161-
function test_sweep_ERC20() public {
161+
function test_sweepERC20() public {
162162
// set self as Builder
163163
vm.coinbase(address(this));
164164

@@ -176,4 +176,63 @@ contract OrdersTest is Test {
176176
vm.expectRevert(OrderOrigin.OnlyBuilder.selector);
177177
target.sweep(recipient, token);
178178
}
179+
180+
function test_fill_ETH() public {
181+
outputs[0].token = address(0);
182+
183+
vm.expectEmit();
184+
emit Filled(outputs);
185+
target.fill{value: amount}(outputs);
186+
187+
// ETH is transferred to recipient
188+
assertEq(recipient.balance, amount);
189+
}
190+
191+
function test_fill_ERC20() public {
192+
vm.expectEmit();
193+
emit Filled(outputs);
194+
vm.expectCall(token, abi.encodeWithSelector(ERC20.transferFrom.selector, address(this), recipient, amount));
195+
target.fill(outputs);
196+
}
197+
198+
function test_fill_both() public {
199+
// add ETH output
200+
outputs.push(Output(address(0), amount * 2, recipient, chainId));
201+
202+
// expect Outputs are filled, ERC20 is transferred
203+
vm.expectEmit();
204+
emit Filled(outputs);
205+
vm.expectCall(token, abi.encodeWithSelector(ERC20.transferFrom.selector, address(this), recipient, amount));
206+
target.fill{value: amount * 2}(outputs);
207+
208+
// ETH is transferred to recipient
209+
assertEq(recipient.balance, amount * 2);
210+
}
211+
212+
// fill multiple ETH outputs
213+
function test_fill_multiETH() public {
214+
// change first output to ETH
215+
outputs[0].token = address(0);
216+
// add second ETH oputput
217+
outputs.push(Output(address(0), amount * 2, recipient, chainId));
218+
219+
// expect Order event is initiated
220+
vm.expectEmit();
221+
emit Filled(outputs);
222+
target.fill{value: amount * 3}(outputs);
223+
224+
// ETH is transferred to recipient
225+
assertEq(recipient.balance, amount * 3);
226+
}
227+
228+
function test_fill_underflowETH() public {
229+
// change first output to ETH
230+
outputs[0].token = address(0);
231+
// add second ETH output
232+
outputs.push(Output(address(0), 1, recipient, chainId));
233+
234+
// total ETH outputs should be `amount` + 1; function should underflow only sending `amount`
235+
vm.expectRevert();
236+
target.fill{value: amount}(outputs);
237+
}
179238
}

0 commit comments

Comments
 (0)