diff --git a/contracts/interchain-token-service/src/contract/execute.rs b/contracts/interchain-token-service/src/contract/execute.rs index f59dbd80e..a8036142b 100644 --- a/contracts/interchain-token-service/src/contract/execute.rs +++ b/contracts/interchain-token-service/src/contract/execute.rs @@ -71,6 +71,11 @@ pub enum Error { token_id: TokenId, chain: ChainNameRaw, }, + #[error("token decimals not set for token {token_id} on chain {chain}")] + TokenDecimalsNotSet { + token_id: TokenId, + chain: ChainNameRaw, + }, } /// Executes an incoming ITS message. @@ -150,7 +155,7 @@ fn apply_to_hub( match message { Message::InterchainTransfer(transfer) => { apply_transfer(storage, source_chain, destination_chain, &transfer) - .map(|_| Message::InterchainTransfer(transfer))? + .map(Message::InterchainTransfer)? } Message::DeployInterchainToken(deploy_token) => { apply_token_deployment(storage, &source_chain, &destination_chain, deploy_token) @@ -263,7 +268,6 @@ pub fn set_chain_config( } } -#[allow(unused)] /// Calculates the destination on token transfer amount. /// /// The amount is calculated based on the token decimals on the source and destination chains. @@ -279,15 +283,12 @@ fn destination_amount( token_id: TokenId, source_amount: nonempty::Uint256, ) -> Result { - let source_chain_decimals = - state::load_token_decimals(storage, source_chain, token_id).change_context(Error::State)?; + let source_chain_decimals = try_load_token_decimals(storage, source_chain.clone(), token_id)?; let destination_chain_decimals = - state::load_token_decimals(storage, destination_chain, token_id) - .change_context(Error::State)?; + try_load_token_decimals(storage, destination_chain.clone(), token_id)?; let destination_chain_max_uint = state::load_chain_config(storage, destination_chain) .change_context(Error::State)? .max_uint; - // dest_chain_amount = amount * 10 ^ (dest_chain_decimals - src_chain_decimals) // It's intentionally written in this way since the end result may still be fine even if // 1) amount * (10 ^ (dest_chain_decimals)) overflows // 2) amount / (10 ^ (src_chain_decimals)) is zero @@ -384,54 +385,69 @@ fn apply_transfer( source_chain: ChainNameRaw, destination_chain: ChainNameRaw, transfer: &InterchainTransfer, -) -> Result<(), Error> { - subtract_amount_from_source(storage, source_chain, transfer)?; - add_amount_to_destination(storage, destination_chain, transfer) +) -> Result { + let destination_amount = destination_amount( + storage, + &source_chain, + &destination_chain, + transfer.token_id, + transfer.amount, + )?; + + subtract_amount_from_source(storage, source_chain, transfer.token_id, transfer.amount)?; + add_amount_to_destination( + storage, + destination_chain, + transfer.token_id, + destination_amount, + )?; + + Ok(InterchainTransfer { + amount: destination_amount, + + ..transfer.clone() + }) } fn subtract_amount_from_source( storage: &mut dyn Storage, source_chain: ChainNameRaw, - transfer: &InterchainTransfer, + token_id: TokenId, + amount: nonempty::Uint256, ) -> Result<(), Error> { - let mut source_instance = - try_load_token_instance(storage, source_chain.clone(), transfer.token_id)?; + let mut source_instance = try_load_token_instance(storage, source_chain.clone(), token_id)?; source_instance.supply = source_instance .supply - .checked_sub(transfer.amount) + .checked_sub(amount) .change_context_lazy(|| Error::TokenSupplyInvariantViolated { - token_id: transfer.token_id, + token_id, chain: source_chain.clone(), })?; - state::save_token_instance(storage, source_chain, transfer.token_id, &source_instance) + state::save_token_instance(storage, source_chain, token_id, &source_instance) .change_context(Error::State) } fn add_amount_to_destination( storage: &mut dyn Storage, destination_chain: ChainNameRaw, - transfer: &InterchainTransfer, + token_id: TokenId, + amount: nonempty::Uint256, ) -> Result<(), Error> { let mut destination_instance = - try_load_token_instance(storage, destination_chain.clone(), transfer.token_id)?; + try_load_token_instance(storage, destination_chain.clone(), token_id)?; destination_instance.supply = destination_instance .supply - .checked_add(transfer.amount) + .checked_add(amount) .change_context_lazy(|| Error::TokenSupplyInvariantViolated { - token_id: transfer.token_id, + token_id, chain: destination_chain.clone(), })?; - state::save_token_instance( - storage, - destination_chain, - transfer.token_id, - &destination_instance, - ) - .change_context(Error::State) + state::save_token_instance(storage, destination_chain, token_id, &destination_instance) + .change_context(Error::State) } fn apply_token_deployment( @@ -546,8 +562,18 @@ fn ensure_matching_original_deployment( Ok(()) } +fn try_load_token_decimals( + storage: &dyn Storage, + chain: ChainNameRaw, + token_id: TokenId, +) -> Result { + try_load_token_instance(storage, chain.clone(), token_id) + .map(|instance| instance.decimals)? + .ok_or(report!(Error::TokenDecimalsNotSet { token_id, chain })) +} + fn try_load_token_instance( - storage: &mut dyn Storage, + storage: &dyn Storage, chain: ChainNameRaw, token_id: TokenId, ) -> Result { diff --git a/contracts/interchain-token-service/src/state.rs b/contracts/interchain-token-service/src/state.rs index ae98e1b1e..40fddaa49 100644 --- a/contracts/interchain-token-service/src/state.rs +++ b/contracts/interchain-token-service/src/state.rs @@ -160,14 +160,6 @@ pub fn save_chain_config( .change_context(Error::Storage) } -pub fn load_token_decimals( - _storage: &dyn Storage, - _chain: &ChainNameRaw, - _token_id: TokenId, -) -> Result { - todo!() -} - pub fn may_load_its_contract( storage: &dyn Storage, chain: &ChainNameRaw,