Skip to content

Commit 95f9ecc

Browse files
authored
fix: improve error handling and messages (#205, #206) (#473)
1 parent c151ad8 commit 95f9ecc

File tree

57 files changed

+989
-193
lines changed

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

57 files changed

+989
-193
lines changed

.gitignore

+1
Original file line numberDiff line numberDiff line change
@@ -19,6 +19,7 @@ tsconfig.tsbuildinfo
1919
# Misc
2020
.env
2121
.DS_Store
22+
.idea
2223
.secrets
2324
.wallet.json
2425

contracts/evm/GatewayEVM.sol

+20-11
Original file line numberDiff line numberDiff line change
@@ -179,8 +179,8 @@ contract GatewayEVM is
179179
if (amount == 0) revert InsufficientERC20Amount();
180180
if (to == address(0)) revert ZeroAddress();
181181
// Approve the target contract to spend the tokens
182-
if (!_resetApproval(token, to)) revert ApprovalFailed();
183-
if (!IERC20(token).approve(to, amount)) revert ApprovalFailed();
182+
if (!_resetApproval(token, to)) revert ApprovalFailed(token, to);
183+
if (!IERC20(token).approve(to, amount)) revert ApprovalFailed(token, to);
184184
// Execute the call on the target contract
185185
// if sender is provided in messageContext call is authenticated and target is Callable.onCall
186186
// otherwise, call is arbitrary
@@ -191,7 +191,7 @@ contract GatewayEVM is
191191
}
192192

193193
// Reset approval
194-
if (!_resetApproval(token, to)) revert ApprovalFailed();
194+
if (!_resetApproval(token, to)) revert ApprovalFailed(token, to);
195195

196196
// Transfer any remaining tokens back to the custody/connector contract
197197
uint256 remainingBalance = IERC20(token).balanceOf(address(this));
@@ -236,7 +236,9 @@ contract GatewayEVM is
236236
function deposit(address receiver, RevertOptions calldata revertOptions) external payable whenNotPaused {
237237
if (msg.value == 0) revert InsufficientETHAmount();
238238
if (receiver == address(0)) revert ZeroAddress();
239-
if (revertOptions.revertMessage.length > MAX_PAYLOAD_SIZE) revert PayloadSizeExceeded();
239+
if (revertOptions.revertMessage.length > MAX_PAYLOAD_SIZE) {
240+
revert PayloadSizeExceeded(revertOptions.revertMessage.length, MAX_PAYLOAD_SIZE);
241+
}
240242

241243
(bool deposited,) = tssAddress.call{ value: msg.value }("");
242244

@@ -261,7 +263,9 @@ contract GatewayEVM is
261263
{
262264
if (amount == 0) revert InsufficientERC20Amount();
263265
if (receiver == address(0)) revert ZeroAddress();
264-
if (revertOptions.revertMessage.length > MAX_PAYLOAD_SIZE) revert PayloadSizeExceeded();
266+
if (revertOptions.revertMessage.length > MAX_PAYLOAD_SIZE) {
267+
revert PayloadSizeExceeded(revertOptions.revertMessage.length, MAX_PAYLOAD_SIZE);
268+
}
265269

266270
_transferFromToAssetHandler(msg.sender, asset, amount);
267271

@@ -283,7 +287,9 @@ contract GatewayEVM is
283287
{
284288
if (msg.value == 0) revert InsufficientETHAmount();
285289
if (receiver == address(0)) revert ZeroAddress();
286-
if (payload.length + revertOptions.revertMessage.length > MAX_PAYLOAD_SIZE) revert PayloadSizeExceeded();
290+
if (payload.length + revertOptions.revertMessage.length > MAX_PAYLOAD_SIZE) {
291+
revert PayloadSizeExceeded(payload.length + revertOptions.revertMessage.length, MAX_PAYLOAD_SIZE);
292+
}
287293

288294
(bool deposited,) = tssAddress.call{ value: msg.value }("");
289295

@@ -310,7 +316,9 @@ contract GatewayEVM is
310316
{
311317
if (amount == 0) revert InsufficientERC20Amount();
312318
if (receiver == address(0)) revert ZeroAddress();
313-
if (payload.length + revertOptions.revertMessage.length > MAX_PAYLOAD_SIZE) revert PayloadSizeExceeded();
319+
if (payload.length + revertOptions.revertMessage.length > MAX_PAYLOAD_SIZE) {
320+
revert PayloadSizeExceeded(payload.length + revertOptions.revertMessage.length, MAX_PAYLOAD_SIZE);
321+
}
314322

315323
_transferFromToAssetHandler(msg.sender, asset, amount);
316324

@@ -331,7 +339,8 @@ contract GatewayEVM is
331339
{
332340
if (revertOptions.callOnRevert) revert CallOnRevertNotSupported();
333341
if (receiver == address(0)) revert ZeroAddress();
334-
if (payload.length + revertOptions.revertMessage.length > MAX_PAYLOAD_SIZE) revert PayloadSizeExceeded();
342+
uint256 payloadSize = payload.length + revertOptions.revertMessage.length;
343+
if (payloadSize > MAX_PAYLOAD_SIZE) revert PayloadSizeExceeded(payloadSize, MAX_PAYLOAD_SIZE);
335344

336345
emit Called(msg.sender, receiver, payload, revertOptions);
337346
}
@@ -387,7 +396,7 @@ contract GatewayEVM is
387396
// ZetaConnectorBase(zetaConnector).receiveTokens(amount);
388397
} else {
389398
// transfer to custody
390-
if (!IERC20Custody(custody).whitelisted(token)) revert NotWhitelistedInCustody();
399+
if (!IERC20Custody(custody).whitelisted(token)) revert NotWhitelistedInCustody(token);
391400
IERC20(token).safeTransferFrom(from, custody, amount);
392401
}
393402
}
@@ -401,12 +410,12 @@ contract GatewayEVM is
401410
if (token == zetaToken) {
402411
// transfer to connector
403412
// approve connector to handle tokens depending on connector version (eg. lock or burn)
404-
if (!IERC20(token).approve(zetaConnector, amount)) revert ApprovalFailed();
413+
if (!IERC20(token).approve(zetaConnector, amount)) revert ApprovalFailed(token, zetaConnector);
405414
// send tokens to connector
406415
ZetaConnectorBase(zetaConnector).receiveTokens(amount);
407416
} else {
408417
// transfer to custody
409-
if (!IERC20Custody(custody).whitelisted(token)) revert NotWhitelistedInCustody();
418+
if (!IERC20Custody(custody).whitelisted(token)) revert NotWhitelistedInCustody(token);
410419
IERC20(token).safeTransfer(custody, amount);
411420
}
412421
}

contracts/evm/interfaces/IGatewayEVM.sol

+8-3
Original file line numberDiff line numberDiff line change
@@ -91,7 +91,9 @@ interface IGatewayEVMErrors {
9191
error ZeroAddress();
9292

9393
/// @notice Error for failed token approval.
94-
error ApprovalFailed();
94+
/// @param token The address of the token for which approval failed.
95+
/// @param spender The address that was supposed to be approved to spend the tokens.
96+
error ApprovalFailed(address token, address spender);
9597

9698
/// @notice Error for already initialized custody.
9799
error CustodyInitialized();
@@ -100,7 +102,8 @@ interface IGatewayEVMErrors {
100102
error ConnectorInitialized();
101103

102104
/// @notice Error when trying to transfer not whitelisted token to custody.
103-
error NotWhitelistedInCustody();
105+
/// @param token The address of the token that is not whitelisted in custody.
106+
error NotWhitelistedInCustody(address token);
104107

105108
/// @notice Error when trying to call onCall method using arbitrary call.
106109
error NotAllowedToCallOnCall();
@@ -109,7 +112,9 @@ interface IGatewayEVMErrors {
109112
error NotAllowedToCallOnRevert();
110113

111114
/// @notice Error indicating payload size exceeded in external functions.
112-
error PayloadSizeExceeded();
115+
/// @param provided The size of the payload that was provided.
116+
/// @param maximum The maximum allowed payload size.
117+
error PayloadSizeExceeded(uint256 provided, uint256 maximum);
113118
}
114119

115120
/// @title IGatewayEVM

contracts/zevm/GatewayZEVM.sol

+78-17
Original file line numberDiff line numberDiff line change
@@ -89,6 +89,45 @@ contract GatewayZEVM is
8989
_unpause();
9090
}
9191

92+
/// @notice Helper function to safely execute transferFrom
93+
/// @param zrc20 The ZRC20 token address
94+
/// @param from The sender address
95+
/// @param to The recipient address
96+
/// @param amount The amount to transfer
97+
/// @return True if the transfer was successful, false otherwise.
98+
function _safeTransferFrom(address zrc20, address from, address to, uint256 amount) private returns (bool) {
99+
try IZRC20(zrc20).transferFrom(from, to, amount) returns (bool success) {
100+
return success;
101+
} catch {
102+
return false;
103+
}
104+
}
105+
106+
// @notice Helper function to safely burn ZRC20 tokens
107+
// @param zrc20 The ZRC20 token address
108+
// @param amount The amount to burn
109+
// @return True if the burn was successful, false otherwise
110+
function _safeBurn(address zrc20, uint256 amount) private returns (bool) {
111+
try IZRC20(zrc20).burn(amount) returns (bool success) {
112+
return success;
113+
} catch {
114+
return false;
115+
}
116+
}
117+
118+
// @notice Helper function to safely deposit
119+
// @param zrc20 The ZRC20 token address
120+
// @param amount The target address to receive the deposited tokens
121+
// @param amount The amount to deposit
122+
// @return True if the deposit was successful, false otherwise
123+
function _safeDeposit(address zrc20, address target, uint256 amount) private returns (bool) {
124+
try IZRC20(zrc20).deposit(target, amount) returns (bool success) {
125+
return success;
126+
} catch {
127+
return false;
128+
}
129+
}
130+
92131
/// @dev Private function to withdraw ZRC20 tokens.
93132
/// @param amount The amount of tokens to withdraw.
94133
/// @param zrc20 The address of the ZRC20 token.
@@ -105,15 +144,18 @@ contract GatewayZEVM is
105144
/// @return The gas fee for the withdrawal.
106145
function _withdrawZRC20WithGasLimit(uint256 amount, address zrc20, uint256 gasLimit) private returns (uint256) {
107146
(address gasZRC20, uint256 gasFee) = IZRC20(zrc20).withdrawGasFeeWithGasLimit(gasLimit);
108-
if (!IZRC20(gasZRC20).transferFrom(msg.sender, PROTOCOL_ADDRESS, gasFee)) {
109-
revert GasFeeTransferFailed();
147+
148+
if (!_safeTransferFrom(gasZRC20, msg.sender, PROTOCOL_ADDRESS, gasFee)) {
149+
revert GasFeeTransferFailed(gasZRC20, PROTOCOL_ADDRESS, gasFee);
110150
}
111151

112-
if (!IZRC20(zrc20).transferFrom(msg.sender, address(this), amount)) {
113-
revert ZRC20TransferFailed();
152+
if (!_safeTransferFrom(zrc20, msg.sender, address(this), amount)) {
153+
revert ZRC20TransferFailed(zrc20, msg.sender, address(this), amount);
114154
}
115155

116-
if (!IZRC20(zrc20).burn(amount)) revert ZRC20BurnFailed();
156+
if (!_safeBurn(zrc20, amount)) {
157+
revert ZRC20BurnFailed(zrc20, amount);
158+
}
117159

118160
return gasFee;
119161
}
@@ -122,10 +164,15 @@ contract GatewayZEVM is
122164
/// @param amount The amount of tokens to transfer.
123165
/// @param to The address to transfer the tokens to.
124166
function _transferZETA(uint256 amount, address to) private {
125-
if (!IWETH9(zetaToken).transferFrom(msg.sender, address(this), amount)) revert FailedZetaSent();
126-
IWETH9(zetaToken).withdraw(amount);
127-
(bool sent,) = to.call{ value: amount }("");
128-
if (!sent) revert FailedZetaSent();
167+
if (!_safeTransferFrom(zetaToken, msg.sender, address(this), amount)) {
168+
revert FailedZetaSent(address(this), amount);
169+
}
170+
try IWETH9(zetaToken).withdraw(amount) {
171+
(bool sent,) = to.call{ value: amount }("");
172+
if (!sent) revert FailedZetaSent(to, amount);
173+
} catch {
174+
revert FailedZetaSent(to, amount);
175+
}
129176
}
130177

131178
/// @notice Withdraw ZRC20 tokens to an external chain.
@@ -144,7 +191,9 @@ contract GatewayZEVM is
144191
{
145192
if (receiver.length == 0) revert ZeroAddress();
146193
if (amount == 0) revert InsufficientZRC20Amount();
147-
if (revertOptions.revertMessage.length > MAX_MESSAGE_SIZE) revert MessageSizeExceeded();
194+
if (revertOptions.revertMessage.length > MAX_MESSAGE_SIZE) {
195+
revert MessageSizeExceeded(revertOptions.revertMessage.length, MAX_MESSAGE_SIZE);
196+
}
148197

149198
uint256 gasFee = _withdrawZRC20(amount, zrc20);
150199
emit Withdrawn(
@@ -182,7 +231,9 @@ contract GatewayZEVM is
182231
if (receiver.length == 0) revert ZeroAddress();
183232
if (amount == 0) revert InsufficientZRC20Amount();
184233
if (callOptions.gasLimit == 0) revert InsufficientGasLimit();
185-
if (message.length + revertOptions.revertMessage.length > MAX_MESSAGE_SIZE) revert MessageSizeExceeded();
234+
if (message.length + revertOptions.revertMessage.length > MAX_MESSAGE_SIZE) {
235+
revert MessageSizeExceeded(message.length + revertOptions.revertMessage.length, MAX_MESSAGE_SIZE);
236+
}
186237

187238
uint256 gasFee = _withdrawZRC20WithGasLimit(amount, zrc20, callOptions.gasLimit);
188239
emit WithdrawnAndCalled(
@@ -289,7 +340,9 @@ contract GatewayZEVM is
289340
whenNotPaused
290341
{
291342
if (callOptions.gasLimit == 0) revert InsufficientGasLimit();
292-
if (message.length + revertOptions.revertMessage.length > MAX_MESSAGE_SIZE) revert MessageSizeExceeded();
343+
if (message.length + revertOptions.revertMessage.length > MAX_MESSAGE_SIZE) {
344+
revert MessageSizeExceeded(message.length + revertOptions.revertMessage.length, MAX_MESSAGE_SIZE);
345+
}
293346

294347
_call(receiver, zrc20, message, callOptions, revertOptions);
295348
}
@@ -306,8 +359,8 @@ contract GatewayZEVM is
306359
if (receiver.length == 0) revert ZeroAddress();
307360

308361
(address gasZRC20, uint256 gasFee) = IZRC20(zrc20).withdrawGasFeeWithGasLimit(callOptions.gasLimit);
309-
if (!IZRC20(gasZRC20).transferFrom(msg.sender, PROTOCOL_ADDRESS, gasFee)) {
310-
revert GasFeeTransferFailed();
362+
if (!_safeTransferFrom(gasZRC20, msg.sender, PROTOCOL_ADDRESS, gasFee)) {
363+
revert GasFeeTransferFailed(gasZRC20, PROTOCOL_ADDRESS, gasFee);
311364
}
312365

313366
emit Called(msg.sender, zrc20, receiver, message, callOptions, revertOptions);
@@ -323,7 +376,9 @@ contract GatewayZEVM is
323376

324377
if (target == PROTOCOL_ADDRESS || target == address(this)) revert InvalidTarget();
325378

326-
if (!IZRC20(zrc20).deposit(target, amount)) revert ZRC20DepositFailed();
379+
if (!_safeDeposit(zrc20, target, amount)) {
380+
revert ZRC20DepositFailed(zrc20, target, amount);
381+
}
327382
}
328383

329384
/// @notice Execute a user-specified contract on ZEVM.
@@ -371,7 +426,10 @@ contract GatewayZEVM is
371426
if (amount == 0) revert InsufficientZRC20Amount();
372427
if (target == PROTOCOL_ADDRESS || target == address(this)) revert InvalidTarget();
373428

374-
if (!IZRC20(zrc20).deposit(target, amount)) revert ZRC20DepositFailed();
429+
if (!_safeDeposit(zrc20, target, amount)) {
430+
revert ZRC20DepositFailed(zrc20, target, amount);
431+
}
432+
375433
UniversalContract(target).onCall(context, zrc20, amount, message);
376434
}
377435

@@ -436,7 +494,10 @@ contract GatewayZEVM is
436494
if (amount == 0) revert InsufficientZRC20Amount();
437495
if (target == PROTOCOL_ADDRESS || target == address(this)) revert InvalidTarget();
438496

439-
if (!IZRC20(zrc20).deposit(target, amount)) revert ZRC20DepositFailed();
497+
if (!_safeDeposit(zrc20, target, amount)) {
498+
revert ZRC20DepositFailed(zrc20, target, amount);
499+
}
500+
440501
Revertable(target).onRevert(revertContext);
441502
}
442503

contracts/zevm/interfaces/IGatewayZEVM.sol

+26-7
Original file line numberDiff line numberDiff line change
@@ -76,7 +76,10 @@ interface IGatewayZEVMEvents {
7676
/// @notice Interface for the errors used in the GatewayZEVM contract.
7777
interface IGatewayZEVMErrors {
7878
/// @notice Error indicating a withdrawal failure.
79-
error WithdrawalFailed();
79+
/// @param token The address of the token that failed to withdraw.
80+
/// @param recipient The address that was supposed to receive the tokens.
81+
/// @param amount The amount of tokens that failed to withdraw.
82+
error WithdrawalFailed(address token, address recipient, uint256 amount);
8083

8184
/// @notice Error indicating an insufficient ZRC20 token amount.
8285
error InsufficientZRC20Amount();
@@ -85,16 +88,28 @@ interface IGatewayZEVMErrors {
8588
error InsufficientZetaAmount();
8689

8790
/// @notice Error indicating a failure to burn ZRC20 tokens.
88-
error ZRC20BurnFailed();
91+
/// @param zrc20 The address of the ZRC20 token that failed to burn.
92+
/// @param amount The amount of tokens that failed to burn.
93+
error ZRC20BurnFailed(address zrc20, uint256 amount);
8994

9095
/// @notice Error indicating a failure to transfer ZRC20 tokens.
91-
error ZRC20TransferFailed();
96+
/// @param zrc20 The address of the ZRC20 token that failed to transfer.
97+
/// @param from The address sending the tokens.
98+
/// @param to The address receiving the tokens.
99+
/// @param amount The amount of tokens that failed to transfer.
100+
error ZRC20TransferFailed(address zrc20, address from, address to, uint256 amount);
92101

93102
/// @notice Error indicating a failure to deposit ZRC20 tokens.
94-
error ZRC20DepositFailed();
103+
/// @param zrc20 The address of the ZRC20 token that failed to deposit.
104+
/// @param to The address that was supposed to receive the deposit.
105+
/// @param amount The amount of tokens that failed to deposit.
106+
error ZRC20DepositFailed(address zrc20, address to, uint256 amount);
95107

96108
/// @notice Error indicating a failure to transfer gas fee.
97-
error GasFeeTransferFailed();
109+
/// @param token The address of the token used for gas fee.
110+
/// @param to The address that was supposed to receive the gas fee.
111+
/// @param amount The amount of gas fee that failed to transfer.
112+
error GasFeeTransferFailed(address token, address to, uint256 amount);
98113

99114
/// @notice Error indicating that the caller is not the protocol account.
100115
error CallerIsNotProtocol();
@@ -103,7 +118,9 @@ interface IGatewayZEVMErrors {
103118
error InvalidTarget();
104119

105120
/// @notice Error indicating a failure to send ZETA tokens.
106-
error FailedZetaSent();
121+
/// @param recipient The address that was supposed to receive the ZETA tokens.
122+
/// @param amount The amount of ZETA tokens that failed to send.
123+
error FailedZetaSent(address recipient, uint256 amount);
107124

108125
/// @notice Error indicating that only WZETA or the protocol address can call the function.
109126
error OnlyWZETAOrProtocol();
@@ -112,7 +129,9 @@ interface IGatewayZEVMErrors {
112129
error InsufficientGasLimit();
113130

114131
/// @notice Error indicating message size exceeded in external functions.
115-
error MessageSizeExceeded();
132+
/// @param provided The size of the message that was provided.
133+
/// @param maximum The maximum allowed message size.
134+
error MessageSizeExceeded(uint256 provided, uint256 maximum);
116135
}
117136

118137
/// @title IGatewayZEVM

data/addresses.mainnet.json

-2
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,3 @@
1-
2-
31
[
42
{
53
"address": "0x70e967acFcC17c3941E87562161406d41676FD83",

data/addresses.testnet.json

-2
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,3 @@
1-
2-
31
[
42
{
53
"address": "0x0000ecb8cdd25a18f12daa23f6422e07fbf8b9e1",

docs/book.toml

-1
Original file line numberDiff line numberDiff line change
@@ -6,7 +6,6 @@ title = ""
66
no-section-label = true
77
additional-js = ["solidity.min.js"]
88
additional-css = ["book.css"]
9-
mathjax-support = true
109
git-repository-url = "https://github.com/zeta-chain/protocol-contracts"
1110

1211
[output.html.fold]

0 commit comments

Comments
 (0)