/*! * lib/bitcoin/hd-accounts-helper.js * Copyright © 2019 – Katana Cryptographic Ltd. All Rights Reserved. */ 'use strict' const cp = require('child_process') const LRU = require('lru-cache') const bitcoin = require('bitcoinjs-lib') const bs58check = require('bs58check') const bs58 = require('bs58') const errors = require('../errors') const Logger = require('../logger') const ForkPool = require('../fork-pool') const network = require('./network') const activeNet = network.network const keys = require('../../keys/')[network.key] const addrHelper = require('./addresses-helper') /** * 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 // 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 = 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 = new ForkPool( `${__dirname}/parallel-address-derivation.js`, { networkKey: network.key, max: poolKeys.maxNbChildren, min: poolKeys.minNbChildren, acquireTimeoutMillis: poolKeys.acquireTimeoutMillis } ) this.externalDerivationActivated = true } /** * 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') return '' } 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 {integer} 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 {integer} type - HD Account type (db encoding) * @param {boolean} locked - lock status of the hd account * @returns {integer} */ 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 {integer} 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 { // 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 {bip32} */ getNode(xpub) { if (this.isValid(xpub)) return this.nodes.get(xpub) else return null } /** * Derives an address for an hd account * @param {int} chain - chain to be derived * must have a value on [0,1] for BIP44/BIP49/BIP84 derivation * @param {bip32} chainNode - Parent bip32 used for derivation * @param {int} index - index to be derived * @param {int} type - type of derivation * @returns {Promise - object} returns an object {address: '...', chain: , index: } */ 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 {int} chain - chain to be derived * must have a value on [0,1] for BIP44/BIP49/BIP84 derivation * @param {int[]} indices - array of indices to be derived * @param {int} type - type of derivation * @returns {Promise - object[]} array of {address: '...', chain: , index: } */ 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 let promises = indices.map(index => { return this.deriveAddress(chain, chainNode, index, info.type) }) return Promise.all(promises) } 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 } const msg = await this.derivationPool.enqueue(data) if (msg.status = 'ok') { resolve(msg.addresses) } else { Logger.error(null, 'A problem was met during parallel addresses derivation') reject() } } catch(e) { Logger.error(e, 'A problem was met during parallel addresses derivation') reject(e) } }) } } catch(e) { return Promise.reject(e) } } } module.exports = new HDAccountsHelper()