From 8a488f2823cd19dce651251e774998f4e97cc4bf Mon Sep 17 00:00:00 2001 From: mmaurello <93129175+mmaurello@users.noreply.github.com> Date: Wed, 27 Mar 2024 21:58:20 +1300 Subject: [PATCH] Inform required amount in error message for multi assets (#223) * control destination fee balance before calculating contract fee and throw error if balance not enough * return 0 as fees in multiassets and let UI handle the balance * revert unimplemented option * add changeset * remove check for balances 0 --- .changeset/chilled-pugs-greet.md | 5 ++ .../src/contract/contracts/Xtokens/Xtokens.ts | 1 - .../sdk/src/getTransferData/getSourceData.ts | 47 +++++++++++++++++-- 3 files changed, 47 insertions(+), 6 deletions(-) create mode 100644 .changeset/chilled-pugs-greet.md diff --git a/.changeset/chilled-pugs-greet.md b/.changeset/chilled-pugs-greet.md new file mode 100644 index 00000000..92940e74 --- /dev/null +++ b/.changeset/chilled-pugs-greet.md @@ -0,0 +1,5 @@ +--- +'@moonbeam-network/xcm-sdk': patch +--- + +Inform required amount in error message for multi assets diff --git a/packages/sdk/src/contract/contracts/Xtokens/Xtokens.ts b/packages/sdk/src/contract/contracts/Xtokens/Xtokens.ts index a34b8f00..f2490cde 100644 --- a/packages/sdk/src/contract/contracts/Xtokens/Xtokens.ts +++ b/packages/sdk/src/contract/contracts/Xtokens/Xtokens.ts @@ -70,7 +70,6 @@ export class Xtokens implements TransferContractInterface { if (amount === 0n) { return 0n; } - /** * Contract can throw an error if user balance is smaller than fee. * Or if you try to send 0 as amount. diff --git a/packages/sdk/src/getTransferData/getSourceData.ts b/packages/sdk/src/getTransferData/getSourceData.ts index 8f26d8c9..c0cd819b 100644 --- a/packages/sdk/src/getTransferData/getSourceData.ts +++ b/packages/sdk/src/getTransferData/getSourceData.ts @@ -4,9 +4,17 @@ import { ExtrinsicConfig, SubstrateQueryConfig, } from '@moonbeam-network/xcm-builder'; -import { FeeAssetConfig, TransferConfig } from '@moonbeam-network/xcm-config'; +import { + DestinationFeeConfig, + FeeAssetConfig, + TransferConfig, +} from '@moonbeam-network/xcm-config'; import { AnyChain, AssetAmount } from '@moonbeam-network/xcm-types'; -import { convertDecimals, toBigInt } from '@moonbeam-network/xcm-utils'; +import { + convertDecimals, + toBigInt, + toDecimal, +} from '@moonbeam-network/xcm-utils'; import Big from 'big.js'; import { TransferContractInterface, createContract } from '../contract'; import { PolkadotService } from '../polkadot'; @@ -129,11 +137,17 @@ export async function getSourceData({ fee: destinationFee.amount, feeAsset: chain.getAssetId(destinationFee), }); + const destinationFeeBalanceAmount = zeroDestinationFeeAmount.copyWith({ + amount: destinationFeeBalance, + }); + const fee = await getFee({ balance, chain, contract, decimals: zeroFeeAmount.decimals, + destinationFeeBalanceAmount, + destinationFeeConfig: config.destinationFee, evmSigner, extrinsic, feeConfig: config.fee, @@ -145,9 +159,7 @@ export async function getSourceData({ const { existentialDeposit } = polkadot; const feeAmount = zeroFeeAmount.copyWith({ amount: fee }); const feeBalanceAmount = zeroFeeAmount.copyWith({ amount: feeBalance }); - const destinationFeeBalanceAmount = zeroDestinationFeeAmount.copyWith({ - amount: destinationFeeBalance, - }); + const minAmount = zeroAmount.copyWith({ amount: min }); const maxAmount = getMax({ @@ -207,6 +219,8 @@ export interface GetFeeParams { evmSigner?: EvmSigner; extrinsic?: ExtrinsicConfig; feeConfig?: FeeAssetConfig; + destinationFeeConfig?: DestinationFeeConfig; + destinationFeeBalanceAmount?: AssetAmount; polkadot: PolkadotService; sourceAddress: string; } @@ -216,6 +230,8 @@ export async function getFee({ chain, contract, decimals, + destinationFeeConfig, + destinationFeeBalanceAmount, evmSigner, extrinsic, feeConfig, @@ -227,6 +243,27 @@ export async function getFee({ throw new Error('EVM Signer must be provided'); } + if ( + destinationFeeConfig && + destinationFeeBalanceAmount && + typeof destinationFeeConfig.amount === 'number' + ) { + const destinationFeeBalance = Number( + toDecimal( + destinationFeeBalanceAmount.amount, + destinationFeeBalanceAmount.decimals, + ), + ); + if ( + destinationFeeBalance && + destinationFeeConfig.amount > destinationFeeBalance + ) { + throw new Error( + `Can't get a fee, make sure you have ${destinationFeeConfig?.amount} ${destinationFeeConfig?.asset.originSymbol} needed for fees in destination`, + ); + } + } + return getContractFee(balance, contract, decimals, evmSigner); }