|
|
|
/*!
|
|
|
|
* accounts/xpub-rest-api.js
|
|
|
|
* Copyright © 2019 – Katana Cryptographic Ltd. All Rights Reserved.
|
|
|
|
*/
|
|
|
|
'use strict'
|
|
|
|
|
|
|
|
const validator = require('validator')
|
|
|
|
const bodyParser = require('body-parser')
|
|
|
|
const errors = require('../lib/errors')
|
|
|
|
const network = require('../lib/bitcoin/network')
|
|
|
|
const Logger = require('../lib/logger')
|
|
|
|
const db = require('../lib/db/mysql-db-wrapper')
|
|
|
|
const hdaHelper = require('../lib/bitcoin/hd-accounts-helper')
|
|
|
|
const hdaService = require('../lib/bitcoin/hd-accounts-service')
|
|
|
|
const RpcClient = require('../lib/bitcoind-rpc/rpc-client')
|
|
|
|
const HdAccountInfo = require('../lib/wallet/hd-account-info')
|
|
|
|
const authMgr = require('../lib/auth/authorizations-manager')
|
|
|
|
const HttpServer = require('../lib/http-server/http-server')
|
|
|
|
const remoteImporter = require('../lib/remote-importer/remote-importer')
|
|
|
|
|
|
|
|
const debugApi = !!(process.argv.indexOf('api-debug') > -1)
|
|
|
|
const gap = require('../keys/')[network.key].gap
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* XPub API endpoints
|
|
|
|
*/
|
|
|
|
class XPubRestApi {
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Constructor
|
|
|
|
* @param {pushtx.HttpServer} httpServer - HTTP server
|
|
|
|
*/
|
|
|
|
constructor(httpServer) {
|
|
|
|
this.httpServer = httpServer
|
|
|
|
|
|
|
|
// Initialize the rpc client
|
|
|
|
this.rpcClient = new RpcClient()
|
|
|
|
|
|
|
|
// Establish routes
|
|
|
|
const urlencodedParser = bodyParser.urlencoded({ extended: true })
|
|
|
|
|
|
|
|
this.httpServer.app.post(
|
|
|
|
'/xpub/',
|
|
|
|
urlencodedParser,
|
|
|
|
authMgr.checkAuthentication.bind(authMgr),
|
|
|
|
this.validateArgsPostXpub.bind(this),
|
|
|
|
this.postXpub.bind(this),
|
|
|
|
HttpServer.sendAuthError
|
|
|
|
)
|
|
|
|
|
|
|
|
this.httpServer.app.get(
|
|
|
|
'/xpub/:xpub/import/status',
|
|
|
|
authMgr.checkAuthentication.bind(authMgr),
|
|
|
|
this.validateArgsGetXpub.bind(this),
|
|
|
|
this.getXpubImportStatus.bind(this),
|
|
|
|
HttpServer.sendAuthError
|
|
|
|
)
|
|
|
|
|
|
|
|
this.httpServer.app.get(
|
|
|
|
'/xpub/:xpub',
|
|
|
|
authMgr.checkAuthentication.bind(authMgr),
|
|
|
|
this.validateArgsGetXpub.bind(this),
|
|
|
|
this.getXpub.bind(this),
|
|
|
|
HttpServer.sendAuthError
|
|
|
|
)
|
|
|
|
|
|
|
|
this.httpServer.app.post(
|
|
|
|
'/xpub/:xpub/lock',
|
|
|
|
urlencodedParser,
|
|
|
|
authMgr.checkAuthentication.bind(authMgr),
|
|
|
|
this.validateArgsPostLockXpub.bind(this),
|
|
|
|
this.postLockXpub.bind(this),
|
|
|
|
HttpServer.sendAuthError
|
|
|
|
)
|
|
|
|
|
|
|
|
this.httpServer.app.delete(
|
|
|
|
'/xpub/:xpub',
|
|
|
|
urlencodedParser,
|
|
|
|
authMgr.checkAuthentication.bind(authMgr),
|
|
|
|
this.validateArgsDeleteXpub.bind(this),
|
|
|
|
this.deleteXpub.bind(this),
|
|
|
|
HttpServer.sendAuthError
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Handle xPub POST request
|
|
|
|
* @param {object} req - http request object
|
|
|
|
* @param {object} res - http response object
|
|
|
|
*/
|
|
|
|
async postXpub(req, res) {
|
|
|
|
try {
|
|
|
|
let xpub
|
|
|
|
|
|
|
|
// Check request arguments
|
|
|
|
if (!req.body)
|
|
|
|
return HttpServer.sendError(res, errors.body.NODATA)
|
|
|
|
|
|
|
|
if (!req.body.xpub)
|
|
|
|
return HttpServer.sendError(res, errors.body.NOXPUB)
|
|
|
|
|
|
|
|
if (!req.body.type)
|
|
|
|
return HttpServer.sendError(res, errors.body.NOTYPE)
|
|
|
|
|
|
|
|
// Extracts arguments
|
|
|
|
const argXpub = req.body.xpub
|
|
|
|
const argSegwit = req.body.segwit
|
|
|
|
const argAction = req.body.type
|
|
|
|
const argForceOverride = req.body.force
|
|
|
|
|
|
|
|
// Translate xpub if needed
|
|
|
|
try {
|
|
|
|
const ret = this.xlatHdAccount(argXpub, true)
|
|
|
|
xpub = ret.xpub
|
|
|
|
} catch(e) {
|
|
|
|
return HttpServer.sendError(res, e)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Define the derivation scheme
|
|
|
|
let scheme = hdaHelper.BIP44
|
|
|
|
|
|
|
|
if (argSegwit) {
|
|
|
|
const segwit = argSegwit.toLowerCase()
|
|
|
|
if (segwit == 'bip49')
|
|
|
|
scheme = hdaHelper.BIP49
|
|
|
|
else if (segwit == 'bip84')
|
|
|
|
scheme = hdaHelper.BIP84
|
|
|
|
else
|
|
|
|
return HttpServer.sendError(res, errors.xpub.SEGWIT)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Define default forceOverride if needed
|
|
|
|
const forceOverride = argForceOverride ? argForceOverride : false
|
|
|
|
|
|
|
|
// Process action
|
|
|
|
if (argAction == 'new') {
|
|
|
|
// New hd account
|
|
|
|
try {
|
|
|
|
await hdaService.createHdAccount(xpub, scheme)
|
|
|
|
HttpServer.sendOk(res)
|
|
|
|
} catch(e) {
|
|
|
|
HttpServer.sendError(res, e)
|
|
|
|
}
|
|
|
|
} else if (argAction == 'restore') {
|
|
|
|
// Restore hd account
|
|
|
|
try {
|
|
|
|
await hdaService.restoreHdAccount(xpub, scheme, forceOverride)
|
|
|
|
HttpServer.sendOk(res)
|
|
|
|
} catch(e) {
|
|
|
|
HttpServer.sendError(res, e)
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// Unknown action
|
|
|
|
return HttpServer.sendError(res, errors.body.INVTYPE)
|
|
|
|
}
|
|
|
|
|
|
|
|
} catch(e) {
|
|
|
|
return HttpServer.sendError(res, errors.generic.GEN)
|
|
|
|
|
|
|
|
} finally {
|
|
|
|
debugApi && Logger.info(`API : Completed POST /xpub ${req.body.xpub}`)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Handle xPub GET request
|
|
|
|
* @param {object} req - http request object
|
|
|
|
* @param {object} res - http response object
|
|
|
|
*/
|
|
|
|
async getXpub(req, res) {
|
|
|
|
try {
|
|
|
|
let xpub
|
|
|
|
|
|
|
|
// Extracts arguments
|
|
|
|
const argXpub = req.params.xpub
|
|
|
|
|
|
|
|
// Translate xpub if needed
|
|
|
|
try {
|
|
|
|
const ret = this.xlatHdAccount(argXpub)
|
|
|
|
xpub = ret.xpub
|
|
|
|
} catch(e) {
|
|
|
|
return HttpServer.sendError(res, e)
|
|
|
|
}
|
|
|
|
|
|
|
|
const hdaInfo = new HdAccountInfo(xpub)
|
|
|
|
|
|
|
|
const info = await hdaInfo.loadInfo()
|
|
|
|
if (!info)
|
|
|
|
return Promise.reject()
|
|
|
|
|
|
|
|
const ret = {
|
|
|
|
balance: hdaInfo.finalBalance,
|
|
|
|
unused: {
|
|
|
|
external: hdaInfo.accountIndex,
|
|
|
|
internal: hdaInfo.changeIndex,
|
|
|
|
},
|
|
|
|
derivation: hdaInfo.derivation,
|
|
|
|
created: hdaInfo.created
|
|
|
|
}
|
|
|
|
|
|
|
|
HttpServer.sendOkData(res, ret)
|
|
|
|
|
|
|
|
} catch(e) {
|
|
|
|
Logger.error(e, 'API : XpubRestApi.getXpub()')
|
|
|
|
HttpServer.sendError(res, e)
|
|
|
|
|
|
|
|
} finally {
|
|
|
|
debugApi && Logger.info(`API : Completed GET /xpub/${req.params.xpub}`)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Handle xPub/import/status GET request
|
|
|
|
* @param {object} req - http request object
|
|
|
|
* @param {object} res - http response object
|
|
|
|
*/
|
|
|
|
async getXpubImportStatus(req, res) {
|
|
|
|
try {
|
|
|
|
let xpub
|
|
|
|
|
|
|
|
// Extracts arguments
|
|
|
|
const argXpub = req.params.xpub
|
|
|
|
|
|
|
|
// Translate xpub if needed
|
|
|
|
try {
|
|
|
|
const xlatXpub = this.xlatHdAccount(argXpub)
|
|
|
|
xpub = xlatXpub.xpub
|
|
|
|
} catch(e) {
|
|
|
|
return HttpServer.sendError(res, e)
|
|
|
|
}
|
|
|
|
|
|
|
|
let ret = {
|
|
|
|
import_in_progress: false
|
|
|
|
}
|
|
|
|
|
|
|
|
const status = hdaService.importInProgress(xpub)
|
|
|
|
if (status != null) {
|
|
|
|
ret['import_in_progress'] = true
|
|
|
|
ret['status'] = status['status']
|
|
|
|
if (ret['status'] == remoteImporter.STATUS_RESCAN)
|
|
|
|
ret['hits'] = status['txs_int'] + status['txs_ext']
|
|
|
|
else
|
|
|
|
ret['hits'] = status['txs']
|
|
|
|
}
|
|
|
|
|
|
|
|
HttpServer.sendOkData(res, ret)
|
|
|
|
|
|
|
|
} catch(e) {
|
|
|
|
Logger.error(e, 'API : XpubRestApi.getXpubImportStatus()')
|
|
|
|
HttpServer.sendError(res, e)
|
|
|
|
|
|
|
|
} finally {
|
|
|
|
debugApi && Logger.info(`API : Completed GET /xpub/${req.params.xpub}/import/status`)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Handle Lock XPub POST request
|
|
|
|
* @param {object} req - http request object
|
|
|
|
* @param {object} res - http response object
|
|
|
|
*/
|
|
|
|
async postLockXpub(req, res) {
|
|
|
|
try {
|
|
|
|
let xpub, scheme
|
|
|
|
|
|
|
|
// Check request arguments
|
|
|
|
if (!req.body)
|
|
|
|
return HttpServer.sendError(res, errors.body.NODATA)
|
|
|
|
|
|
|
|
if (!req.body.address)
|
|
|
|
return HttpServer.sendError(res, errors.body.NOADDR)
|
|
|
|
|
|
|
|
if (!req.body.signature)
|
|
|
|
return HttpServer.sendError(res, errors.body.NOSIG)
|
|
|
|
|
|
|
|
if (!req.body.message)
|
|
|
|
return HttpServer.sendError(res, errors.body.NOMSG)
|
|
|
|
|
|
|
|
if (!(req.body.message == 'lock' || req.body.message == 'unlock'))
|
|
|
|
return HttpServer.sendError(res, errors.sig.INVMSG)
|
|
|
|
|
|
|
|
// Extract arguments
|
|
|
|
const argXpub = req.params.xpub
|
|
|
|
const argAddr = req.body.address
|
|
|
|
const argSig = req.body.signature
|
|
|
|
const argMsg = req.body.message
|
|
|
|
|
|
|
|
// Translate xpub if needed
|
|
|
|
try {
|
|
|
|
const ret = this.xlatHdAccount(argXpub)
|
|
|
|
xpub = ret.xpub
|
|
|
|
scheme = ret.scheme
|
|
|
|
} catch(e) {
|
|
|
|
return HttpServer.sendError(res, e)
|
|
|
|
}
|
|
|
|
|
|
|
|
try {
|
|
|
|
// Check the signature and process the request
|
|
|
|
await hdaService.verifyXpubSignature(xpub, argAddr, argSig, argMsg, scheme)
|
|
|
|
const lock = (argMsg == 'unlock') ? false : true
|
|
|
|
const ret = await hdaService.lockHdAccount(xpub, lock)
|
|
|
|
HttpServer.sendOkData(res, {derivation: ret})
|
|
|
|
} catch(e) {
|
|
|
|
HttpServer.sendError(res, errors.generic.GEN)
|
|
|
|
}
|
|
|
|
|
|
|
|
} finally {
|
|
|
|
debugApi && Logger.info(`API : Completed POST /xpub/${req.params.xpub}/lock`)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Handle XPub DELETE request
|
|
|
|
* @param {object} req - http request object
|
|
|
|
* @param {object} res - http response object
|
|
|
|
*/
|
|
|
|
async deleteXpub(req, res) {
|
|
|
|
try {
|
|
|
|
let xpub, scheme
|
|
|
|
|
|
|
|
// Check request arguments
|
|
|
|
if (!req.body)
|
|
|
|
return HttpServer.sendError(res, errors.body.NODATA)
|
|
|
|
|
|
|
|
if (!req.body.address)
|
|
|
|
return HttpServer.sendError(res, errors.body.NOADDR)
|
|
|
|
|
|
|
|
if (!req.body.signature)
|
|
|
|
return HttpServer.sendError(res, errors.body.NOSIG)
|
|
|
|
|
|
|
|
// Extract arguments
|
|
|
|
const argXpub = req.params.xpub
|
|
|
|
const argAddr = req.body.address
|
|
|
|
const argSig = req.body.signature
|
|
|
|
|
|
|
|
// Translate xpub if needed
|
|
|
|
try {
|
|
|
|
const ret = this.xlatHdAccount(argXpub)
|
|
|
|
xpub = ret.xpub
|
|
|
|
scheme = ret.scheme
|
|
|
|
} catch(e) {
|
|
|
|
return HttpServer.sendError(res, e)
|
|
|
|
}
|
|
|
|
|
|
|
|
try {
|
|
|
|
// Check the signature and process the request
|
|
|
|
await hdaService.verifyXpubSignature(xpub, argAddr, argSig, argXpub, scheme)
|
|
|
|
await hdaService.deleteHdAccount(xpub)
|
|
|
|
HttpServer.sendOk(res)
|
|
|
|
} catch(e) {
|
|
|
|
HttpServer.sendError(res, e)
|
|
|
|
}
|
|
|
|
|
|
|
|
} catch(e) {
|
|
|
|
HttpServer.sendError(res, errors.generic.GEN)
|
|
|
|
|
|
|
|
} finally {
|
|
|
|
debugApi && Logger.info(`API : Completed DELETE /xpub/${req.params.xpub}`)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Translate a ypub/zpub into a xpub
|
|
|
|
* @param {string} origXpub - original hd account
|
|
|
|
* @param {boolean} trace - flag indicating if we shoudl trace the conversion in our logs
|
|
|
|
* @returns {object} returns an object {xpub: <translated_xpub>, scheme: <derivation_scheme>}
|
|
|
|
* or raises an exception
|
|
|
|
*/
|
|
|
|
xlatHdAccount(origXpub, trace) {
|
|
|
|
try {
|
|
|
|
// Translate xpub if needed
|
|
|
|
let xpub = origXpub
|
|
|
|
let scheme = hdaHelper.BIP44
|
|
|
|
|
|
|
|
const isYpub = hdaHelper.isYpub(origXpub)
|
|
|
|
const isZpub = hdaHelper.isZpub(origXpub)
|
|
|
|
|
|
|
|
if (isYpub || isZpub) {
|
|
|
|
xpub = hdaHelper.xlatXPUB(origXpub)
|
|
|
|
scheme = isYpub ? hdaHelper.BIP49 : hdaHelper.BIP84
|
|
|
|
if (trace) {
|
|
|
|
Logger.info('API : Converted: ' + origXpub)
|
|
|
|
Logger.info('API : Resulting xpub: ' + xpub)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!hdaHelper.isValid(xpub))
|
|
|
|
throw errors.xpub.INVALID
|
|
|
|
|
|
|
|
return {
|
|
|
|
xpub: xpub,
|
|
|
|
scheme: scheme
|
|
|
|
}
|
|
|
|
|
|
|
|
} catch(e) {
|
|
|
|
const err = (e == errors.xpub.PRIVKEY) ? e : errors.xpub.INVALID
|
|
|
|
throw err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Validate arguments of postXpub requests
|
|
|
|
* @param {object} req - http request object
|
|
|
|
* @param {object} res - http response object
|
|
|
|
* @param {function} next - next express middleware
|
|
|
|
*/
|
|
|
|
validateArgsPostXpub(req, res, next) {
|
|
|
|
const isValidXpub = validator.isAlphanumeric(req.body.xpub)
|
|
|
|
|
|
|
|
const isValidSegwit =
|
|
|
|
!req.body.segwit
|
|
|
|
|| validator.isAlphanumeric(req.body.segwit)
|
|
|
|
|
|
|
|
const isValidType =
|
|
|
|
!req.body.type
|
|
|
|
|| validator.isAlphanumeric(req.body.type)
|
|
|
|
|
|
|
|
const isValidForce =
|
|
|
|
!req.body.force
|
|
|
|
|| validator.isAlphanumeric(req.body.force)
|
|
|
|
|
|
|
|
if (!(isValidXpub && isValidSegwit && isValidType && isValidForce)) {
|
|
|
|
HttpServer.sendError(res, errors.body.INVDATA)
|
|
|
|
Logger.error(
|
|
|
|
req.body,
|
|
|
|
'API : XpubRestApi.validateArgsPostXpub() : Invalid arguments'
|
|
|
|
)
|
|
|
|
} else {
|
|
|
|
next()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Validate arguments of getXpub requests
|
|
|
|
* @param {object} req - http request object
|
|
|
|
* @param {object} res - http response object
|
|
|
|
* @param {function} next - next express middleware
|
|
|
|
*/
|
|
|
|
validateArgsGetXpub(req, res, next) {
|
|
|
|
const isValidXpub = validator.isAlphanumeric(req.params.xpub)
|
|
|
|
|
|
|
|
if (!isValidXpub) {
|
|
|
|
HttpServer.sendError(res, errors.body.INVDATA)
|
|
|
|
Logger.error(
|
|
|
|
req.params.xpub,
|
|
|
|
'API : XpubRestApi.validateArgsGetXpub() : Invalid arguments'
|
|
|
|
)
|
|
|
|
} else {
|
|
|
|
next()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Validate arguments of postLockXpub requests
|
|
|
|
* @param {object} req - http request object
|
|
|
|
* @param {object} res - http response object
|
|
|
|
* @param {function} next - next express middleware
|
|
|
|
*/
|
|
|
|
validateArgsPostLockXpub(req, res, next) {
|
|
|
|
const isValidXpub = validator.isAlphanumeric(req.params.xpub)
|
|
|
|
const isValidAddr = validator.isAlphanumeric(req.body.address)
|
|
|
|
const isValidSig = validator.isBase64(req.body.signature)
|
|
|
|
const isValidMsg = validator.isAlphanumeric(req.body.message)
|
|
|
|
|
|
|
|
if (!(isValidXpub && isValidAddr && isValidSig && isValidMsg)) {
|
|
|
|
HttpServer.sendError(res, errors.body.INVDATA)
|
|
|
|
Logger.error(
|
|
|
|
req.params,
|
|
|
|
'API : XpubRestApi.validateArgsPostLockXpub() : Invalid arguments'
|
|
|
|
)
|
|
|
|
Logger.error(req.body, '')
|
|
|
|
} else {
|
|
|
|
next()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Validate arguments of deleteXpub requests
|
|
|
|
* @param {object} req - http request object
|
|
|
|
* @param {object} res - http response object
|
|
|
|
* @param {function} next - next express middleware
|
|
|
|
*/
|
|
|
|
validateArgsDeleteXpub(req, res, next) {
|
|
|
|
const isValidXpub = validator.isAlphanumeric(req.params.xpub)
|
|
|
|
const isValidAddr = validator.isAlphanumeric(req.body.address)
|
|
|
|
const isValidSig = validator.isBase64(req.body.signature)
|
|
|
|
|
|
|
|
if (!(isValidXpub && isValidAddr && isValidSig)) {
|
|
|
|
HttpServer.sendError(res, errors.body.INVDATA)
|
|
|
|
Logger.error(
|
|
|
|
req.params,
|
|
|
|
'API : XpubRestApi.validateArgsDeleteXpub() : Invalid arguments'
|
|
|
|
)
|
|
|
|
Logger.error(req.body, '')
|
|
|
|
} else {
|
|
|
|
next()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
module.exports = XPubRestApi
|