Browse Source

Merge pull request #848 from meriadec/handle-split

Libcore handling of splitted currencies
master
Gaëtan Renaudeau 7 years ago
committed by GitHub
parent
commit
099f97c240
No known key found for this signature in database GPG Key ID: 4AEE18F83AFDEB23
  1. 15
      src/components/modals/AddAccounts/steps/03-step-import.js
  2. 2
      src/config/constants.js
  3. 3
      src/helpers/accountName.js
  4. 17
      src/helpers/bip32.js
  5. 86
      src/helpers/libcore.js
  6. 2
      static/i18n/en/app.yml

15
src/components/modals/AddAccounts/steps/03-step-import.js

@ -70,10 +70,21 @@ class StepImport extends PureComponent<StepProps> {
const { t } = this.props const { t } = this.props
let { name } = account let { name } = account
const isLegacy = name.indexOf('legacy') !== -1
const isUnsplit = name.indexOf('unsplit') !== -1
if (name === 'New Account') { if (name === 'New Account') {
name = t('app:addAccounts.newAccount') name = t('app:addAccounts.newAccount')
} else if (name.indexOf('legacy') !== -1) { } else if (isLegacy) {
name = t('app:addAccounts.legacyAccount', { accountName: name.replace(' (legacy)', '') }) if (isUnsplit) {
name = t('app:addAccounts.legacyUnsplitAccount', {
accountName: name.replace(' (legacy)', '').replace(' (unsplit)', ''),
})
} else {
name = t('app:addAccounts.legacyAccount', { accountName: name.replace(' (legacy)', '') })
}
} else if (isUnsplit) {
name = t('app:addAccounts.unsplitAccount', { accountName: name.replace(' (unsplit)', '') })
} }
return { return {

2
src/config/constants.js

@ -88,7 +88,7 @@ export const EXPERIMENTAL_MARKET_INDICATOR_SETTINGS = boolFromEnv(
// Other constants // Other constants
export const MAX_ACCOUNT_NAME_SIZE = 30 export const MAX_ACCOUNT_NAME_SIZE = 50
export const MODAL_ADD_ACCOUNTS = 'MODAL_ADD_ACCOUNTS' export const MODAL_ADD_ACCOUNTS = 'MODAL_ADD_ACCOUNTS'
export const MODAL_OPERATION_DETAILS = 'MODAL_OPERATION_DETAILS' export const MODAL_OPERATION_DETAILS = 'MODAL_OPERATION_DETAILS'

3
src/helpers/accountName.js

@ -6,7 +6,8 @@ export const getAccountPlaceholderName = (
c: CryptoCurrency, c: CryptoCurrency,
index: number, index: number,
isLegacy: boolean = false, isLegacy: boolean = false,
) => `${c.name} ${index + 1}${isLegacy ? ' (legacy)' : ''}` isUnsplit: boolean = false,
) => `${c.name} ${index + 1}${isLegacy ? ' (legacy)' : ''}${isUnsplit ? ' (unsplit)' : ''}`
export const getNewAccountPlaceholderName = getAccountPlaceholderName // same naming export const getNewAccountPlaceholderName = getAccountPlaceholderName // same naming
// export const getNewAccountPlaceholderName = (_c: CryptoCurrency, _index: number) => `New Account` // export const getNewAccountPlaceholderName = (_c: CryptoCurrency, _index: number) => `New Account`

17
src/helpers/bip32.js

@ -1,7 +1,24 @@
// @flow // @flow
import type { Account, AccountRaw } from '@ledgerhq/live-common/lib/types' import type { Account, AccountRaw } from '@ledgerhq/live-common/lib/types'
type SplitConfig = {
coinType: number,
}
export const isSegwitPath = (path: string): boolean => path.startsWith("49'") export const isSegwitPath = (path: string): boolean => path.startsWith("49'")
export const isSegwitAccount = (account: Account | AccountRaw): boolean => export const isSegwitAccount = (account: Account | AccountRaw): boolean =>
isSegwitPath(account.freshAddressPath) isSegwitPath(account.freshAddressPath)
export const isUnsplitPath = (path: string, splitConfig: SplitConfig) => {
try {
const coinType = parseInt(path.split('/')[1], 10)
return coinType === splitConfig.coinType
} catch (e) {
return false
}
}
export const isUnsplitAccount = (account: Account | AccountRaw, splitConfig: ?SplitConfig) =>
!!splitConfig && isUnsplitPath(account.freshAddressPath, splitConfig)

86
src/helpers/libcore.js

@ -9,13 +9,23 @@ import { SHOW_LEGACY_NEW_ACCOUNT } from 'config/constants'
import type { AccountRaw, OperationRaw, OperationType } from '@ledgerhq/live-common/lib/types' import type { AccountRaw, OperationRaw, OperationType } from '@ledgerhq/live-common/lib/types'
import type { NJSAccount, NJSOperation } from '@ledgerhq/ledger-core/src/ledgercore_doc' import type { NJSAccount, NJSOperation } from '@ledgerhq/ledger-core/src/ledgercore_doc'
import { isSegwitAccount } from 'helpers/bip32' import { isSegwitAccount, isUnsplitAccount } from 'helpers/bip32'
import * as accountIdHelper from 'helpers/accountId' import * as accountIdHelper from 'helpers/accountId'
import { createCustomErrorClass, deserializeError } from './errors' import { createCustomErrorClass, deserializeError } from './errors'
import { getAccountPlaceholderName, getNewAccountPlaceholderName } from './accountName' import { getAccountPlaceholderName, getNewAccountPlaceholderName } from './accountName'
const NoAddressesFound = createCustomErrorClass('NoAddressesFound') const NoAddressesFound = createCustomErrorClass('NoAddressesFound')
// TODO: put that info inside currency itself
const SPLITTED_CURRENCIES = {
bitcoin_cash: {
coinType: 0,
},
bitcoin_gold: {
coinType: 0,
},
}
export function isValidAddress(core: *, currency: *, address: string): boolean { export function isValidAddress(core: *, currency: *, address: string): boolean {
const addr = new core.NJSAddress(address, currency) const addr = new core.NJSAddress(address, currency)
return addr.isValid(address, currency) return addr.isValid(address, currency)
@ -48,6 +58,7 @@ export function scanAccountsOnDevice(props: Props): Promise<AccountRaw[]> {
...commonParams, ...commonParams,
showNewAccount: !!SHOW_LEGACY_NEW_ACCOUNT || !currency.supportsSegwit, showNewAccount: !!SHOW_LEGACY_NEW_ACCOUNT || !currency.supportsSegwit,
isSegwit: false, isSegwit: false,
isUnsplit: false,
}) })
allAccounts = allAccounts.concat(nonSegwitAccounts) allAccounts = allAccounts.concat(nonSegwitAccounts)
@ -56,10 +67,32 @@ export function scanAccountsOnDevice(props: Props): Promise<AccountRaw[]> {
...commonParams, ...commonParams,
showNewAccount: true, showNewAccount: true,
isSegwit: true, isSegwit: true,
isUnsplit: false,
}) })
allAccounts = allAccounts.concat(segwitAccounts) allAccounts = allAccounts.concat(segwitAccounts)
} }
// TODO: put that info inside currency itself
if (currencyId in SPLITTED_CURRENCIES) {
const splittedAccounts = await scanAccountsOnDeviceBySegwit({
...commonParams,
isSegwit: false,
showNewAccount: false,
isUnsplit: true,
})
allAccounts = allAccounts.concat(splittedAccounts)
if (currency.supportsSegwit) {
const segwitAccounts = await scanAccountsOnDeviceBySegwit({
...commonParams,
showNewAccount: false,
isUnsplit: true,
isSegwit: true,
})
allAccounts = allAccounts.concat(segwitAccounts)
}
}
return allAccounts return allAccounts
}) })
} }
@ -68,12 +101,15 @@ function encodeWalletName({
publicKey, publicKey,
currencyId, currencyId,
isSegwit, isSegwit,
isUnsplit,
}: { }: {
publicKey: string, publicKey: string,
currencyId: string, currencyId: string,
isSegwit: boolean, isSegwit: boolean,
isUnsplit: boolean,
}) { }) {
return `${publicKey}__${currencyId}${isSegwit ? '_segwit' : ''}` const splitConfig = isUnsplit ? SPLITTED_CURRENCIES[currencyId] || null : null
return `${publicKey}__${currencyId}${isSegwit ? '_segwit' : ''}${splitConfig ? '_unsplit' : ''}`
} }
async function scanAccountsOnDeviceBySegwit({ async function scanAccountsOnDeviceBySegwit({
@ -82,6 +118,7 @@ async function scanAccountsOnDeviceBySegwit({
currencyId, currencyId,
onAccountScanned, onAccountScanned,
isSegwit, isSegwit,
isUnsplit,
showNewAccount, showNewAccount,
}: { }: {
core: *, core: *,
@ -90,17 +127,20 @@ async function scanAccountsOnDeviceBySegwit({
onAccountScanned: AccountRaw => void, onAccountScanned: AccountRaw => void,
isSegwit: boolean, // FIXME all segwit to change to 'purpose' isSegwit: boolean, // FIXME all segwit to change to 'purpose'
showNewAccount: boolean, showNewAccount: boolean,
isUnsplit: boolean,
}): Promise<AccountRaw[]> { }): Promise<AccountRaw[]> {
const { coinType } = getCryptoCurrencyById(currencyId) const customOpts =
const { publicKey } = await hwApp.getWalletPublicKey( isUnsplit && SPLITTED_CURRENCIES[currencyId] ? SPLITTED_CURRENCIES[currencyId] : null
`${isSegwit ? '49' : '44'}'/${coinType}'`, const { coinType } = customOpts ? customOpts.coinType : getCryptoCurrencyById(currencyId)
false,
isSegwit, const path = `${isSegwit ? '49' : '44'}'/${coinType}'`
)
const walletName = encodeWalletName({ publicKey, currencyId, isSegwit }) const { publicKey } = await hwApp.getWalletPublicKey(path, false, isSegwit)
const walletName = encodeWalletName({ publicKey, currencyId, isSegwit, isUnsplit })
// retrieve or create the wallet // retrieve or create the wallet
const wallet = await getOrCreateWallet(core, walletName, currencyId, isSegwit) const wallet = await getOrCreateWallet(core, walletName, currencyId, isSegwit, isUnsplit)
const accountsCount = await wallet.getAccountCount() const accountsCount = await wallet.getAccountCount()
// recursively scan all accounts on device on the given app // recursively scan all accounts on device on the given app
@ -115,6 +155,7 @@ async function scanAccountsOnDeviceBySegwit({
accounts: [], accounts: [],
onAccountScanned, onAccountScanned,
isSegwit, isSegwit,
isUnsplit,
showNewAccount, showNewAccount,
}) })
@ -188,6 +229,7 @@ async function scanNextAccount(props: {
accounts: AccountRaw[], accounts: AccountRaw[],
onAccountScanned: AccountRaw => void, onAccountScanned: AccountRaw => void,
isSegwit: boolean, isSegwit: boolean,
isUnsplit: boolean,
showNewAccount: boolean, showNewAccount: boolean,
}): Promise<AccountRaw[]> { }): Promise<AccountRaw[]> {
const { const {
@ -200,6 +242,7 @@ async function scanNextAccount(props: {
accounts, accounts,
onAccountScanned, onAccountScanned,
isSegwit, isSegwit,
isUnsplit,
showNewAccount, showNewAccount,
} = props } = props
@ -222,6 +265,7 @@ async function scanNextAccount(props: {
const account = await buildAccountRaw({ const account = await buildAccountRaw({
njsAccount, njsAccount,
isSegwit, isSegwit,
isUnsplit,
accountIndex, accountIndex,
wallet, wallet,
currencyId, currencyId,
@ -259,6 +303,7 @@ async function getOrCreateWallet(
WALLET_IDENTIFIER: string, WALLET_IDENTIFIER: string,
currencyId: string, currencyId: string,
isSegwit: boolean, isSegwit: boolean,
isUnsplit: boolean,
): NJSWallet { ): NJSWallet {
const pool = core.getPoolInstance() const pool = core.getPoolInstance()
try { try {
@ -266,12 +311,18 @@ async function getOrCreateWallet(
return wallet return wallet
} catch (err) { } catch (err) {
const currency = await pool.getCurrency(currencyId) const currency = await pool.getCurrency(currencyId)
const splitConfig = isUnsplit ? SPLITTED_CURRENCIES[currencyId] || null : null
const coinType = splitConfig ? splitConfig.coinType : '<coin_type>'
const walletConfig = isSegwit const walletConfig = isSegwit
? { ? {
KEYCHAIN_ENGINE: 'BIP49_P2SH', KEYCHAIN_ENGINE: 'BIP49_P2SH',
KEYCHAIN_DERIVATION_SCHEME: "49'/<coin_type>'/<account>'/<node>/<address>", KEYCHAIN_DERIVATION_SCHEME: `49'/${coinType}'/<account>'/<node>/<address>`,
} }
: undefined : splitConfig
? {
KEYCHAIN_DERIVATION_SCHEME: `44'/${coinType}'/<account>'/<node>/<address>`,
}
: undefined
const njsWalletConfig = createWalletConfig(core, walletConfig) const njsWalletConfig = createWalletConfig(core, walletConfig)
const wallet = await core const wallet = await core
.getPoolInstance() .getPoolInstance()
@ -283,6 +334,7 @@ async function getOrCreateWallet(
async function buildAccountRaw({ async function buildAccountRaw({
njsAccount, njsAccount,
isSegwit, isSegwit,
isUnsplit,
wallet, wallet,
currencyId, currencyId,
core, core,
@ -291,6 +343,7 @@ async function buildAccountRaw({
}: { }: {
njsAccount: NJSAccount, njsAccount: NJSAccount,
isSegwit: boolean, isSegwit: boolean,
isUnsplit: boolean,
wallet: NJSWallet, wallet: NJSWallet,
currencyId: string, currencyId: string,
accountIndex: number, accountIndex: number,
@ -336,6 +389,7 @@ async function buildAccountRaw({
currency, currency,
accountIndex, accountIndex,
(currency.supportsSegwit && !isSegwit) || false, (currency.supportsSegwit && !isSegwit) || false,
isUnsplit,
) )
const rawAccount: AccountRaw = { const rawAccount: AccountRaw = {
@ -411,6 +465,8 @@ function buildOperationRaw({
export async function syncAccount({ rawAccount, core }: { core: *, rawAccount: AccountRaw }) { export async function syncAccount({ rawAccount, core }: { core: *, rawAccount: AccountRaw }) {
const decodedAccountId = accountIdHelper.decode(rawAccount.id) const decodedAccountId = accountIdHelper.decode(rawAccount.id)
const isSegwit = isSegwitAccount(rawAccount)
const isUnsplit = isUnsplitAccount(rawAccount, SPLITTED_CURRENCIES[rawAccount.currencyId])
let njsWallet let njsWallet
try { try {
njsWallet = await core.getPoolInstance().getWallet(decodedAccountId.walletName) njsWallet = await core.getPoolInstance().getWallet(decodedAccountId.walletName)
@ -420,7 +476,8 @@ export async function syncAccount({ rawAccount, core }: { core: *, rawAccount: A
core, core,
decodedAccountId.walletName, decodedAccountId.walletName,
rawAccount.currencyId, rawAccount.currencyId,
isSegwitAccount(rawAccount), isSegwit,
isUnsplit,
) )
} }
@ -443,7 +500,8 @@ export async function syncAccount({ rawAccount, core }: { core: *, rawAccount: A
const syncedRawAccount = await buildAccountRaw({ const syncedRawAccount = await buildAccountRaw({
njsAccount, njsAccount,
isSegwit: isSegwitAccount(rawAccount), isSegwit,
isUnsplit,
accountIndex: rawAccount.index, accountIndex: rawAccount.index,
wallet: njsWallet, wallet: njsWallet,
currencyId: rawAccount.currencyId, currencyId: rawAccount.currencyId,

2
static/i18n/en/app.yml

@ -169,6 +169,8 @@ addAccounts:
editName: Edit name editName: Edit name
newAccount: New account newAccount: New account
legacyAccount: '{{accountName}} (legacy)' legacyAccount: '{{accountName}} (legacy)'
legacyUnsplitAccount: '{{accountName}} (legacy) (unsplit)'
unsplitAccount: '{{accountName}} (unsplit)'
noAccountToImport: No existing {{currencyName}} accounts to add noAccountToImport: No existing {{currencyName}} accounts to add
success: Account successfully added to your portfolio success: Account successfully added to your portfolio
success_plural: Accounts successfully added to your portfolio success_plural: Accounts successfully added to your portfolio

Loading…
Cancel
Save