|
@ -1,9 +1,11 @@ |
|
|
// @flow
|
|
|
// @flow
|
|
|
|
|
|
|
|
|
import axios from 'axios' |
|
|
// import axios from 'axios'
|
|
|
import bitcoin from 'bitcoinjs-lib' |
|
|
import bitcoin from 'bitcoinjs-lib' |
|
|
import { formatCurrencyUnit } from '@ledgerhq/common/lib/data/currency' |
|
|
import { formatCurrencyUnit } from '@ledgerhq/common/lib/data/currency' |
|
|
|
|
|
|
|
|
|
|
|
const blockexplorer = require('blockchain.info/blockexplorer').usingNetwork(3) |
|
|
|
|
|
|
|
|
export function format(v: string | number, options: Object = { alwaysShowSign: true }) { |
|
|
export function format(v: string | number, options: Object = { alwaysShowSign: true }) { |
|
|
return formatCurrencyUnit( |
|
|
return formatCurrencyUnit( |
|
|
{ |
|
|
{ |
|
@ -30,13 +32,26 @@ export const networks = [ |
|
|
] |
|
|
] |
|
|
|
|
|
|
|
|
export function computeTransaction(addresses: Array<*>) { |
|
|
export function computeTransaction(addresses: Array<*>) { |
|
|
|
|
|
// return (transaction: Object) => {
|
|
|
|
|
|
// const outputVal = transaction.outputs
|
|
|
|
|
|
// .filter(o => addresses.includes(o.address))
|
|
|
|
|
|
// .reduce((acc, cur) => acc + cur.value, 0)
|
|
|
|
|
|
// const inputVal = transaction.inputs
|
|
|
|
|
|
// .filter(i => addresses.includes(i.address))
|
|
|
|
|
|
// .reduce((acc, cur) => acc + cur.value, 0)
|
|
|
|
|
|
// const balance = outputVal - inputVal
|
|
|
|
|
|
// return {
|
|
|
|
|
|
// ...transaction,
|
|
|
|
|
|
// balance,
|
|
|
|
|
|
// }
|
|
|
|
|
|
// }
|
|
|
return (transaction: Object) => { |
|
|
return (transaction: Object) => { |
|
|
const outputVal = transaction.outputs |
|
|
const outputVal = transaction.out |
|
|
.filter(o => addresses.includes(o.address)) |
|
|
.filter(o => addresses.includes(o.addr)) |
|
|
.reduce((acc, cur) => acc + cur.value, 0) |
|
|
.reduce((acc, cur) => acc + cur.value, 0) |
|
|
const inputVal = transaction.inputs |
|
|
const inputVal = transaction.inputs |
|
|
.filter(i => addresses.includes(i.address)) |
|
|
.filter(i => addresses.includes(i.prev_out.addr)) |
|
|
.reduce((acc, cur) => acc + cur.value, 0) |
|
|
.reduce((acc, cur) => acc + cur.prev_out.value, 0) |
|
|
const balance = outputVal - inputVal |
|
|
const balance = outputVal - inputVal |
|
|
return { |
|
|
return { |
|
|
...transaction, |
|
|
...transaction, |
|
@ -46,11 +61,12 @@ export function computeTransaction(addresses: Array<*>) { |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
export function getTransactions(addresses: Array<string>) { |
|
|
export function getTransactions(addresses: Array<string>) { |
|
|
return axios.get( |
|
|
// return axios.get(
|
|
|
`http://api.ledgerwallet.com/blockchain/v2/btc_testnet/addresses/${addresses.join( |
|
|
// `http://api.ledgerwallet.com/blockchain/v2/btc_testnet/addresses/${addresses.join(
|
|
|
',', |
|
|
// ',',
|
|
|
)}/transactions?noToken=true`,
|
|
|
// )}/transactions?noToken=true`,
|
|
|
) |
|
|
// )
|
|
|
|
|
|
return blockexplorer.getMultiAddress(addresses) |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
export async function getAccount({ |
|
|
export async function getAccount({ |
|
@ -64,9 +80,11 @@ export async function getAccount({ |
|
|
segwit: boolean, |
|
|
segwit: boolean, |
|
|
network: Object, |
|
|
network: Object, |
|
|
}) { |
|
|
}) { |
|
|
|
|
|
const gapLimit = 20 |
|
|
const script = segwit ? parseInt(network.scriptHash, 10) : parseInt(network.pubKeyHash, 10) |
|
|
const script = segwit ? parseInt(network.scriptHash, 10) : parseInt(network.pubKeyHash, 10) |
|
|
|
|
|
|
|
|
let transactions = [] |
|
|
let transactions = [] |
|
|
|
|
|
let lastAddress = null |
|
|
|
|
|
|
|
|
const pubKeyToSegwitAddress = (pubKey, scriptVersion) => { |
|
|
const pubKeyToSegwitAddress = (pubKey, scriptVersion) => { |
|
|
const script = [0x00, 0x14].concat(Array.from(bitcoin.crypto.hash160(pubKey))) |
|
|
const script = [0x00, 0x14].concat(Array.from(bitcoin.crypto.hash160(pubKey))) |
|
@ -82,26 +100,47 @@ export async function getAccount({ |
|
|
return pubKeyToSegwitAddress(hdnode.getPublicKeyBuffer(), script) |
|
|
return pubKeyToSegwitAddress(hdnode.getPublicKeyBuffer(), script) |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
const nextPath = (index = 0) => { |
|
|
const getAddress = ({ type, index }) => ({ |
|
|
const count = 20 |
|
|
type, |
|
|
const getAddress = path => getPublicAddress({ hdnode, path, script, segwit }) |
|
|
index, |
|
|
|
|
|
address: getPublicAddress({ |
|
|
|
|
|
hdnode, |
|
|
|
|
|
path: `${type === 'external' ? 0 : 1}/${index}`, |
|
|
|
|
|
script, |
|
|
|
|
|
segwit, |
|
|
|
|
|
}), |
|
|
|
|
|
}) |
|
|
|
|
|
|
|
|
|
|
|
const getLastAddress = (addresses, lastTx) => { |
|
|
|
|
|
const address = addresses |
|
|
|
|
|
.filter(a => a.type === 'external') |
|
|
|
|
|
.find(a => a.address === lastTx.addr) || { index: 0 } |
|
|
|
|
|
|
|
|
|
|
|
return getAddress({ type: 'external', index: address.index + 1 }) |
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
return Promise.all( |
|
|
const nextPath = (index = 0) => |
|
|
Array.from(Array(count).keys()).map(v => |
|
|
Promise.all( |
|
|
|
|
|
Array.from(new Array(gapLimit).keys()).map(v => |
|
|
Promise.all([ |
|
|
Promise.all([ |
|
|
getAddress(`0/${v + index}`), // external chain
|
|
|
getAddress({ type: 'external', index: v + index }), |
|
|
getAddress(`1/${v + index}`), // internal chain
|
|
|
getAddress({ type: 'internal', index: v + index }), |
|
|
]), |
|
|
]), |
|
|
), |
|
|
), |
|
|
).then(async results => { |
|
|
).then(async results => { |
|
|
const currentAddresses = results.reduce((result, v) => [...result, ...v], []) |
|
|
const addresses = results.reduce((result, v) => [...result, ...v], []) |
|
|
|
|
|
|
|
|
const { data: { txs } } = await getTransactions(currentAddresses) |
|
|
const listAddresses = addresses.map(a => a.address) |
|
|
|
|
|
|
|
|
transactions = [...transactions, ...txs.map(computeTransaction(currentAddresses))] |
|
|
const { txs } = await getTransactions(listAddresses) |
|
|
|
|
|
|
|
|
if (txs.length > 0) { |
|
|
const hasTransactions = txs.length > 0 |
|
|
return nextPath(index + (count - 1)) |
|
|
|
|
|
|
|
|
transactions = [...transactions, ...txs.map(computeTransaction(listAddresses))] |
|
|
|
|
|
lastAddress = hasTransactions ? getLastAddress(addresses, txs[0].out[0]) : lastAddress |
|
|
|
|
|
|
|
|
|
|
|
if (hasTransactions) { |
|
|
|
|
|
return nextPath(index + (gapLimit - 1)) |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
return { |
|
|
return { |
|
@ -110,9 +149,14 @@ export async function getAccount({ |
|
|
return result |
|
|
return result |
|
|
}, 0), |
|
|
}, 0), |
|
|
transactions, |
|
|
transactions, |
|
|
|
|
|
...(lastAddress !== null |
|
|
|
|
|
? { |
|
|
|
|
|
currentIndex: lastAddress.index, |
|
|
|
|
|
address: lastAddress.address, |
|
|
|
|
|
} |
|
|
|
|
|
: {}), |
|
|
} |
|
|
} |
|
|
}) |
|
|
}) |
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
return nextPath(currentIndex) |
|
|
return nextPath(currentIndex) |
|
|
} |
|
|
} |
|
|