// SPDX-License-Identifier: MIT pragma solidity ^0.8.26; /// @title Serai Router (without functions overriden by selector collisions) /// @author Luke Parker /// @notice Intakes coins for the Serai network and handles relaying batches of transfers out interface IRouterWithoutCollisions { /// @notice Emitted when the key for Serai's Ethereum validators is updated /// @param nonce The nonce consumed to update this key /// @param key The key updated to event SeraiKeyUpdated(uint256 indexed nonce, bytes32 indexed key); /// @notice Emitted when an InInstruction occurs /// @param from The address which called `inInstruction` and caused this event to be emitted /// @param coin The coin transferred in /// @param amount The amount of the coin transferred in /// @param instruction The Shorthand-encoded InInstruction for Serai to decode and handle event InInstruction( address indexed from, address indexed coin, uint256 amount, bytes instruction ); /// @notice Emitted when a batch of `OutInstruction`s occurs /// @param nonce The nonce consumed to execute this batch of transactions /// @param messageHash The hash of the message signed for the executed batch event Executed(uint256 indexed nonce, bytes32 indexed messageHash); /// @notice Emitted when `escapeHatch` is invoked /// @param escapeTo The address to escape to event EscapeHatch(address indexed escapeTo); /// @notice Emitted when coins escape through the escape hatch /// @param coin The coin which escaped event Escaped(address indexed coin); /// @notice The contract has had its escape hatch invoked and won't accept further actions error EscapeHatchInvoked(); /// @notice The signature was invalid error InvalidSignature(); /// @notice The amount specified didn't match `msg.value` error AmountMismatchesMsgValue(); /// @notice The call to an ERC20's `transferFrom` failed error TransferFromFailed(); /// @notice `execute` was re-entered error ReenteredExecute(); /// @notice An invalid address to escape to was specified. error InvalidEscapeAddress(); /// @notice Escaping when escape hatch wasn't invoked. error EscapeHatchNotInvoked(); /// @notice Transfer coins into Serai with an instruction /// @param coin The coin to transfer in (address(0) if Ether) /// @param amount The amount to transfer in (msg.value if Ether) /** * @param instruction The Shorthand-encoded InInstruction for Serai to associate with this * transfer in */ // Re-entrancy doesn't bork this function // slither-disable-next-line reentrancy-events function inInstruction(address coin, uint256 amount, bytes memory instruction) external payable; /// @notice Execute some arbitrary code within a secure sandbox /** * @dev This performs sandboxing by deploying this code with `CREATE`. This is an external * function as we can't meter `CREATE`/internal functions. We work around this by calling this * function with `CALL` (which we can meter). This does forward `msg.value` to the newly * deployed contract. */ /// @param code The code to execute function executeArbitraryCode(bytes memory code) external payable; /// @notice Escape coins after the escape hatch has been invoked /// @param coin The coin to escape function escape(address coin) external; /// @notice Fetch the next nonce to use by an action published to this contract /// return The next nonce to use by an action published to this contract function nextNonce() external view returns (uint256); /// @notice Fetch the current key for Serai's Ethereum validator set /// @return The current key for Serai's Ethereum validator set function seraiKey() external view returns (bytes32); /// @notice Fetch the address escaped to /// @return The address which was escaped to (address(0) if the escape hatch hasn't been invoked) function escapedTo() external view returns (address); } /// @title Serai Router /// @author Luke Parker /// @notice Intakes coins for the Serai network and handles relaying batches of transfers out interface IRouter is IRouterWithoutCollisions { /// @title A signature /// @dev Thin wrapper around `c, s` to simplify the API struct Signature { bytes32 c; bytes32 s; } /// @title The type of destination /// @dev A destination is either an address or a blob of code to deploy and call enum DestinationType { Address, Code } /// @title A code destination /** * @dev If transferring an ERC20 to this destination, it will be transferred to the address the * code will be deployed to. If transferring ETH, it will be transferred with the deployment of * the code. `code` is deployed with CREATE (calling its constructor). The entire deployment * (and associated sandboxing) must consume less than `gasLimit` units of gas or it will revert. */ struct CodeDestination { uint32 gasLimit; bytes code; } /// @title An instruction to transfer coins out /// @dev Specifies a destination and amount but not the coin as that's assumed to be contextual struct OutInstruction { DestinationType destinationType; bytes destination; uint256 amount; } /// @notice Update the key representing Serai's Ethereum validators /// @dev This assumes the key is correct. No checks on it are performed /// @param signature The signature by the current key authorizing this update /// @param newSeraiKey The key to update to function updateSeraiKey(Signature calldata signature, bytes32 newSeraiKey) external; /// @notice Execute a batch of `OutInstruction`s /** * @dev All `OutInstruction`s in a batch are only for a single coin to simplify handling of the * fee */ /// @param signature The signature by the current key for Serai's Ethereum validators /// @param coin The coin all of these `OutInstruction`s are for /// @param fee The fee to pay (in coin) to the caller for their relaying of this batch /// @param outs The `OutInstruction`s to act on function execute( Signature calldata signature, address coin, uint256 fee, OutInstruction[] calldata outs ) external; /// @notice Escapes to a new smart contract /// @dev This should be used upon an invariant being reached or new functionality being needed /// @param signature The signature by the current key for Serai's Ethereum validators /// @param escapeTo The address to escape to function escapeHatch(Signature calldata signature, address escapeTo) external; }