Skip to content

Commit

Permalink
refactor(precompiles): chain as source of truth
Browse files Browse the repository at this point in the history
The `ExocoreGateway` allows the contract owner to call precompiles which
allow the registration of a client chain or a token. It also enables
updated of these items. To check whether a request is an update or an
addition, it relies on the ExocoreGateway keeping a copy of registered
items. However, this does not work for items which were registered at
genesis.

This PR works around that problem by requiring that the single source of
truth be the precompiles, whose responsibility is to report if a request
was an update or an addition.
  • Loading branch information
MaxMustermann2 committed Jul 30, 2024
1 parent 29f2982 commit 9c469d4
Show file tree
Hide file tree
Showing 12 changed files with 143 additions and 287 deletions.
8 changes: 3 additions & 5 deletions docs/contracts-owner-manual.md
Original file line number Diff line number Diff line change
Expand Up @@ -17,12 +17,10 @@ While for `Vault` and `ExoCapsule`, they are deployed with upgradeable beacon pr

After all contracts are deployed, before the protocol starts to work, there are a few works left to be done by contract owner to enable restaking. One of the most important tasks is to register the client chain id and meta info to Exocore to mark this client chain as valid. This is done by the contract caller calling `ExocoreGateway.registerOrUpdateClientChain` to write `clientChainId`, `addressLength`, `name`, `signatureType` and other meta data to Exocore native module to finish registration. This operation would also call `ExocoreGateway.setPeer` to enable LayerZero messaging by setting remote `ClientChainGateway` as trusted peer to send/receive messages. After finishing registration, contract owner could call `ExocoreGateway.registerOrUpdateClientChain` again to update the meta data and set new peer, or contract owner could solely call `ExocoreGateway.setPeer` to change the address of remote peer contract.

## add tokens to whitelist
## add or update tokens to whitelist

Another important task before restaking being activated is to add tokens to whitelist to mark them as stake-able on both Exocore and client chain. This is done by contract owner calling `ExocoreGateway.addWhitelistTokens` to write token addresses, decimals, TVL limits and other metadata to Exocore, as well as sending a cross-chain message through layerzero to client chain to add these token addresses to the whitelist of `ClientChainGateway`.
Another important task before restaking being activated is to add tokens to whitelist to mark them as stake-able on both Exocore and client chain. This is done by contract owner calling `ExocoreGateway.addOrUpdateWhitelistTokens` to write token addresses, decimals, TVL limits and other metadata to Exocore, as well as sending a cross-chain message through layerzero to client chain to add these token addresses to the whitelist of `ClientChainGateway`.

Notice: contract owner must make sure the token data is correct like address, decimals and TVL limit, more importantly contract owner must ensure that for the same index, the data in different arrays like `tokens`, `decimals`, `tvlLimits` must point to the same token to be composed as complete token data.

## upgrade the meta data of whitelisted tokens

After adding tokens to whitelist, contract owner could call `ExocoreGateway.updateWhitelistedTokens` to update the meta data of already whitelisted tokens, and this function would not send a cross-chain message to client chain since the whitelist of `ClientChainGateway` only stores the token addresses.
After adding tokens to whitelist, contract owner could call `ExocoreGateway.addOrUpdateWhitelistTokens` to update the meta data of already whitelisted tokens, and this function would not send a cross-chain message to client chain since the whitelist of `ClientChainGateway` only stores the token addresses.
2 changes: 1 addition & 1 deletion script/3_Setup.s.sol
Original file line number Diff line number Diff line change
Expand Up @@ -118,7 +118,7 @@ contract SetupScript is BaseScript {
vm.selectFork(exocore);
uint256 messageLength = TOKEN_ADDRESS_BYTES_LENGTH * whitelistTokensBytes32.length + 2;
uint256 nativeFee = exocoreGateway.quote(clientChainId, new bytes(messageLength));
exocoreGateway.addWhitelistTokens{value: nativeFee}(
exocoreGateway.addOrUpdateWhitelistTokens{value: nativeFee}(
clientChainId, whitelistTokensBytes32, decimals, tvlLimits, names, metaData
);
vm.stopBroadcast();
Expand Down
147 changes: 51 additions & 96 deletions src/core/ExocoreGateway.sol
Original file line number Diff line number Diff line change
Expand Up @@ -142,20 +142,20 @@ contract ExocoreGateway is
) {
revert Errors.ZeroValue();
}
// signature type could be left as empty for current implementation
_registerClientChain(clientChainId, addressLength, name, metaInfo, signatureType);

bool updated = _registerOrUpdateClientChain(clientChainId, addressLength, name, metaInfo, signatureType);
// the peer is always set, regardless of `updated`
super.setPeer(clientChainId, peer);

if (!isRegisteredClientChain[clientChainId]) {
isRegisteredClientChain[clientChainId] = true;
emit ClientChainRegistered(clientChainId);
} else {
if (updated) {
emit ClientChainUpdated(clientChainId);
} else {
emit ClientChainRegistered(clientChainId);
}
}

/// @notice Sets a peer on the destination chain for this contract.
/// @dev This is the LayerZero peer.
/// @dev This is the LayerZero peer. This function is only here for the modifiers.
/// @param clientChainId The id of the client chain.
/// @param clientChainGateway The address of the peer as bytes32.
function setPeer(uint32 clientChainId, bytes32 clientChainGateway)
Expand All @@ -164,35 +164,55 @@ contract ExocoreGateway is
onlyOwner
whenNotPaused
{
if (!isRegisteredClientChain[clientChainId]) {
revert Errors.ExocoreGatewayNotRegisteredClientChainId();
}

super.setPeer(clientChainId, clientChainGateway);
}

/// @inheritdoc IExocoreGateway
function addWhitelistTokens(
function addOrUpdateWhitelistTokens(
uint32 clientChainId,
bytes32[] calldata tokens,
uint8[] calldata decimals,
uint256[] calldata tvlLimits,
string[] calldata names,
string[] calldata metaData
) external payable onlyOwner whenNotPaused nonReentrant {
_addOrUpdateWhitelistTokens(clientChainId, tokens, decimals, tvlLimits, names, metaData, true);
}
_validateWhitelistTokensInput(tokens, decimals, tvlLimits, names, metaData);

/// @inheritdoc IExocoreGateway
function updateWhitelistedTokens(
uint32 clientChainId,
bytes32[] calldata tokens,
uint8[] calldata decimals,
uint256[] calldata tvlLimits,
string[] calldata names,
string[] calldata metaData
) external onlyOwner whenNotPaused {
_addOrUpdateWhitelistTokens(clientChainId, tokens, decimals, tvlLimits, names, metaData, false);
bool success;
bool updated;
for (uint256 i; i < tokens.length; i++) {
require(tokens[i] != bytes32(0), "ExocoreGateway: token cannot be zero address");
require(tvlLimits[i] > 0, "ExocoreGateway: tvl limit should not be zero");
require(bytes(names[i]).length != 0, "ExocoreGateway: name cannot be empty");
require(bytes(metaData[i]).length != 0, "ExocoreGateway: meta data cannot be empty");

(success, updated) = ASSETS_CONTRACT.registerOrUpdateToken(
clientChainId, abi.encodePacked(tokens[i]), decimals[i], tvlLimits[i], names[i], metaData[i]
);

if (success) {
if (!updated) {
emit WhitelistTokenAdded(clientChainId, tokens[i]);
} else {
emit WhitelistTokenUpdated(clientChainId, tokens[i]);
}
} else {
if (!updated) {
revert AddWhitelistTokenFailed(tokens[i]);
} else {
revert UpdateWhitelistTokenFailed(tokens[i]);
}
}
}

if (!updated) {
_sendInterchainMsg(
clientChainId,
Action.REQUEST_ADD_WHITELIST_TOKENS,
abi.encodePacked(uint8(tokens.length), tokens),
false
);
}
}

/**
Expand Down Expand Up @@ -228,86 +248,19 @@ contract ExocoreGateway is
}
}

/// @dev The internal version of addWhitelistTokens and updateWhitelistedTokens.
/// @param clientChainId Source client chain id
/// @param tokens List of token addresses
/// @param decimals List of token decimals (like 18)
/// @param tvlLimits List of TVL limits (like max supply)
/// @param names List of token names
/// @param metaData List of arbitrary meta data for each token
/// @param add Whether to add or update the tokens
/// @dev Validates that lengths are equal, <= 255, and that the chain is registered.
// Though this function would call precompiled contract, all precompiled contracts belong to Exocore
// and we could make sure its implementation does not have dangerous behavior like reentrancy.
// slither-disable-next-line reentrancy-no-eth
function _addOrUpdateWhitelistTokens(
uint32 clientChainId,
bytes32[] calldata tokens,
uint8[] calldata decimals,
uint256[] calldata tvlLimits,
string[] calldata names,
string[] calldata metaData,
bool add
) internal {
_validateWhitelistTokensInput(clientChainId, tokens, decimals, tvlLimits, names, metaData);

for (uint256 i; i < tokens.length; i++) {
require(tokens[i] != bytes32(0), "ExocoreGateway: token cannot be zero address");
if (!add) {
require(isWhitelistedToken[tokens[i]], "ExocoreGateway: token has not been added to whitelist before");
}
require(tvlLimits[i] > 0, "ExocoreGateway: tvl limit should not be zero");
require(bytes(names[i]).length != 0, "ExocoreGateway: name cannot be empty");
require(bytes(metaData[i]).length != 0, "ExocoreGateway: meta data cannot be empty");

bool success = ASSETS_CONTRACT.registerToken(
clientChainId, abi.encodePacked(tokens[i]), decimals[i], tvlLimits[i], names[i], metaData[i]
);

if (success) {
if (add) {
isWhitelistedToken[tokens[i]] = true;
emit WhitelistTokenAdded(clientChainId, tokens[i]);
} else {
emit WhitelistTokenUpdated(clientChainId, tokens[i]);
}
} else {
if (add) {
revert AddWhitelistTokenFailed(tokens[i]);
} else {
revert UpdateWhitelistTokenFailed(tokens[i]);
}
}
}
if (add) {
_sendInterchainMsg(
clientChainId,
Action.REQUEST_ADD_WHITELIST_TOKENS,
abi.encodePacked(uint8(tokens.length), tokens),
false
);
}
}

/// @dev Validates the input for whitelist tokens.
/// @param clientChainId The client chain id, which must have been previously registered.
/// @param tokens The list of token addresses, length must be <= 255.
/// @param decimals The list of token decimals, length must be equal to that of @param tokens.
/// @param tvlLimits The list of token TVL limits, length must be equal to that of @param tokens.
/// @param names The list of token names, length must be equal to that of @param tokens.
/// @param metaData The list of token meta data, length must be equal to that of @param tokens.
function _validateWhitelistTokensInput(
uint32 clientChainId,
bytes32[] calldata tokens,
uint8[] calldata decimals,
uint256[] calldata tvlLimits,
string[] calldata names,
string[] calldata metaData
) internal view {
if (!isRegisteredClientChain[clientChainId]) {
revert ClientChainIDNotRegisteredBefore(clientChainId);
}

) internal pure {
uint256 expectedLength = tokens.length;
if (expectedLength > type(uint8).max) {
revert WhitelistTokensListTooLong();
Expand All @@ -321,23 +274,25 @@ contract ExocoreGateway is
}
}

/// @dev The internal version of registerClientChain.
/// @dev The internal version of registerOrUpdateClientChain.
/// @param clientChainId The client chain id.
/// @param addressLength The length of the address type on the client chain.
/// @param name The name of the client chain.
/// @param metaInfo The arbitrary metadata for the client chain.
/// @param signatureType The signature type supported by the client chain.
function _registerClientChain(
function _registerOrUpdateClientChain(
uint32 clientChainId,
uint8 addressLength,
string calldata name,
string calldata metaInfo,
string calldata signatureType
) internal {
bool success = ASSETS_CONTRACT.registerClientChain(clientChainId, addressLength, name, metaInfo, signatureType);
) internal returns (bool) {
(bool success, bool updated) =
ASSETS_CONTRACT.registerOrUpdateClientChain(clientChainId, addressLength, name, metaInfo, signatureType);
if (!success) {
revert RegisterClientChainToExocoreFailed(clientChainId);
}
return updated;
}

/// @inheritdoc OAppReceiverUpgradeable
Expand Down
19 changes: 1 addition & 18 deletions src/interfaces/IExocoreGateway.sol
Original file line number Diff line number Diff line change
Expand Up @@ -49,7 +49,7 @@ interface IExocoreGateway is IOAppReceiver, IOAppCore {
/// @param names The names of the tokens, in the same order as the tokens list.
/// @param metaData The meta information of the tokens, in the same order as the tokens list.
/// @dev The chain must be registered before adding tokens.
function addWhitelistTokens(
function addOrUpdateWhitelistTokens(
uint32 clientChainId,
bytes32[] calldata tokens,
uint8[] calldata decimals,
Expand All @@ -58,21 +58,4 @@ interface IExocoreGateway is IOAppReceiver, IOAppCore {
string[] calldata metaData
) external payable;

/// @notice Updates a list of whitelisted tokens to the client chain.
/// @param clientChainId The LayerZero chain id of the client chain.
/// @param tokens The list of token addresses to be whitelisted.
/// @param decimals The list of token decimals, in the same order as the tokens list.
/// @param tvlLimits The list of token TVL limits (typically max supply),in the same order as the tokens list.
/// @param names The names of the tokens, in the same order as the tokens list.
/// @param metaData The meta information of the tokens, in the same order as the tokens list.
/// @dev The chain must be registered before updating tokens, and the token as well.
function updateWhitelistedTokens(
uint32 clientChainId,
bytes32[] calldata tokens,
uint8[] calldata decimals,
uint256[] calldata tvlLimits,
string[] calldata names,
string[] calldata metaData
) external;

}
52 changes: 30 additions & 22 deletions src/interfaces/precompiles/IAssets.sol
Original file line number Diff line number Diff line change
Expand Up @@ -17,59 +17,67 @@ interface IAssets {
/// @dev deposit the client chain assets for the staker,
/// that will change the state in deposit module
/// Note that this address cannot be a module account.
/// @param clientChainLzID The LzID of client chain
/// @param clientChainID is the layerZero chainID if it is supported.
// It might be allocated by Exocore when the client chain isn't supported
// by layerZero
/// @param assetsAddress The client chain asset address
/// @param stakerAddress The staker address
/// @param opAmount The amount to deposit
function depositTo(uint32 clientChainLzID, bytes memory assetsAddress, bytes memory stakerAddress, uint256 opAmount)
function depositTo(uint32 clientChainID, bytes memory assetsAddress, bytes memory stakerAddress, uint256 opAmount)
external
returns (bool success, uint256 latestAssetState);

/// TRANSACTIONS
/// @dev withdraw To the staker, that will change the state in withdraw module
/// Note that this address cannot be a module account.
/// @param clientChainLzID The LzID of client chain
/// @param clientChainID is the layerZero chainID if it is supported.
// It might be allocated by Exocore when the client chain isn't supported
// by layerZero
/// @param assetsAddress The client chain asset Address
/// @param withdrawAddress The withdraw address
/// @param opAmount The withdraw amount
function withdrawPrincipal(
uint32 clientChainLzID,
uint32 clientChainID,
bytes memory assetsAddress,
bytes memory withdrawAddress,
uint256 opAmount
) external returns (bool success, uint256 latestAssetState);

/// QUERIES
/// @dev Returns the chain indices of the client chains.
function getClientChains() external view returns (bool, uint32[] memory);

/// TRANSACTIONS
/// @dev register some client chain to allow token registration from that chain, staking
/// from that chain, and other operations from that chain.
/// @dev registers or updates a client chain to allow deposits / staking, etc.
/// from that chain.
/// @param clientChainID is the layerZero chainID if it is supported.
// It might be allocated by Exocore when the client chain isn't supported
// by layerZero
function registerClientChain(
function registerOrUpdateClientChain(
uint32 clientChainID,
uint8 addressLength,
string calldata name,
string calldata metaInfo,
string calldata signatureType
) external returns (bool success);
) external returns (bool success, bool updated);

/// TRANSACTIONS
/// @dev register unwhitelisted token address to exocore assets module
/// @param clientChainID is the layerZero chainID if it is supported.
// It might be allocated by Exocore when the client chain isn't supported
// by layerZero
/// @param token The token address that would be registered to exocore
function registerToken(
/// @dev register or update token addresses to exocore
/// @dev note that there is no way to delete a token. If a token is to be removed,
/// the TVL limit should be set to 0.
/// @param clientChainID is the identifier of the token's home chain (LZ or otherwise)
/// @param token is the address of the token on the home chain
/// @param decimals is the number of decimals of the token
/// @param tvlLimit is the number of tokens that can be deposited in the system. Set to
/// maxSupply if there is no limit
/// @param name is the name of the token
/// @param metaData is the arbitrary metadata of the token
/// @return success if the token registration is successful
/// @return updated whether the token was added or updated
function registerOrUpdateToken(
uint32 clientChainID,
bytes calldata token,
uint8 decimals,
uint256 tvlLimit,
string calldata name,
string calldata metaData
) external returns (bool success);
) external returns (bool success, bool updated);

/// QUERIES
/// @dev Returns the chain indices of the client chains.
function getClientChains() external view returns (bool, uint32[] memory);

}
3 changes: 0 additions & 3 deletions src/libraries/Errors.sol
Original file line number Diff line number Diff line change
Expand Up @@ -163,9 +163,6 @@ library Errors {
/// @dev ExocoreGateway: failed to decode client chain ids
error ExocoreGatewayFailedToDecodeClientChainIds();

/// @dev ExocoreGateway: client chain should be registered before setting peer to change peer address
error ExocoreGatewayNotRegisteredClientChainId();

/// @dev ExocoreGateway: thrown when associateOperatorWithEVMStaker failed
error AssociateOperatorFailed(uint32 clientChainId, address staker, string operator);

Expand Down
Loading

0 comments on commit 9c469d4

Please sign in to comment.