You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 
 

435 lines
12 KiB

/*!
* lib/bitcoin/hd-accounts-helper.js
* Copyright © 2019 – Katana Cryptographic Ltd. All Rights Reserved.
*/
'use strict'
const LRU = require('lru-cache')
const workerPool = require('workerpool')
const bitcoin = require('bitcoinjs-lib')
const bs58check = require('bs58check')
const bs58 = require('bs58')
const errors = require('../errors')
const Logger = require('../logger')
const network = require('./network')
const activeNet = network.network
const keys = require('../../keys/')[network.key]
const addrHelper = require('./addresses-helper')
const MAX_SAFE_INT_32 = Math.pow(2, 31) - 1;
/**
* A singleton providing HD Accounts helper functions
*/
class HDAccountsHelper {
/**
* Constructor
*/
constructor() {
// HD accounts types
this.BIP44 = 0
this.BIP49 = 1
this.BIP84 = 2
this.LOCKED = 1<<7
// known HD accounts
this.RICOCHET_ACCT = MAX_SAFE_INT_32;
this.POSTMIX_ACCT = MAX_SAFE_INT_32 - 1;
this.PREMIX_ACCT = MAX_SAFE_INT_32 - 2;
this.BADBANK_ACCT = MAX_SAFE_INT_32 - 3;
// Magic numbers
this.MAGIC_XPUB = 0x0488b21e
this.MAGIC_TPUB = 0x043587cf
this.MAGIC_YPUB = 0x049d7cb2
this.MAGIC_UPUB = 0x044a5262
this.MAGIC_ZPUB = 0x04b24746
this.MAGIC_VPUB = 0x045f1cf6
// HD accounts cache
this.nodes = new LRU({
// Maximum number of nodes to store in cache
max: 1000,
// Function used to compute length of item
length: (n, key) => 1,
// Maximum age for items in the cache. Items do not expire
maxAge: Infinity
})
// Default = external addresses derivation deactivated
this.externalDerivationActivated = false
this.derivationPool = null
}
/**
* Activate external derivation of addresses
* (provides improved performances)
*/
activateExternalDerivation() {
// Pool of child processes used for derivation of addresses
const poolKeys = keys.addrDerivationPool
this.derivationPool = workerPool.pool(
`${__dirname}/parallel-address-derivation.js`,
{
maxWorkers: poolKeys.maxNbChildren,
minWorkers: poolKeys.minNbChildren,
workerType: 'thread'
}
)
this.externalDerivationActivated = true
Logger.info(`Created ${poolKeys.minNbChildren} worker threads for addresses derivation (max = ${poolKeys.maxNbChildren})`)
}
/**
* Check if a string encodes a xpub/tpub
* @param {string} xpub - extended public key to be checked
* @returns {boolean} returns true if xpub encodes a xpub/tpub, false otherwise
*/
isXpub(xpub) {
return (xpub.indexOf('xpub') === 0) || (xpub.indexOf('tpub') === 0)
}
/**
* Check if a string encodes a ypub/upub
* @param {string} xpub - extended public key to be checked
* @returns {boolean} returns true if xpub encodes a ypub/upub, false otherwise
*/
isYpub(xpub) {
return (xpub.indexOf('ypub') === 0) || (xpub.indexOf('upub') === 0)
}
/**
* Check if a string encodes a zpub/vpub
* @param {string} xpub - extended public key to be checked
* @returns {boolean} returns true if xpub encodes a zpub/vpub, false otherwise
*/
isZpub(xpub) {
return (xpub.indexOf('zpub') === 0) || (xpub.indexOf('vpub') === 0)
}
/**
* Translates
* - a xpub/ypub/zpub into a xpub
* - a tpub/upub/vpub into a tpub
* @param {string} xpub - extended public key to be translated
* @returns {boolean} returns the translated extended public key
*/
xlatXPUB(xpub) {
const decoded = bs58check.decode(xpub)
const ver = decoded.readInt32BE()
if (
ver !== this.MAGIC_XPUB
&& ver !== this.MAGIC_TPUB
&& ver !== this.MAGIC_YPUB
&& ver !== this.MAGIC_UPUB
&& ver !== this.MAGIC_ZPUB
&& ver !== this.MAGIC_VPUB
) {
Logger.error(null, 'HdAccountsHelper : xlatXPUB() : Incorrect format')
throw errors.xpub.INVALID
}
let xlatVer = 0
switch(ver) {
case this.MAGIC_XPUB:
return xpub
break
case this.MAGIC_YPUB:
xlatVer = this.MAGIC_XPUB
break
case this.MAGIC_ZPUB:
xlatVer = this.MAGIC_XPUB
break
case this.MAGIC_TPUB:
return xpub
break
case this.MAGIC_UPUB:
xlatVer = this.MAGIC_TPUB
break
case this.MAGIC_VPUB:
xlatVer = this.MAGIC_TPUB
break
}
let b = Buffer.alloc(4)
b.writeInt32BE(xlatVer)
decoded.writeInt32BE(xlatVer, 0)
const checksum = bitcoin.crypto.hash256(decoded).slice(0, 4)
const xlatXpub = Buffer.alloc(decoded.length + checksum.length)
decoded.copy(xlatXpub, 0, 0, decoded.length)
checksum.copy(xlatXpub, xlatXpub.length - 4, 0, checksum.length)
const encoded = bs58.encode(xlatXpub)
return encoded
}
/**
* Classify the hd account type retrieved from db
* @param {number} v - HD Account type (db encoding)
* @returns {object} object storing the type and lock status of the hd account
*/
classify(v) {
const ret = {
type: null,
locked: false,
}
let p = v
if (p >= this.LOCKED) {
ret.locked = true
p -= this.LOCKED
}
switch (p) {
case this.BIP44:
case this.BIP49:
case this.BIP84:
ret.type = p
break
}
return ret
}
/**
* Encode hd account type and lock status in db format
* @param {number} type - HD Account type (db encoding)
* @param {boolean} locked - lock status of the hd account
* @returns {number}
*/
makeType(type, locked) {
let p =
(type >= this.LOCKED)
? type - this.LOCKED
: type
locked = !!locked
if (locked)
p += this.LOCKED
return p
}
/**
* Return a string representation of the hd account type
* @param {number} v - HD Account type (db encoding)
* @returns {string}
*/
typeString(v) {
const info = this.classify(v)
const prefix = info.locked ? 'LOCKED ' : ''
let suffix = ''
switch (info.type) {
case this.BIP44:
suffix = 'BIP44'
break
case this.BIP49:
suffix = 'BIP49'
break
case this.BIP84:
suffix = 'BIP84'
break
default:
suffix = 'UNKNOWN'
break
}
return prefix + suffix
}
/**
* Checks if a hd account is a valid bip32
* @param {string} xpub - hd account
* @returns {boolean} returns true if hd account is valid, false otherwise
*/
isValid(xpub) {
if (this.nodes.has(xpub))
return true
try {
if (!(this.isXpub(xpub) || this.isYpub(xpub) || this.isZpub(xpub))) {
throw errors.xpub.INVALID
}
// Translate the xpub
const xlatedXpub = this.xlatXPUB(xpub)
// Parse input as an HD Node. Throws if invalid
const node = bitcoin.bip32.fromBase58(xlatedXpub, activeNet)
// Check and see if this is a private key
if (!node.isNeutered())
throw errors.xpub.PRIVKEY
// Store the external and internal chain nodes in the proper indices.
// Store the parent node as well, at index 2.
this.nodes.set(xpub, [node.derive(0), node.derive(1), node])
return true
} catch(e) {
if (e === errors.xpub.PRIVKEY) throw e
return false
}
}
/**
* Get the hd node associated to an hd account
* @param {string} xpub - hd account
* @returns {[bitcoin.bip32.BIP32Interface, bitcoin.bip32.BIP32Interface, bitcoin.bip32.BIP32Interface]}
*/
getNode(xpub) {
if (this.isValid(xpub))
return this.nodes.get(xpub)
else
return null
}
/**
* Derives an address for an hd account
* @param {number} chain - chain to be derived
* must have a value on [0,1] for BIP44/BIP49/BIP84 derivation
* @param {bitcoin.bip32.BIP32Interface} chainNode - Parent bip32 used for derivation
* @param {number} index - index to be derived
* @param {number} type - type of derivation
* @returns {Promise<object>} returns an object {address: '...', chain: <int>, index: <int>, address: string }
*/
async deriveAddress(chain, chainNode, index, type) {
// Derive M/chain/index
const indexNode = chainNode.derive(index)
const addr = {
chain: chain,
index: index,
}
switch (type) {
case this.BIP44:
addr.address = addrHelper.p2pkhAddress(indexNode.publicKey)
break
case this.BIP49:
addr.address = addrHelper.p2wpkhP2shAddress(indexNode.publicKey)
break
case this.BIP84:
addr.address = addrHelper.p2wpkhAddress(indexNode.publicKey)
break
}
return addr
}
/**
* Derives addresses for an hd account
* @param {string} xpub - hd account to be derived
* @param {number} chain - chain to be derived
* must have a value on [0,1] for BIP44/BIP49/BIP84 derivation
* @param {number[]} indices - array of indices to be derived
* @param {number} type - type of derivation
* @returns {Promise<{ chain: number, index: number, publicKey: Buffer, address: string }[]>} array of address objects
*/
async deriveAddresses(xpub, chain, indices, type) {
const ret = []
try {
const node = this.getNode(xpub)
if (node === null)
throw errors.xpub.INVALID
if (chain > 1 || chain < 0)
throw errors.xpub.CHAIN
if (typeof type == 'undefined')
type = this.makeType(this.BIP44, false)
const info = this.classify(type)
// Node at M/chain
const chainNode = node[chain]
// Optimization: if number of addresses beyond a given treshold
// derivation is done in a child process
if (
!this.externalDerivationActivated
|| indices.length <= keys.addrDerivationPool.thresholdParallelDerivation
) {
// Few addresses to be derived or external derivation deactivated
// Let's do it here
const promises = indices.map(index => {
return this.deriveAddress(chain, chainNode, index, info.type)
})
// Generate additional change address types for postmix account
if (this.isPostmixAcct(node) && chain === 1) {
indices.forEach((index) => {
promises.push(this.deriveAddress(chain, chainNode, index, this.BIP44))
promises.push(this.deriveAddress(chain, chainNode, index, this.BIP49))
})
}
const addresses = await Promise.all(promises)
return addresses;
} else {
// Many addresses to be derived
// Let's do it in a child process
return new Promise(async (resolve, reject) => {
try {
const data = {
xpub: this.xlatXPUB(xpub),
chain: chain,
indices: indices,
type: info.type,
isPostmixChange: this.isPostmixAcct(node) && chain === 1
}
const msg = await this.derivationPool.exec('deriveAddresses', [data])
if (msg.status === 'ok') {
resolve(msg.addresses)
} else {
Logger.error(null, 'HdAccountsHelper : A problem was met during parallel addresses derivation')
reject()
}
} catch(e) {
Logger.error(null, 'HdAccountsHelper : A problem was met during parallel addresses derivation')
reject(e)
}
})
}
} catch(e) {
return Promise.reject(e)
}
}
/**
* @description Detect postmix account
* @param {[bitcoin.bip32.BIP32Interface, bitcoin.bip32.BIP32Interface, bitcoin.bip32.BIP32Interface]} node - array of BIP32 node interfaces
* @returns {boolean}
*/
isPostmixAcct(node) {
const index = node[2].index
const threshold = Math.pow(2,31)
const hardened = (index >= threshold)
const account = hardened ? (index - threshold) : index
return account === this.POSTMIX_ACCT
}
}
module.exports = new HDAccountsHelper()