191 lines
5.6 KiB
JavaScript
191 lines
5.6 KiB
JavaScript
const _ = require('lodash/fp')
|
|
|
|
const { BitGoAPI } = require('@bitgo/sdk-api')
|
|
const { toLegacyAddress, toCashAddress } = require('bchaddrjs')
|
|
|
|
const BN = require('../../../bn')
|
|
|
|
const E = require('../../../error')
|
|
|
|
const pjson = require('../../../../package.json')
|
|
const userAgent = 'Lamassu-Server/' + pjson.version
|
|
|
|
const NAME = 'BitGo'
|
|
|
|
const BITGO_MODULES = {
|
|
BCH: require('@bitgo/sdk-coin-bch'),
|
|
BTC: require('@bitgo/sdk-coin-btc'),
|
|
DASH: require('@bitgo/sdk-coin-dash'),
|
|
LTC: require('@bitgo/sdk-coin-ltc'),
|
|
ZEC: require('@bitgo/sdk-coin-zec'),
|
|
}
|
|
const SUPPORTED_COINS = _.keys(BITGO_MODULES)
|
|
const BCH_CODES = ['BCH', 'TBCH']
|
|
|
|
const getWallet = (account, cryptoCode) => {
|
|
const accessToken = account.token.trim()
|
|
const env = account.environment === 'test' ? 'test' : 'prod'
|
|
const walletId = account[`${cryptoCode}WalletId`]
|
|
|
|
const bitgo = new BitGoAPI({ accessToken, env, userAgent })
|
|
BITGO_MODULES[cryptoCode].register(bitgo)
|
|
|
|
cryptoCode = cryptoCode.toLowerCase()
|
|
const coin = env === 'test' ? `t${cryptoCode}` : cryptoCode
|
|
|
|
return bitgo.coin(coin).wallets().get({ id: walletId })
|
|
}
|
|
|
|
function checkCryptoCode(cryptoCode) {
|
|
if (!SUPPORTED_COINS.includes(cryptoCode)) {
|
|
return Promise.reject(new Error('Unsupported crypto: ' + cryptoCode))
|
|
}
|
|
|
|
return Promise.resolve()
|
|
}
|
|
|
|
function getLegacyAddress(address, cryptoCode) {
|
|
if (!BCH_CODES.includes(cryptoCode)) return address
|
|
|
|
return toLegacyAddress(address)
|
|
}
|
|
|
|
function getCashAddress(address, cryptoCode) {
|
|
if (!BCH_CODES.includes(cryptoCode)) return address
|
|
|
|
return toCashAddress(address)
|
|
}
|
|
|
|
function formatToGetStatus(address, cryptoCode) {
|
|
if (!BCH_CODES.includes(cryptoCode)) return address
|
|
|
|
const [part1, part2] = getLegacyAddress(address, cryptoCode).split(':')
|
|
return part2 || part1
|
|
}
|
|
|
|
function sendCoins(account, tx) {
|
|
const { toAddress, cryptoAtoms, cryptoCode } = tx
|
|
return checkCryptoCode(cryptoCode)
|
|
.then(() => getWallet(account, cryptoCode))
|
|
.then(wallet => {
|
|
const params = {
|
|
address: getLegacyAddress(toAddress, cryptoCode),
|
|
amount: cryptoAtoms.toNumber(),
|
|
walletPassphrase: account[`${cryptoCode}WalletPassphrase`],
|
|
enforceMinConfirmsForChange: false,
|
|
}
|
|
return wallet.send(params)
|
|
})
|
|
.then(result => {
|
|
let fee = parseFloat(result.transfer.feeString)
|
|
let txid = result.transfer.txid
|
|
|
|
return { txid: txid, fee: new BN(fee).decimalPlaces(0) }
|
|
})
|
|
.catch(err => {
|
|
if (err.message === 'insufficient funds')
|
|
throw new E.InsufficientFundsError()
|
|
throw err
|
|
})
|
|
}
|
|
|
|
function balance(account, cryptoCode) {
|
|
return checkCryptoCode(cryptoCode)
|
|
.then(() => getWallet(account, cryptoCode))
|
|
.then(wallet => new BN(wallet._wallet.spendableBalanceString))
|
|
}
|
|
|
|
function newAddress(account, info) {
|
|
return checkCryptoCode(info.cryptoCode)
|
|
.then(() => getWallet(account, info.cryptoCode))
|
|
.then(wallet => {
|
|
return wallet.createAddress().then(result => {
|
|
const address = result.address
|
|
|
|
// If a label was provided, set the label
|
|
if (info.label) {
|
|
return wallet
|
|
.updateAddress({ address: address, label: info.label })
|
|
.then(() => getCashAddress(address, info.cryptoCode))
|
|
}
|
|
|
|
return getCashAddress(address, info.cryptoCode)
|
|
})
|
|
})
|
|
}
|
|
|
|
function getStatus(account, tx, requested) {
|
|
const { toAddress, cryptoCode } = tx
|
|
return checkCryptoCode(cryptoCode)
|
|
.then(() => getWallet(account, cryptoCode))
|
|
.then(wallet =>
|
|
wallet.transfers({
|
|
type: 'receive',
|
|
address: formatToGetStatus(toAddress, cryptoCode),
|
|
}),
|
|
)
|
|
.then(({ transfers }) => {
|
|
const filterConfirmed = _.filter(
|
|
it => it.state === 'confirmed' && it.type === 'receive',
|
|
)
|
|
const filterPending = _.filter(
|
|
it =>
|
|
(it.state === 'confirmed' || it.state === 'unconfirmed') &&
|
|
it.type === 'receive',
|
|
)
|
|
|
|
const sum = _.reduce((acc, val) => val.plus(acc), new BN(0))
|
|
const toBn = _.map(it => new BN(it.valueString))
|
|
|
|
const confirmed = _.compose(sum, toBn, filterConfirmed)(transfers)
|
|
const pending = _.compose(sum, toBn, filterPending)(transfers)
|
|
|
|
if (confirmed.gte(requested))
|
|
return { receivedCryptoAtoms: confirmed, status: 'confirmed' }
|
|
if (pending.gte(requested))
|
|
return { receivedCryptoAtoms: pending, status: 'authorized' }
|
|
if (pending.gt(0))
|
|
return { receivedCryptoAtoms: pending, status: 'insufficientFunds' }
|
|
return { receivedCryptoAtoms: pending, status: 'notSeen' }
|
|
})
|
|
}
|
|
|
|
function newFunding(account, cryptoCode) {
|
|
return checkCryptoCode(cryptoCode).then(() => {
|
|
return getWallet(account, cryptoCode).then(wallet => {
|
|
return wallet.createAddress().then(result => {
|
|
const fundingAddress = result.address
|
|
return wallet
|
|
.updateAddress({ address: fundingAddress, label: 'Funding Address' })
|
|
.then(() => ({
|
|
fundingPendingBalance: new BN(wallet._wallet.balance).minus(
|
|
wallet._wallet.confirmedBalance,
|
|
),
|
|
fundingConfirmedBalance: new BN(wallet._wallet.confirmedBalance),
|
|
fundingAddress: getCashAddress(fundingAddress, cryptoCode),
|
|
}))
|
|
})
|
|
})
|
|
})
|
|
}
|
|
|
|
function cryptoNetwork(account, cryptoCode) {
|
|
return checkCryptoCode(cryptoCode).then(() =>
|
|
account.environment === 'test' ? 'test' : 'main',
|
|
)
|
|
}
|
|
|
|
function checkBlockchainStatus(cryptoCode) {
|
|
return checkCryptoCode(cryptoCode).then(() => Promise.resolve('ready'))
|
|
}
|
|
|
|
module.exports = {
|
|
NAME,
|
|
balance,
|
|
sendCoins,
|
|
newAddress,
|
|
getStatus,
|
|
newFunding,
|
|
cryptoNetwork,
|
|
checkBlockchainStatus,
|
|
}
|