Browse Source

Ripple: handle case the destination does not exist

gre-patch-1
Gaëtan Renaudeau 6 years ago
parent
commit
26d788fe9c
No known key found for this signature in database GPG Key ID: 7B66B85F042E5451
  1. 46
      src/bridge/RippleJSBridge.js
  2. 3
      src/config/errors.js
  3. 3
      static/i18n/en/errors.json

46
src/bridge/RippleJSBridge.js

@ -20,7 +20,7 @@ import {
import FeesRippleKind from 'components/FeesField/RippleKind' import FeesRippleKind from 'components/FeesField/RippleKind'
import AdvancedOptionsRippleKind from 'components/AdvancedOptions/RippleKind' import AdvancedOptionsRippleKind from 'components/AdvancedOptions/RippleKind'
import { getAccountPlaceholderName, getNewAccountPlaceholderName } from 'helpers/accountName' import { getAccountPlaceholderName, getNewAccountPlaceholderName } from 'helpers/accountName'
import { NotEnoughBalance } from 'config/errors' import { NotEnoughBalance, NotEnoughBalanceBecauseDestinationNotCreated } from 'config/errors'
import type { WalletBridge, EditProps } from './types' import type { WalletBridge, EditProps } from './types'
type Transaction = { type Transaction = {
@ -114,7 +114,7 @@ async function signAndBroadcast({ a, t, deviceId, isCancelled, onSigned, onOpera
} }
} }
function isRecipientValid(currency, recipient) { function isRecipientValid(recipient) {
try { try {
bs58check.decode(recipient) bs58check.decode(recipient)
return true return true
@ -241,6 +241,32 @@ const getServerInfo = (map => endpointConfig => {
return f() return f()
})({}) })({})
const recipientIsNew = async (endpointConfig, recipient) => {
if (!isRecipientValid(recipient)) return false
const api = apiForEndpointConfig(endpointConfig)
try {
await api.connect()
try {
await api.getAccountInfo(recipient)
return false
} catch (e) {
if (e.message !== 'actNotFound') {
throw e
}
return true
}
} finally {
api.disconnect()
}
}
const cacheRecipientsNew = {}
const cachedRecipientIsNew= (endpointConfig, recipient) => {
if (recipient in cacheRecipientsNew) return cacheRecipientsNew[recipient]
return (cacheRecipientsNew[recipient] = recipientIsNew(endpointConfig, recipient))
}
const RippleJSBridge: WalletBridge<Transaction> = { const RippleJSBridge: WalletBridge<Transaction> = {
scanAccountsOnDevice: (currency, deviceId) => scanAccountsOnDevice: (currency, deviceId) =>
Observable.create(o => { Observable.create(o => {
@ -446,7 +472,7 @@ const RippleJSBridge: WalletBridge<Transaction> = {
pullMoreOperations: () => Promise.resolve(a => a), // FIXME not implemented pullMoreOperations: () => Promise.resolve(a => a), // FIXME not implemented
isRecipientValid: (currency, recipient) => Promise.resolve(isRecipientValid(currency, recipient)), isRecipientValid: (currency, recipient) => Promise.resolve(isRecipientValid(recipient)),
getRecipientWarning: () => Promise.resolve(null), getRecipientWarning: () => Promise.resolve(null),
createTransaction: () => ({ createTransaction: () => ({
@ -496,10 +522,21 @@ const RippleJSBridge: WalletBridge<Transaction> = {
checkValidTransaction: async (a, t) => { checkValidTransaction: async (a, t) => {
const r = await getServerInfo(a.endpointConfig) const r = await getServerInfo(a.endpointConfig)
const reserveBaseXRP = parseAPIValue(r.validatedLedger.reserveBaseXRP)
if (t.recipient) {
if (await cachedRecipientIsNew(a.endpointConfig, t.recipient)) {
if (t.amount.lt(reserveBaseXRP)) {
const f = formatAPICurrencyXRP(reserveBaseXRP)
throw new NotEnoughBalanceBecauseDestinationNotCreated('', {
minimalAmount: `${f.currency} ${f.value}`,
})
}
}
}
if ( if (
t.amount t.amount
.plus(t.fee) .plus(t.fee)
.plus(parseAPIValue(r.validatedLedger.reserveBaseXRP)) .plus(reserveBaseXRP)
.isLessThanOrEqualTo(a.balance) .isLessThanOrEqualTo(a.balance)
) { ) {
return true return true
@ -513,6 +550,7 @@ const RippleJSBridge: WalletBridge<Transaction> = {
signAndBroadcast: (a, t, deviceId) => signAndBroadcast: (a, t, deviceId) =>
Observable.create(o => { Observable.create(o => {
delete cacheRecipientsNew[t.recipient]
let cancelled = false let cancelled = false
const isCancelled = () => cancelled const isCancelled = () => cancelled
const onSigned = () => { const onSigned = () => {

3
src/config/errors.js

@ -30,6 +30,9 @@ export const ManagerUninstallBTCDep = createCustomErrorClass('ManagerUninstallBT
export const NetworkDown = createCustomErrorClass('NetworkDown') export const NetworkDown = createCustomErrorClass('NetworkDown')
export const NoAddressesFound = createCustomErrorClass('NoAddressesFound') export const NoAddressesFound = createCustomErrorClass('NoAddressesFound')
export const NotEnoughBalance = createCustomErrorClass('NotEnoughBalance') export const NotEnoughBalance = createCustomErrorClass('NotEnoughBalance')
export const NotEnoughBalanceBecauseDestinationNotCreated = createCustomErrorClass(
'NotEnoughBalanceBecauseDestinationNotCreated',
)
export const PasswordsDontMatchError = createCustomErrorClass('PasswordsDontMatch') export const PasswordsDontMatchError = createCustomErrorClass('PasswordsDontMatch')
export const PasswordIncorrectError = createCustomErrorClass('PasswordIncorrect') export const PasswordIncorrectError = createCustomErrorClass('PasswordIncorrect')
export const TimeoutTagged = createCustomErrorClass('TimeoutTagged') export const TimeoutTagged = createCustomErrorClass('TimeoutTagged')

3
static/i18n/en/errors.json

@ -99,6 +99,9 @@
"title": "Oops, insufficient balance", "title": "Oops, insufficient balance",
"description": "Make sure the account to debit has sufficient balance" "description": "Make sure the account to debit has sufficient balance"
}, },
"NotEnoughBalanceBecauseDestinationNotCreated": {
"title": "Destination does not exist. Send at least {{minimalAmount}}"
},
"PasswordsDontMatch": { "PasswordsDontMatch": {
"title": "Passwords don't match", "title": "Passwords don't match",
"description": "Please try again" "description": "Please try again"

Loading…
Cancel
Save