Skip to content

Commit

Permalink
feat: Introduce opt-in deterministic relay data hashes (again) (#639)
Browse files Browse the repository at this point in the history
* Revert "feat(SpokePool): Introduce opt-in deterministic relay data hashes (#583)"

This reverts commit 9d21d1b.

* Reapply "feat(SpokePool): Introduce opt-in deterministic relay data hashes (#583)"

This reverts commit d363bf0.

* add deposit nonces to 7683

Signed-off-by: Matt Rice <matthewcrice32@gmail.com>

* fix

Signed-off-by: Matt Rice <matthewcrice32@gmail.com>

* WIP

Signed-off-by: Matt Rice <matthewcrice32@gmail.com>

* feat(SpokePool): Introduce opt-in deterministic relay data hashes (#583)

* fix(SpokePool): Apply exclusivity consistently

The new relative exclusivity check has not been propagated to
fillV3RelayWithUpdatedDeposit(). Identified via test case failures in
the relayer.

Signed-off-by: Paul <108695806+pxrl@users.noreply.github.com>

* Also check on slow fill requests

* Update contracts/SpokePool.sol

* lint

* Update

* Add pure

* Fix

* Add tests

* improve(SpokePool): _depositV3 interprets `exclusivityParameter` as 0, an offset, or a timestamp

There should be a way for the deposit transaction to remove chain re-org risk affecting the block.timestamp by allowing the caller to set a fixed `exclusivityDeadline` value. This supports the existing behavior where the `exclusivityDeadline` is always emitted as its passed in.

The new behavior is that if the `exclusivityParameter`, which replaces the `exclusivityDeadlineOffset` parameter, is 0 or greater than 1 year in seconds, then the `exclusivityDeadline` is equal to this parameter. Otherwise, its interpreted by `_depositV3()` as an offset. The offset would be useful in cases where the origin chain will not re-org, for example.

* Update SpokePool.sol

* Update SpokePool.Relay.ts

* Update SpokePool.SlowRelay.ts

* Update contracts/SpokePool.sol

Co-authored-by: Paul <108695806+pxrl@users.noreply.github.com>

* Update SpokePool.sol

* Update contracts/SpokePool.sol

* rebase

* Update SpokePool.sol

* Revert "Merge branch 'npai/exclusivity-switch' into mrice32/deterministic-new"

This reverts commit 2432944, reversing
changes made to 6fe3534.

* Revert "Merge branch 'npai/exclusivity-switch' into mrice32/deterministic-new"

This reverts commit 2432944, reversing
changes made to 6fe3534.

* revert

* Update SpokePool.sol

* Fix

* Update SpokePool.sol

Co-authored-by: Chris Maree <christopher.maree@gmail.com>

* WIP

* WIP

* wip

* Update SpokePool.Relay.ts

* Fix

* Update SpokePool.sol

* Update SpokePool.sol

---------

Signed-off-by: Matt Rice <matthewcrice32@gmail.com>
Signed-off-by: Paul <108695806+pxrl@users.noreply.github.com>
Co-authored-by: nicholaspai <npai.nyc@gmail.com>
Co-authored-by: nicholaspai <9457025+nicholaspai@users.noreply.github.com>
Co-authored-by: Paul <108695806+pxrl@users.noreply.github.com>
Co-authored-by: Chris Maree <christopher.maree@gmail.com>
  • Loading branch information
5 people authored Nov 25, 2024
1 parent d1f7a3a commit 9905481
Show file tree
Hide file tree
Showing 9 changed files with 297 additions and 48 deletions.
173 changes: 161 additions & 12 deletions contracts/SpokePool.sol
Original file line number Diff line number Diff line change
Expand Up @@ -60,7 +60,12 @@ abstract contract SpokePool is
WETH9Interface private DEPRECATED_wrappedNativeToken;
uint32 private DEPRECATED_depositQuoteTimeBuffer;

// Count of deposits is used to construct a unique deposit identifier for this spoke pool.
// Count of deposits is used to construct a unique deposit identifier for this spoke pool. This value
// gets emitted and incremented on each depositV3 call. Because its a uint32, it will get implicitly cast to
// uint256 in the emitted V3FundsDeposited event by setting its most significant bits to 0.
// This variable name `numberOfDeposits` should ideally be re-named to
// depositNonceCounter or something similar because its not a true representation of the number of deposits
// because `unsafeDepositV3` can be called directly and bypass this increment.
uint32 public numberOfDeposits;

// Whether deposits and fills are disabled.
Expand Down Expand Up @@ -135,12 +140,12 @@ abstract contract SpokePool is

bytes32 public constant UPDATE_V3_DEPOSIT_DETAILS_HASH =
keccak256(
"UpdateDepositDetails(uint32 depositId,uint256 originChainId,uint256 updatedOutputAmount,bytes32 updatedRecipient,bytes updatedMessage)"
"UpdateDepositDetails(uint256 depositId,uint256 originChainId,uint256 updatedOutputAmount,bytes32 updatedRecipient,bytes updatedMessage)"
);

bytes32 public constant UPDATE_V3_DEPOSIT_ADDRESS_OVERLOAD_DETAILS_HASH =
keccak256(
"UpdateDepositDetails(uint32 depositId,uint256 originChainId,uint256 updatedOutputAmount,address updatedRecipient,bytes updatedMessage)"
"UpdateDepositDetails(uint256 depositId,uint256 originChainId,uint256 updatedOutputAmount,address updatedRecipient,bytes updatedMessage)"
);

// Default chain Id used to signify that no repayment is requested, for example when executing a slow fill.
Expand Down Expand Up @@ -487,7 +492,7 @@ abstract contract SpokePool is
* the fill will revert on the destination chain. Must be set between [currentTime, currentTime + fillDeadlineBuffer]
* where currentTime is block.timestamp on this chain or this transaction will revert.
* @param exclusivityParameter This value is used to set the exclusivity deadline timestamp in the emitted deposit
* event. Before this destinationchain timestamp, only the exclusiveRelayer (if set to a non-zero address),
* event. Before this destination chain timestamp, only the exclusiveRelayer (if set to a non-zero address),
* can fill this deposit. There are three ways to use this parameter:
* 1. NO EXCLUSIVITY: If this value is set to 0, then a timestamp of 0 will be emitted,
* meaning that there is no exclusivity period.
Expand All @@ -514,6 +519,13 @@ abstract contract SpokePool is
uint32 exclusivityParameter,
bytes calldata message
) public payable override nonReentrant unpausedDeposits {
// Increment deposit nonce variable `numberOfDeposits` so that deposit ID for this deposit on this
// spoke pool is unique. This variable `numberOfDeposits` should ideally be re-named to
// depositNonceCounter or something similar because its not a true representation of the number of deposits
// because `unsafeDepositV3` can be called directly and bypass this increment.
// The `numberOfDeposits` is a uint32 that will get implicitly cast to uint256 by setting the
// most significant bits to 0, which creates very little chance this an unsafe deposit ID collides
// with a safe deposit ID.
DepositV3Params memory params = DepositV3Params({
depositor: depositor,
recipient: recipient,
Expand All @@ -523,7 +535,7 @@ abstract contract SpokePool is
outputAmount: outputAmount,
destinationChainId: destinationChainId,
exclusiveRelayer: exclusiveRelayer,
depositId: numberOfDeposits++, // Increment count of deposits so that deposit ID for this spoke pool is unique.
depositId: numberOfDeposits++,
quoteTimestamp: quoteTimestamp,
fillDeadline: fillDeadline,
exclusivityParameter: exclusivityParameter,
Expand Down Expand Up @@ -563,8 +575,17 @@ abstract contract SpokePool is
* @param fillDeadline The deadline for the relayer to fill the deposit. After this destination chain timestamp, the fill will
* revert on the destination chain. Must be set between [currentTime, currentTime + fillDeadlineBuffer] where currentTime
* is block.timestamp on this chain.
* @param exclusivityPeriod Added to the current time to set the exclusive relayer deadline. After this timestamp,
* anyone can fill the deposit.
* @param exclusivityParameter This value is used to set the exclusivity deadline timestamp in the emitted deposit
* event. Before this destination chain timestamp, only the exclusiveRelayer (if set to a non-zero address),
* can fill this deposit. There are three ways to use this parameter:
* 1. NO EXCLUSIVITY: If this value is set to 0, then a timestamp of 0 will be emitted,
* meaning that there is no exclusivity period.
* 2. OFFSET: If this value is less than MAX_EXCLUSIVITY_PERIOD_SECONDS, then add this value to
* the block.timestamp to derive the exclusive relayer deadline. Note that using the parameter in this way
* will expose the filler of the deposit to the risk that the block.timestamp of this event gets changed
* due to a chain-reorg, which would also change the exclusivity timestamp.
* 3. TIMESTAMP: Otherwise, set this value as the exclusivity deadline timestamp.
* which is the deadline for the exclusiveRelayer to fill the deposit.
* @param message The message to send to the recipient on the destination chain if the recipient is a contract. If the
* message is not empty, the recipient contract must implement `handleV3AcrossMessage()` or the fill will revert.
*/
Expand All @@ -579,7 +600,7 @@ abstract contract SpokePool is
address exclusiveRelayer,
uint32 quoteTimestamp,
uint32 fillDeadline,
uint32 exclusivityPeriod,
uint32 exclusivityParameter,
bytes calldata message
) public payable override {
depositV3(
Expand All @@ -593,11 +614,121 @@ abstract contract SpokePool is
exclusiveRelayer.toBytes32(),
quoteTimestamp,
fillDeadline,
exclusivityPeriod,
exclusivityParameter,
message
);
}

/**
* @notice An overloaded version of `unsafeDepositV3` that accepts `address` types for backward compatibility. *
* @dev This version mirrors the original `unsafeDepositV3` function, but uses `address` types for `depositor`, `recipient`,
* `inputToken`, `outputToken`, and `exclusiveRelayer` for compatibility with contracts using the `address` type.
*
* The key functionality and logic remain identical, ensuring interoperability across both versions.
*/
function unsafeDepositV3(
address depositor,
address recipient,
address inputToken,
address outputToken,
uint256 inputAmount,
uint256 outputAmount,
uint256 destinationChainId,
address exclusiveRelayer,
uint256 depositNonce,
uint32 quoteTimestamp,
uint32 fillDeadline,
uint32 exclusivityParameter,
bytes calldata message
) public payable {
unsafeDepositV3(
depositor.toBytes32(),
recipient.toBytes32(),
inputToken.toBytes32(),
outputToken.toBytes32(),
inputAmount,
outputAmount,
destinationChainId,
exclusiveRelayer.toBytes32(),
depositNonce,
quoteTimestamp,
fillDeadline,
exclusivityParameter,
message
);
}

/**
* @notice See depositV3 for details. This function is identical to depositV3 except that it does not use the
* global deposit ID counter as a deposit nonce, instead allowing the caller to pass in a deposit nonce. This
* function is designed to be used by anyone who wants to pre-compute their resultant relay data hash, which
* could be useful for filling a deposit faster and avoiding any risk of a relay hash unexpectedly changing
* due to another deposit front-running this one and incrementing the global deposit ID counter.
* @dev This is labeled "unsafe" because there is no guarantee that the depositId emitted in the resultant
* V3FundsDeposited event is unique which means that the
* corresponding fill might collide with an existing relay hash on the destination chain SpokePool,
* which would make this deposit unfillable. In this case, the depositor would subsequently receive a refund
* of `inputAmount` of `inputToken` on the origin chain after the fill deadline.
* @dev On the destination chain, the hash of the deposit data will be used to uniquely identify this deposit, so
* modifying any params in it will result in a different hash and a different deposit. The hash will comprise
* all parameters to this function along with this chain's chainId(). Relayers are only refunded for filling
* deposits with deposit hashes that map exactly to the one emitted by this contract.
* @param depositNonce The nonce that uniquely identifies this deposit. This function will combine this parameter
* with the msg.sender address to create a unique uint256 depositNonce and ensure that the msg.sender cannot
* use this function to front-run another depositor's unsafe deposit. This function guarantees that the resultant
* deposit nonce will not collide with a safe uint256 deposit nonce whose 24 most significant bytes are always 0.
* @param depositor See identically named parameter in depositV3() comments.
* @param recipient See identically named parameter in depositV3() comments.
* @param inputToken See identically named parameter in depositV3() comments.
* @param outputToken See identically named parameter in depositV3() comments.
* @param inputAmount See identically named parameter in depositV3() comments.
* @param outputAmount See identically named parameter in depositV3() comments.
* @param destinationChainId See identically named parameter in depositV3() comments.
* @param exclusiveRelayer See identically named parameter in depositV3() comments.
* @param quoteTimestamp See identically named parameter in depositV3() comments.
* @param fillDeadline See identically named parameter in depositV3() comments.
* @param exclusivityParameter See identically named parameter in depositV3() comments.
* @param message See identically named parameter in depositV3() comments.
*/
function unsafeDepositV3(
bytes32 depositor,
bytes32 recipient,
bytes32 inputToken,
bytes32 outputToken,
uint256 inputAmount,
uint256 outputAmount,
uint256 destinationChainId,
bytes32 exclusiveRelayer,
uint256 depositNonce,
uint32 quoteTimestamp,
uint32 fillDeadline,
uint32 exclusivityParameter,
bytes calldata message
) public payable nonReentrant unpausedDeposits {
// @dev Create the uint256 deposit ID by concatenating the msg.sender and depositor address with the inputted
// depositNonce parameter. The resultant 32 byte string will be hashed and then casted to an "unsafe"
// uint256 deposit ID. The probability that the resultant ID collides with a "safe" deposit ID is
// equal to the chance that the first 28 bytes of the hash are 0, which is too small for us to consider.

uint256 depositId = getUnsafeDepositId(msg.sender, depositor, depositNonce);
DepositV3Params memory params = DepositV3Params({
depositor: depositor,
recipient: recipient,
inputToken: inputToken,
outputToken: outputToken,
inputAmount: inputAmount,
outputAmount: outputAmount,
destinationChainId: destinationChainId,
exclusiveRelayer: exclusiveRelayer,
depositId: depositId,
quoteTimestamp: quoteTimestamp,
fillDeadline: fillDeadline,
exclusivityParameter: exclusivityParameter,
message: message
});
_depositV3(params);
}

/**
* @notice Submits deposit and sets quoteTimestamp to current Time. Sets fill and exclusivity
* deadlines as offsets added to the current time. This function is designed to be called by users
Expand Down Expand Up @@ -740,7 +871,7 @@ abstract contract SpokePool is
*/
function speedUpV3Deposit(
bytes32 depositor,
uint32 depositId,
uint256 depositId,
uint256 updatedOutputAmount,
bytes32 updatedRecipient,
bytes calldata updatedMessage,
Expand Down Expand Up @@ -794,7 +925,7 @@ abstract contract SpokePool is
*/
function speedUpV3Deposit(
address depositor,
uint32 depositId,
uint256 depositId,
uint256 updatedOutputAmount,
address updatedRecipient,
bytes calldata updatedMessage,
Expand Down Expand Up @@ -1167,6 +1298,24 @@ abstract contract SpokePool is
return block.timestamp; // solhint-disable-line not-rely-on-time
}

/**
* @notice Returns the deposit ID for an unsafe deposit. This function is used to compute the deposit ID
* in unsafeDepositV3 and is provided as a convenience.
* @dev msgSenderand depositor are both used as inputs to allow passthrough depositors to create unique
* deposit hash spaces for unique depositors.
* @param msgSender The caller of the transaction used as input to produce the deposit ID.
* @param depositor The depositor address used as input to produce the deposit ID.
* @param depositNonce The nonce used as input to produce the deposit ID.
* @return The deposit ID for the unsafe deposit.
*/
function getUnsafeDepositId(
address msgSender,
bytes32 depositor,
uint256 depositNonce
) public pure returns (uint256) {
return uint256(keccak256(abi.encodePacked(msgSender, depositor, depositNonce)));
}

function getRelayerRefund(address l2TokenAddress, address refundAddress) public view returns (uint256) {
return relayerRefund[l2TokenAddress][refundAddress];
}
Expand Down Expand Up @@ -1429,7 +1578,7 @@ abstract contract SpokePool is

function _verifyUpdateV3DepositMessage(
address depositor,
uint32 depositId,
uint256 depositId,
uint256 originChainId,
uint256 updatedOutputAmount,
bytes32 updatedRecipient,
Expand Down
20 changes: 16 additions & 4 deletions contracts/erc7683/ERC7683OrderDepositor.sol
Original file line number Diff line number Diff line change
Expand Up @@ -73,6 +73,7 @@ abstract contract ERC7683OrderDepositor is IOriginSettler {
acrossOrderData.outputAmount,
acrossOrderData.destinationChainId,
acrossOriginFillerData.exclusiveRelayer,
acrossOrderData.depositNonce,
// Note: simplifying assumption to avoid quote timestamps that cause orders to expire before the deadline.
SafeCast.toUint32(order.openDeadline - QUOTE_BEFORE_DEADLINE),
order.fillDeadline,
Expand Down Expand Up @@ -103,6 +104,7 @@ abstract contract ERC7683OrderDepositor is IOriginSettler {
acrossOrderData.outputAmount,
acrossOrderData.destinationChainId,
acrossOrderData.exclusiveRelayer,
acrossOrderData.depositNonce,
// Note: simplifying assumption to avoid the order type having to bake in the quote timestamp.
SafeCast.toUint32(block.timestamp),
order.fillDeadline,
Expand Down Expand Up @@ -161,6 +163,17 @@ abstract contract ERC7683OrderDepositor is IOriginSettler {
return SafeCast.toUint32(block.timestamp); // solhint-disable-line not-rely-on-time
}

/**
* @notice Convenience method to compute the Across depositId for orders sent through 7683.
* @dev if a 0 depositNonce is used, the depositId will not be deterministic (meaning it can change depending on
* when the open txn is mined), but you will be safe from collisions. See the unsafeDepositV3 method on SpokePool
* for more details on how to choose between deterministic and non-deterministic.
* @param depositNonce the depositNonce field in the order.
* @param depositor the sender or signer of the order.
* @return the resulting Across depositId.
*/
function computeDepositId(uint256 depositNonce, address depositor) public view virtual returns (uint256);

function _resolveFor(GaslessCrossChainOrder calldata order, bytes calldata fillerData)
internal
view
Expand Down Expand Up @@ -223,7 +236,7 @@ abstract contract ERC7683OrderDepositor is IOriginSettler {
relayData.inputAmount = acrossOrderData.inputAmount;
relayData.outputAmount = acrossOrderData.outputAmount;
relayData.originChainId = block.chainid;
relayData.depositId = _currentDepositId();
relayData.depositId = computeDepositId(acrossOrderData.depositNonce, order.user);
relayData.fillDeadline = order.fillDeadline;
relayData.exclusivityDeadline = acrossOrderData.exclusivityPeriod;
relayData.message = acrossOrderData.message;
Expand Down Expand Up @@ -287,7 +300,7 @@ abstract contract ERC7683OrderDepositor is IOriginSettler {
relayData.inputAmount = acrossOrderData.inputAmount;
relayData.outputAmount = acrossOrderData.outputAmount;
relayData.originChainId = block.chainid;
relayData.depositId = _currentDepositId();
relayData.depositId = computeDepositId(acrossOrderData.depositNonce, msg.sender);
relayData.fillDeadline = order.fillDeadline;
relayData.exclusivityDeadline = acrossOrderData.exclusivityPeriod;
relayData.message = acrossOrderData.message;
Expand Down Expand Up @@ -357,13 +370,12 @@ abstract contract ERC7683OrderDepositor is IOriginSettler {
uint256 outputAmount,
uint256 destinationChainId,
address exclusiveRelayer,
uint256 depositNonce,
uint32 quoteTimestamp,
uint32 fillDeadline,
uint32 exclusivityPeriod,
bytes memory message
) internal virtual;

function _currentDepositId() internal view virtual returns (uint32);

function _destinationSettler(uint256 chainId) internal view virtual returns (address);
}
Loading

0 comments on commit 9905481

Please sign in to comment.