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.
2080 lines
63 KiB
2080 lines
63 KiB
/*!
|
|
* lib/db.js
|
|
* Copyright © 2019 – Katana Cryptographic Ltd. All Rights Reserved.
|
|
*/
|
|
'use strict'
|
|
|
|
const mysql = require('mysql')
|
|
const path = require('path')
|
|
const Logger = require('../logger')
|
|
const util = require('../util')
|
|
const errors = require('../errors')
|
|
const hdaHelper = require('../bitcoin/hd-accounts-helper')
|
|
const network = require('../bitcoin/network')
|
|
const keys = require('../../keys/')[network.key]
|
|
const keysDb = keys.db
|
|
const debug = !!(process.argv.indexOf('db-debug') > -1)
|
|
const queryDebug = !!(process.argv.indexOf('dbquery-debug') > -1)
|
|
|
|
|
|
/**
|
|
* Subqueries used by getAddrAndXpubsNbTransactions()
|
|
*/
|
|
const SUBQUERY_TXIDS_ADDR = '( \
|
|
SELECT `transactions`.`txnTxid` AS txnTxid \
|
|
FROM `outputs` \
|
|
INNER JOIN `transactions` ON `transactions`.`txnID` = `outputs`.`txnID` \
|
|
INNER JOIN `addresses` ON `addresses`.`addrID` = `outputs`.`addrID` \
|
|
WHERE `addresses`.`addrAddress` IN (?) \
|
|
) UNION ( \
|
|
SELECT `transactions`.`txnTxid` AS txnTxid \
|
|
FROM `inputs` \
|
|
INNER JOIN `transactions` ON `transactions`.`txnID` = `inputs`.`txnID` \
|
|
INNER JOIN `outputs` ON `outputs`.`outID` = `inputs`.`outID` \
|
|
INNER JOIN `addresses` ON `addresses`.`addrID` = `outputs`.`addrID` \
|
|
WHERE `addresses`.`addrAddress` IN (?) \
|
|
)'
|
|
|
|
const SUBQUERY_TXIDS_XPUBS = '( \
|
|
SELECT `transactions`.`txnTxid` AS txnTxid \
|
|
FROM `outputs` \
|
|
INNER JOIN `transactions` ON `transactions`.`txnID` = `outputs`.`txnID` \
|
|
INNER JOIN `hd_addresses` ON `hd_addresses`.`addrID` = `outputs`.`addrID` \
|
|
INNER JOIN `hd` ON `hd`.`hdID` = `hd_addresses`.`hdID` \
|
|
WHERE `hd`.`hdXpub` IN (?) \
|
|
) UNION ( \
|
|
SELECT `transactions`.`txnTxid` AS txnTxid \
|
|
FROM `inputs` \
|
|
INNER JOIN `transactions` ON `transactions`.`txnID` = `inputs`.`txnID` \
|
|
INNER JOIN `outputs` ON `outputs`.`outID` = `inputs`.`outID` \
|
|
INNER JOIN `hd_addresses` ON `hd_addresses`.`addrID` = `outputs`.`addrID` \
|
|
INNER JOIN `hd` ON `hd`.`hdID` = `hd_addresses`.`hdID` \
|
|
WHERE `hd`.`hdXpub` IN (?) \
|
|
)'
|
|
|
|
/**
|
|
* Subqueries used by getTxsByAddrAndXpubs()
|
|
*/
|
|
const SUBQUERY_TXS_ADDR = '(\
|
|
SELECT \
|
|
`transactions`.`txnID` AS `txnID`, \
|
|
`transactions`.`txnTxid` AS `txnTxid`, \
|
|
`transactions`.`txnVersion` AS `txnVersion`, \
|
|
`transactions`.`txnLocktime` AS `txnLocktime`, \
|
|
`blocks`.`blockHeight` AS `blockHeight`, \
|
|
LEAST(`transactions`.`txnCreated`, IFNULL(`blocks`.`blockTime`, 32503680000)) AS `time` \
|
|
FROM `transactions` \
|
|
INNER JOIN `outputs` ON `outputs`.`txnID` = `transactions`.`txnID` \
|
|
INNER JOIN `addresses` ON `addresses`.`addrID` = `outputs`.`addrID` \
|
|
LEFT JOIN `blocks` ON `transactions`.`blockID` = `blocks`.`blockID` \
|
|
WHERE `addresses`.`addrAddress` IN (?) \
|
|
) UNION DISTINCT (\
|
|
SELECT \
|
|
`transactions`.`txnID` AS `txnID`, \
|
|
`transactions`.`txnTxid` AS `txnTxid`, \
|
|
`transactions`.`txnVersion` AS `txnVersion`, \
|
|
`transactions`.`txnLocktime` AS `txnLocktime`, \
|
|
`blocks`.`blockHeight` AS `blockHeight`, \
|
|
LEAST(`transactions`.`txnCreated`, IFNULL(`blocks`.`blockTime`, 32503680000)) AS `time` \
|
|
FROM `transactions` \
|
|
INNER JOIN `inputs` ON `inputs`.`txnID` = `transactions`.`txnID` \
|
|
INNER JOIN `outputs` ON `outputs`.`outID` = `inputs`.`outID` \
|
|
INNER JOIN `addresses` ON `addresses`.`addrID` = `outputs`.`addrID` \
|
|
LEFT JOIN `blocks` ON `transactions`.`blockID` = `blocks`.`blockID` \
|
|
WHERE `addresses`.`addrAddress` IN (?) \
|
|
)'
|
|
|
|
const SUBQUERY_TXS_XPUB = '(\
|
|
SELECT \
|
|
`transactions`.`txnID` AS `txnID`, \
|
|
`transactions`.`txnTxid` AS `txnTxid`, \
|
|
`transactions`.`txnVersion` AS `txnVersion`, \
|
|
`transactions`.`txnLocktime` AS `txnLocktime`, \
|
|
`blocks`.`blockHeight` AS `blockHeight`, \
|
|
LEAST(`transactions`.`txnCreated`, IFNULL(`blocks`.`blockTime`, 32503680000)) AS `time` \
|
|
FROM `transactions` \
|
|
INNER JOIN `outputs` ON `outputs`.`txnID` = `transactions`.`txnID` \
|
|
INNER JOIN `addresses` ON `addresses`.`addrID` = `outputs`.`addrID` \
|
|
INNER JOIN `hd_addresses` ON `hd_addresses`.`addrID` = `addresses`.`addrID` \
|
|
INNER JOIN `hd` ON `hd`.`hdID` = `hd_addresses`.`hdID` \
|
|
LEFT JOIN `blocks` ON `transactions`.`blockID` = `blocks`.`blockID` \
|
|
WHERE `hd`.`hdXpub` IN (?) \
|
|
) UNION DISTINCT (\
|
|
SELECT \
|
|
`transactions`.`txnID` AS `txnID`, \
|
|
`transactions`.`txnTxid` AS `txnTxid`, \
|
|
`transactions`.`txnVersion` AS `txnVersion`, \
|
|
`transactions`.`txnLocktime` AS `txnLocktime`, \
|
|
`blocks`.`blockHeight` AS `blockHeight`, \
|
|
LEAST(`transactions`.`txnCreated`, IFNULL(`blocks`.`blockTime`, 32503680000)) AS `time` \
|
|
FROM `transactions` \
|
|
INNER JOIN `inputs` ON `inputs`.`txnID` = `transactions`.`txnID` \
|
|
INNER JOIN `outputs` ON `outputs`.`outID` = `inputs`.`outID` \
|
|
INNER JOIN `addresses` ON `addresses`.`addrID` = `outputs`.`addrID` \
|
|
INNER JOIN `hd_addresses` ON `hd_addresses`.`addrID` = `addresses`.`addrID` \
|
|
INNER JOIN `hd` ON `hd`.`hdID` = `hd_addresses`.`hdID` \
|
|
LEFT JOIN `blocks` ON `transactions`.`blockID` = `blocks`.`blockID` \
|
|
WHERE `hd`.`hdXpub` IN (?) \
|
|
)'
|
|
|
|
const SUBQUERY_UTXOS_ADDR = '(\
|
|
SELECT \
|
|
`transactions`.`txnID` AS `txnID`, \
|
|
null AS `outIndex`, \
|
|
null AS `outAmount`, \
|
|
null AS `outAddress`, \
|
|
`inputs`.`inIndex` AS `inIndex`, \
|
|
`inputs`.`inSequence` AS `inSequence`, \
|
|
`prevTx`.`txnTxid` AS `inOutTxid`, \
|
|
`outputs`.`outIndex` AS `inOutIndex`, \
|
|
`outputs`.`outAmount` AS `inOutAmount`, \
|
|
`addresses`.`addrAddress` AS `inOutAddress`, \
|
|
null AS `hdAddrChain`, \
|
|
null AS `hdAddrIndex`, \
|
|
null AS `hdXpub` \
|
|
FROM `transactions` \
|
|
INNER JOIN `inputs` ON `inputs`.`txnID` = `transactions`.`txnID` \
|
|
INNER JOIN `outputs` ON `outputs`.`outID` = `inputs`.`outID` \
|
|
INNER JOIN `transactions` AS `prevTx` ON `prevTx`.`txnID` = `outputs`.`txnID` \
|
|
INNER JOIN `addresses` ON `addresses`.`addrID` = `outputs`.`addrID` \
|
|
WHERE \
|
|
`transactions`.`txnID` IN (?) AND \
|
|
`addresses`.`addrAddress` IN (?) \
|
|
) UNION ( \
|
|
SELECT \
|
|
`transactions`.`txnID` AS `txnID`, \
|
|
`outputs`.`outIndex` AS `outIndex`, \
|
|
`outputs`.`outAmount` AS `outAmount`, \
|
|
`addresses`.`addrAddress` AS `outAddress`, \
|
|
null AS `inIndex`, \
|
|
null AS `inSequence`, \
|
|
null AS `inOutTxid`, \
|
|
null AS `inOutIndex`, \
|
|
null AS `inOutAmount`, \
|
|
null AS `inOutAddress`, \
|
|
null AS `hdAddrChain`, \
|
|
null AS `hdAddrIndex`, \
|
|
null AS `hdXpub` \
|
|
FROM `transactions` \
|
|
INNER JOIN `outputs` ON `outputs`.`txnID` = `transactions`.`txnID` \
|
|
INNER JOIN `addresses` ON `addresses`.`addrID` = `outputs`.`addrID` \
|
|
WHERE \
|
|
`transactions`.`txnID` IN (?) AND \
|
|
`addresses`.`addrAddress` IN (?) \
|
|
)'
|
|
|
|
const SUBQUERY_UTXOS_XPUB = '(\
|
|
SELECT \
|
|
`transactions`.`txnID` AS `txnID`, \
|
|
null AS `outIndex`, \
|
|
null AS `outAmount`, \
|
|
null AS `outAddress`, \
|
|
`inputs`.`inIndex` AS `inIndex`, \
|
|
`inputs`.`inSequence` AS `inSequence`, \
|
|
`prevTx`.`txnTxid` AS `inOutTxid`, \
|
|
`outputs`.`outIndex` AS `inOutIndex`, \
|
|
`outputs`.`outAmount` AS `inOutAmount`, \
|
|
`addresses`.`addrAddress` AS `inOutAddress`, \
|
|
`hd_addresses`.`hdAddrChain` AS `hdAddrChain`, \
|
|
`hd_addresses`.`hdAddrIndex` AS `hdAddrIndex`, \
|
|
`hd`.`hdXpub` AS `hdXpub` \
|
|
FROM `transactions` \
|
|
INNER JOIN `inputs` ON `inputs`.`txnID` = `transactions`.`txnID` \
|
|
INNER JOIN `outputs` ON `outputs`.`outID` = `inputs`.`outID` \
|
|
INNER JOIN `transactions` AS `prevTx` ON `prevTx`.`txnID` = `outputs`.`txnID` \
|
|
INNER JOIN `addresses` ON `addresses`.`addrID` = `outputs`.`addrID` \
|
|
INNER JOIN `hd_addresses` ON `hd_addresses`.`addrID` = `addresses`.`addrID` \
|
|
INNER JOIN `hd` ON `hd`.`hdID` = `hd_addresses`.`hdID` \
|
|
WHERE \
|
|
`transactions`.`txnID` IN (?) AND \
|
|
`hd`.`hdXpub` IN (?) \
|
|
) UNION ( \
|
|
SELECT \
|
|
`transactions`.`txnID` AS `txnID`, \
|
|
`outputs`.`outIndex` AS `outIndex`, \
|
|
`outputs`.`outAmount` AS `outAmount`, \
|
|
`addresses`.`addrAddress` AS `outAddress`, \
|
|
null AS `inIndex`, \
|
|
null AS `inSequence`, \
|
|
null AS `inOutTxid`, \
|
|
null AS `inOutIndex`, \
|
|
null AS `inOutAmount`, \
|
|
null AS `inOutAddress`, \
|
|
`hd_addresses`.`hdAddrChain` AS `hdAddrChain`, \
|
|
`hd_addresses`.`hdAddrIndex` AS `hdAddrIndex`, \
|
|
`hd`.`hdXpub` AS `hdXpub` \
|
|
FROM `transactions` \
|
|
INNER JOIN `outputs` ON `outputs`.`txnID` = `transactions`.`txnID` \
|
|
INNER JOIN `addresses` ON `addresses`.`addrID` = `outputs`.`addrID` \
|
|
INNER JOIN `hd_addresses` ON `hd_addresses`.`addrID` = `addresses`.`addrID` \
|
|
INNER JOIN `hd` ON `hd`.`hdID` = `hd_addresses`.`hdID` \
|
|
WHERE \
|
|
`transactions`.`txnID` IN (?) AND \
|
|
`hd`.`hdXpub` IN (?) \
|
|
)'
|
|
|
|
const SUBQUERY_GET_TX_OUTS = 'SELECT \
|
|
`transactions`.`txnID`, \
|
|
`transactions`.`txnTxid`, \
|
|
`transactions`.`txnCreated`, \
|
|
`transactions`.`txnVersion`, \
|
|
`transactions`.`txnLocktime`, \
|
|
`blocks`.`blockHeight`, \
|
|
`blocks`.`blockTime`, \
|
|
`outputs`.`outID`, \
|
|
`outputs`.`outIndex`, \
|
|
`outputs`.`outAmount`, \
|
|
`outputs`.`outScript`, \
|
|
`addresses`.`addrAddress`, \
|
|
`hd_addresses`.`hdAddrChain`, \
|
|
`hd_addresses`.`hdAddrIndex`, \
|
|
`hd`.`hdXpub` \
|
|
FROM `transactions` \
|
|
INNER JOIN `outputs` ON `transactions`.`txnID` = `outputs`.`txnID` \
|
|
INNER JOIN `addresses` ON `outputs`.`addrID` = `addresses`.`addrID` \
|
|
LEFT JOIN `hd_addresses` ON `outputs`.`addrID` = `hd_addresses`.`addrID` \
|
|
LEFT JOIN `hd` ON `hd_addresses`.`hdID` = `hd`.`hdID` \
|
|
LEFT JOIN `blocks` ON `transactions`.`blockID` = `blocks`.`blockID` \
|
|
WHERE `transactions`.`txnTxid` = ? \
|
|
ORDER BY `outputs`.`outIndex` ASC'
|
|
|
|
const SUBQUERY_GET_TX_INS = 'SELECT \
|
|
`t_in`.`txnTxid`, \
|
|
`t_in`.`txnCreated`, \
|
|
`t_in`.`txnVersion`, \
|
|
`t_in`.`txnLocktime`, \
|
|
`blocks`.`blockHeight`, \
|
|
`blocks`.`blockTime`, \
|
|
`t_out`.`txnTxid` AS `prevOutTxid`, \
|
|
`inputs`.`inIndex`, \
|
|
`inputs`.`inSequence`, \
|
|
`inputs`.`outID`, \
|
|
`outputs`.`outIndex`, \
|
|
`outputs`.`outAmount`, \
|
|
`outputs`.`outScript`, \
|
|
`addresses`.`addrAddress`, \
|
|
`hd_addresses`.`hdAddrChain`, \
|
|
`hd_addresses`.`hdAddrIndex`, \
|
|
`hd`.`hdXpub` \
|
|
FROM `inputs` \
|
|
INNER JOIN `outputs` ON `outputs`.`outID` = `inputs`.`outID` \
|
|
INNER JOIN `transactions` AS `t_in` ON `t_in`.`txnID` = `inputs`.`txnID` \
|
|
INNER JOIN `transactions` AS `t_out` ON `t_out`.`txnID` = `outputs`.`txnID` \
|
|
INNER JOIN `addresses` ON `outputs`.`addrID` = `addresses`.`addrID` \
|
|
LEFT JOIN `hd_addresses` ON `outputs`.`addrID` = `hd_addresses`.`addrID` \
|
|
LEFT JOIN `hd` ON `hd_addresses`.`hdID` = `hd`.`hdID` \
|
|
LEFT JOIN `blocks` ON `t_in`.`blockID` = `blocks`.`blockID` \
|
|
WHERE `t_in`.`txnTxid` = ? \
|
|
ORDER BY `inputs`.`inIndex` ASC'
|
|
|
|
|
|
/**
|
|
* A singleton providing a MySQL db wrapper
|
|
* Node-mysql doc: https://github.com/felixge/node-mysql
|
|
*/
|
|
class MySqlDbWrapper {
|
|
|
|
/**
|
|
* Constructor
|
|
*/
|
|
constructor() {
|
|
this.dbConfig = null
|
|
// Db connections pool
|
|
this.pool = null
|
|
// Lock preventing multiple reconnects
|
|
this.lockReconnect = false
|
|
// Timer managing reconnects
|
|
this.timerReconnect = null
|
|
}
|
|
|
|
/**
|
|
* Connect the wrapper to the database
|
|
* @param {object} dbConfig - database configuration
|
|
*/
|
|
connect(dbConfig) {
|
|
this.dbConfig = dbConfig
|
|
|
|
try {
|
|
if (this.pool)
|
|
this.handleReconnect()
|
|
else
|
|
this.handleConnect()
|
|
} catch(e) {
|
|
this.handleReconnect()
|
|
}
|
|
|
|
setInterval(this.ping.bind(this), 30000)
|
|
}
|
|
|
|
/**
|
|
* Connect the wrapper to the mysql server
|
|
*/
|
|
handleConnect() {
|
|
try {
|
|
this.pool = mysql.createPool(this.dbConfig)
|
|
Logger.info(`Db Wrapper : Created a database pool of ${this.dbConfig.connectionLimit} connections`)
|
|
|
|
if (debug) {
|
|
this.pool.on('acquire', function (conn) {
|
|
Logger.info(`Db Wrapper : Connection ${conn.threadId} acquired`)
|
|
})
|
|
this.pool.on('enqueue', function (conn) {
|
|
Logger.info('Db Wrapper : Waiting for a new connection slot')
|
|
})
|
|
this.pool.on('release', function (conn) {
|
|
Logger.info(`Db Wrapper : Connection ${conn.threadId} released`)
|
|
})
|
|
}
|
|
} catch(e) {
|
|
Logger.error(err, 'Db Wrapper : handleConnect() : Problem met while trying to initialize a new pool')
|
|
throw e
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Reconnect the wrapper to the mysql server
|
|
*/
|
|
handleReconnect() {
|
|
if (this.pool) {
|
|
// Manage the lock
|
|
if (this.lockReconnect)
|
|
return
|
|
|
|
this.lockReconnect = true
|
|
|
|
if (this.timerReconnect)
|
|
clearTimeout(this.timerReconnect)
|
|
|
|
// Destroy previous pool
|
|
this.pool.end(err => {
|
|
if (err) {
|
|
Logger.error(err, 'Db Wrapper : handleReconnect() : Problem met while terminating the pool')
|
|
this.timerReconnect = setTimeout(this.handleReconnect.bind(this), 2000)
|
|
} else {
|
|
this.handleConnect()
|
|
}
|
|
this.lockReconnect = false
|
|
})
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Ping the mysql server
|
|
*/
|
|
ping() {
|
|
debug && Logger.info(`Db Wrapper : ping() : ${this.pool._freeConnections.length} free connections`)
|
|
|
|
// Iterate over all free connections
|
|
// which might have been disconnected by the mysql server
|
|
for (let c of this.pool._freeConnections) {
|
|
c.query('SELECT 1', (err, res, fields) => {
|
|
if (debug && err) {
|
|
Logger.error(err, `Db Wrapper : ping() : Ping Error`)
|
|
}
|
|
})
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Send a query
|
|
*/
|
|
async _query(query, retries) {
|
|
queryDebug && Logger.info(`Db Wrapper : ${query}`)
|
|
|
|
if (retries == null)
|
|
retries = 5
|
|
|
|
return new Promise((resolve, reject) => {
|
|
try {
|
|
this.pool.query(query, null, async (err, result, fields) => {
|
|
if (err) {
|
|
// Retry the request on lock errors
|
|
if ((err.code == 'ER_LOCK_DEADLOCK' ||
|
|
err.code == 'ER_LOCK_TIMEOUT' ||
|
|
err.code == 'ER_LOCK_WAIT_TIMEOUT') && (retries > 0)
|
|
) {
|
|
try {
|
|
this.queryError('Lock detected. Retry request in a few ms', query)
|
|
const sleepMillis = Math.floor((Math.random() * 100) + 1)
|
|
await new Promise(resolve2 => setTimeout(resolve2, sleepMillis))
|
|
const res = await this._query(query, retries - 1)
|
|
resolve(res)
|
|
} catch(err) {
|
|
reject(err)
|
|
}
|
|
} else {
|
|
reject(err)
|
|
}
|
|
} else {
|
|
queryDebug && Logger.info(`Db Wrapper : ${result}`)
|
|
resolve(result)
|
|
}
|
|
})
|
|
} catch(err) {
|
|
this.queryError(err, query)
|
|
reject(err)
|
|
}
|
|
})
|
|
}
|
|
|
|
/**
|
|
* Log a query error
|
|
*/
|
|
queryError(err, query) {
|
|
Logger.error(err, 'Db Wrapper : query() : Query Error')
|
|
Logger.error(null, `Db Wrapper : ${query}`)
|
|
}
|
|
|
|
/**
|
|
* Get the ID of an address
|
|
* @param {string} address - bitcoin address
|
|
* @returns {integer} returns the address id
|
|
*/
|
|
async getAddressId(address) {
|
|
const sqlQuery = 'SELECT `addrID` FROM `addresses` WHERE `addrAddress` = ?'
|
|
const params = address
|
|
const query = mysql.format(sqlQuery, params)
|
|
const result = await this._query(query)
|
|
|
|
if (result.length > 0)
|
|
return result[0].addrID
|
|
|
|
throw errors.db.ERROR_NO_ADDRESS
|
|
}
|
|
|
|
/**
|
|
* Get the ID of an Address. Ensures that the address exists.
|
|
* @param {string} address - bitcoin address
|
|
* @returns {integer} returns the address id
|
|
*/
|
|
async ensureAddressId(address) {
|
|
const sqlQuery = 'SELECT `addrID` FROM `addresses` WHERE `addrAddress` = ?'
|
|
const params = address
|
|
const query = mysql.format(sqlQuery, params)
|
|
const result = await this._query(query)
|
|
|
|
if (result.length > 0)
|
|
return result[0].addrID
|
|
|
|
const sqlQuery2 = 'INSERT INTO `addresses` SET ?'
|
|
const params2 = { addrAddress: address }
|
|
const query2 = mysql.format(sqlQuery2, params2)
|
|
const result2 = await this._query(query2)
|
|
return result2.insertId
|
|
}
|
|
|
|
/**
|
|
* Get the IDs of an array of Addresses
|
|
* @param {string[]} addresses - array of bitcoin addresses
|
|
* @returns {object} returns a map of addresses to IDs: {[address]: 100}
|
|
*/
|
|
async getAddressesIds(addresses) {
|
|
const ret = {}
|
|
|
|
if (addresses.length == 0)
|
|
return ret
|
|
|
|
const sqlQuery = 'SELECT * FROM `addresses` WHERE `addrAddress` IN (?)'
|
|
const params = [addresses]
|
|
const query = mysql.format(sqlQuery, params)
|
|
const result = await this._query(query)
|
|
|
|
for (let r of result)
|
|
ret[r.addrAddress] = r.addrID
|
|
|
|
return ret
|
|
}
|
|
|
|
/**
|
|
* Bulk insert addresses.
|
|
* @param {string[]} addresses - array of bitcoin addresses
|
|
*/
|
|
async addAddresses(addresses) {
|
|
if (addresses.length == 0)
|
|
return []
|
|
|
|
const sqlQuery = 'INSERT IGNORE INTO `addresses` (addrAddress) VALUES ?'
|
|
const params = [addresses.map(a => [a])]
|
|
const query = mysql.format(sqlQuery, params)
|
|
return this._query(query)
|
|
}
|
|
|
|
/**
|
|
* Bulk select address entries
|
|
* @param {string[]} addresses - array of bitcoin addresses
|
|
*/
|
|
async getAddresses(addresses) {
|
|
if (addresses.length == 0)
|
|
return []
|
|
|
|
const sqlQuery = 'SELECT * FROM `addresses` WHERE `addrAddress` IN (?)'
|
|
const params = [addresses]
|
|
const query = mysql.format(sqlQuery, params)
|
|
return this._query(query)
|
|
}
|
|
|
|
/**
|
|
* Get address balance.
|
|
* @param {string} address - bitcoin address
|
|
* @returns {integer} returns the balance of the address
|
|
*/
|
|
async getAddressBalance(address) {
|
|
if (address == null)
|
|
return null
|
|
|
|
const sqlQuery = 'SELECT SUM(`outputs`.`outAmount`) as balance \
|
|
FROM `addresses` \
|
|
INNER JOIN `outputs` ON `outputs`.`addrID` = `addresses`.`addrID` \
|
|
LEFT JOIN `inputs` ON `outputs`.`outID` = `inputs`.`outID` \
|
|
WHERE \
|
|
`inputs`.`outID` IS NULL AND \
|
|
`addresses`.`addrAddress` = ?'
|
|
|
|
const params = address
|
|
const query = mysql.format(sqlQuery, params)
|
|
const results = await this._query(query)
|
|
|
|
if (results.length == 1) {
|
|
const balance = results[0].balance
|
|
return (balance == null) ? 0 : balance
|
|
}
|
|
|
|
return null
|
|
}
|
|
|
|
/**
|
|
* Get the number of transactions for an address.
|
|
* @param {string} address - bitcoin address
|
|
* @returns {integer} returns the number of transactions for the address
|
|
*/
|
|
async getAddressNbTransactions(address) {
|
|
if(address == null)
|
|
return null
|
|
|
|
const sqlQuery = 'SELECT COUNT(DISTINCT `r`.`txnTxid`) AS nbTxs \
|
|
FROM ( \
|
|
( \
|
|
SELECT `transactions`.`txnTxid` AS txnTxid \
|
|
FROM `outputs` \
|
|
INNER JOIN `transactions` ON `transactions`.`txnID` = `outputs`.`txnID` \
|
|
INNER JOIN `addresses` ON `outputs`.`addrID` = `addresses`.`addrID` \
|
|
WHERE `addresses`.`addrAddress` = ? \
|
|
) UNION ( \
|
|
SELECT `transactions`.`txnTxid` AS txnTxid \
|
|
FROM `inputs` \
|
|
INNER JOIN `transactions` ON `transactions`.`txnID` = `inputs`.`txnID` \
|
|
INNER JOIN `outputs` ON `outputs`.`outID` = `inputs`.`outID` \
|
|
INNER JOIN `addresses` ON `outputs`.`addrID` = `addresses`.`addrID` \
|
|
WHERE `addresses`.`addrAddress` = ? \
|
|
) \
|
|
) AS `r`'
|
|
|
|
const params = [address, address]
|
|
const query = mysql.format(sqlQuery, params)
|
|
const results = await this._query(query)
|
|
|
|
if (results.length == 1) {
|
|
const nbTxs = results[0].nbTxs
|
|
return (nbTxs == null) ? 0 : nbTxs
|
|
}
|
|
|
|
return null
|
|
}
|
|
|
|
/**
|
|
* Get an HD account.
|
|
* @param {string} xpub - xpub
|
|
* @returns {integer} returns {hdID, hdXpub, hdCreated, hdType}
|
|
* throws if no record of xpub
|
|
*/
|
|
async getHDAccount(xpub) {
|
|
const sqlQuery = 'SELECT * FROM `hd` WHERE `hdXpub` = ?'
|
|
const params = xpub
|
|
const query = mysql.format(sqlQuery, params)
|
|
const result = await this._query(query)
|
|
|
|
if (result.length > 0)
|
|
return result[0]
|
|
|
|
throw errors.db.ERROR_NO_HD_ACCOUNT
|
|
}
|
|
|
|
/**
|
|
* Get the ID of an HD account
|
|
* @param {string} xpub - xpub
|
|
* @returns {integer} returns the id of the hd account
|
|
*/
|
|
async getHDAccountId(xpub) {
|
|
const sqlQuery = 'SELECT `hdID` FROM `hd` WHERE `hdXpub` = ?'
|
|
const params = xpub
|
|
const query = mysql.format(sqlQuery, params)
|
|
const result = await this._query(query)
|
|
|
|
if (result.length > 0)
|
|
return result[0].hdID
|
|
|
|
throw errors.db.ERROR_NO_HD_ACCOUNT
|
|
}
|
|
|
|
/**
|
|
* Get the ID of an HD account. Ensures that the account exists.
|
|
* @param {string} xpub - xpub
|
|
* @param {string} type - hd account type
|
|
* @returns {integer} returns the id of the hd account
|
|
*/
|
|
async ensureHDAccountId(xpub, type) {
|
|
const info = hdaHelper.classify(type)
|
|
|
|
if (info.type === null)
|
|
throw errors.xpub.SEGWIT
|
|
|
|
// Get the ID of the xpub
|
|
const sqlQuery = 'SELECT `hdID` FROM `hd` WHERE `hdXpub` = ?'
|
|
const params = xpub
|
|
const query = mysql.format(sqlQuery, params)
|
|
const result = await this._query(query)
|
|
|
|
if (result.length > 0)
|
|
return result[0].hdID
|
|
|
|
const sqlQuery2 = 'INSERT INTO `hd` SET ?'
|
|
const params2 = {
|
|
hdXpub: xpub,
|
|
hdCreated: util.unix(),
|
|
hdType: type,
|
|
}
|
|
const query2 = mysql.format(sqlQuery2, params2)
|
|
const result2 = await this._query(query2)
|
|
return result2.insertId
|
|
|
|
}
|
|
|
|
/**
|
|
* Lock the type of a hd account
|
|
* @param {string} xpub - xpub
|
|
* @returns {boolean} locked - true for locking, false for unlocking
|
|
*/
|
|
async setLockHDAccountType(xpub, locked) {
|
|
locked = !!locked
|
|
|
|
const account = await this.getHDAccount(xpub)
|
|
const info = hdaHelper.classify(account.hdType)
|
|
|
|
if (info.locked == locked)
|
|
return true
|
|
|
|
const hdType = hdaHelper.makeType(account.hdType, locked)
|
|
const sqlQuery = 'UPDATE `hd` SET `hdType` = ? WHERE `hdXpub` = ?'
|
|
const params = [hdType, xpub]
|
|
const query = mysql.format(sqlQuery, params)
|
|
return this._query(query)
|
|
}
|
|
|
|
/**
|
|
* Delete a hd account
|
|
* @param {string} xpub - xpub
|
|
*/
|
|
async deleteHDAccount(xpub) {
|
|
try {
|
|
// Check that this HD account exists
|
|
await this.getHDAccountId(xpub)
|
|
|
|
// Delete addresses associated with this xpub.
|
|
// Address deletion cascades into transaction inputs & outputs.
|
|
const sqlQuery = 'DELETE `addresses`.* FROM `addresses` \
|
|
INNER JOIN `hd_addresses` ON `hd_addresses`.`addrID` = `addresses`.`addrID` \
|
|
INNER JOIN `hd` ON `hd_addresses`.`hdID` = `hd`.`hdID` \
|
|
WHERE `hd`.`hdXpub` = ?'
|
|
const params = xpub
|
|
const query = mysql.format(sqlQuery, params)
|
|
await this._query(query)
|
|
|
|
// Delete HD account entry
|
|
const sqlQuery2 = 'DELETE FROM `hd` WHERE `hdXpub` = ?'
|
|
const params2 = xpub
|
|
const query2 = mysql.format(sqlQuery2, params2)
|
|
return this._query(query2)
|
|
|
|
} catch(e) {}
|
|
}
|
|
|
|
/**
|
|
* Add an address a hd account
|
|
* @param {string} address - bitcoin address
|
|
* @param {string} xpub - xpub
|
|
* @param {integer} chain - derivation chain
|
|
* @param {index} index - derivation index for the address
|
|
*/
|
|
async addAddressToHDAccount(address, xpub, chain, index) {
|
|
const results = await Promise.all([
|
|
this.ensureAddressId(address),
|
|
this.getHDAccountId(xpub)
|
|
])
|
|
|
|
const addrID = results[0]
|
|
const hdID = results[1]
|
|
|
|
if (hdID == null)
|
|
throw null
|
|
|
|
const sqlQuery = 'INSERT INTO `hd_addresses` SET ?'
|
|
const params = {
|
|
hdID: hdID,
|
|
addrID: addrID,
|
|
hdAddrChain: chain,
|
|
hdAddrIndex: index
|
|
}
|
|
const query = mysql.format(sqlQuery, params)
|
|
return this._query(query)
|
|
}
|
|
|
|
/**
|
|
* Bulk-add addresses to an HD account.
|
|
* @param {string} xpub - xpub
|
|
* @param {object[]} addressData - array of {address: '...', chain: <int>, index: <int>}
|
|
* which is the output of the HDAccountsHelper.deriveAddresses()
|
|
*/
|
|
async addAddressesToHDAccount(xpub, addressData) {
|
|
if (addressData.length == 0)
|
|
return
|
|
|
|
const addresses = addressData.map(d => d.address)
|
|
const hdID = await this.getHDAccountId(xpub)
|
|
|
|
// Bulk insert addresses
|
|
await this.addAddresses(addresses)
|
|
|
|
// Bulk get address IDs
|
|
const addrIdMap = await this.getAddressesIds(addresses)
|
|
|
|
// Convert input addressData into database entry format
|
|
const data = []
|
|
for (let entry of addressData) {
|
|
data.push([
|
|
hdID,
|
|
addrIdMap[entry.address],
|
|
entry.chain,
|
|
entry.index
|
|
])
|
|
}
|
|
|
|
const sqlQuery = 'INSERT IGNORE INTO `hd_addresses` \
|
|
(hdID, addrID, hdAddrChain, hdAddrIndex) VALUES ?'
|
|
const params = [data]
|
|
const query = mysql.format(sqlQuery, params)
|
|
return this._query(query)
|
|
}
|
|
|
|
/**
|
|
* Get hd accounts associated to a list of addresses
|
|
* @param {string[]} addresses - array of bitcoin addresses
|
|
* @returns {object[]}
|
|
*/
|
|
async getUngroupedHDAccountsByAddresses(addresses) {
|
|
if (addresses.length == 0) return {}
|
|
|
|
const sqlQuery = 'SELECT \
|
|
`hd`.`hdID`, \
|
|
`hd`.`hdXpub`, \
|
|
`hd`.`hdType`, \
|
|
`addresses`.`addrID`, \
|
|
`addresses`.`addrAddress`, \
|
|
`hd_addresses`.`hdAddrChain`, \
|
|
`hd_addresses`.`hdAddrIndex` \
|
|
FROM `addresses` \
|
|
LEFT JOIN `hd_addresses` ON `hd_addresses`.`addrID` = `addresses`.`addrID` \
|
|
LEFT JOIN `hd` ON `hd_addresses`.`hdID` = `hd`.`hdID` \
|
|
WHERE `addresses`.`addrAddress` IN (?) \
|
|
AND `addresses`.`addrAddress` NOT IN (SELECT addrAddress FROM banned_addresses)'
|
|
|
|
const params = [addresses]
|
|
const query = mysql.format(sqlQuery, params)
|
|
return this._query(query)
|
|
}
|
|
|
|
/**
|
|
* Get any HD accounts that own the input addresses
|
|
* If addresses are known but not associated with an HD account,
|
|
* theyare returned in the `loose` category
|
|
* @param {string[]} addresses - array of bitcoin addresses
|
|
* @returns {object}
|
|
* {
|
|
* hd: {
|
|
* [xpub]: {
|
|
* hdID: N,
|
|
* hdType: M,
|
|
* addresses:[...]
|
|
* },
|
|
* ...
|
|
* }
|
|
* loose:[...]
|
|
* }
|
|
*/
|
|
async getHDAccountsByAddresses(addresses) {
|
|
const ret = {
|
|
hd: {},
|
|
loose: []
|
|
}
|
|
|
|
if (addresses.length == 0)
|
|
return ret
|
|
|
|
const sqlQuery = 'SELECT \
|
|
`hd`.`hdID`, \
|
|
`hd`.`hdXpub`, \
|
|
`hd`.`hdType`, \
|
|
`addresses`.`addrID`, \
|
|
`addresses`.`addrAddress`, \
|
|
`hd_addresses`.`hdAddrChain`, \
|
|
`hd_addresses`.`hdAddrIndex` \
|
|
FROM `addresses` \
|
|
LEFT JOIN `hd_addresses` ON `hd_addresses`.`addrID` = `addresses`.`addrID` \
|
|
LEFT JOIN `hd` ON `hd_addresses`.`hdID` = `hd`.`hdID` \
|
|
WHERE `addresses`.`addrAddress` IN (?) \
|
|
AND `addresses`.`addrAddress` NOT IN (SELECT addrAddress FROM banned_addresses)'
|
|
|
|
const params = [addresses]
|
|
const query = mysql.format(sqlQuery, params)
|
|
const results = await this._query(query)
|
|
|
|
for (let r of results) {
|
|
if (r.hdXpub == null) {
|
|
ret.loose.push({
|
|
addrID: r.addrID,
|
|
addrAddress: r.addrAddress,
|
|
})
|
|
} else {
|
|
if (!ret.hd[r.hdXpub]) {
|
|
ret.hd[r.hdXpub] = {
|
|
hdID: r.hdID,
|
|
hdType: r.hdType,
|
|
addresses: []
|
|
}
|
|
}
|
|
|
|
ret.hd[r.hdXpub].addresses.push({
|
|
addrID: r.addrID,
|
|
addrAddress: r.addrAddress,
|
|
hdAddrChain: r.hdAddrChain,
|
|
hdAddrIndex: r.hdAddrIndex,
|
|
})
|
|
}
|
|
}
|
|
|
|
return ret
|
|
}
|
|
|
|
/**
|
|
* Get an HD account balance
|
|
* @param {string} xpub - xpub
|
|
* @returns {integer} returns the balance of the hd account
|
|
*/
|
|
async getHDAccountBalance(xpub) {
|
|
const sqlQuery = 'SELECT \
|
|
SUM(`outputs`.`outAmount`) as balance \
|
|
FROM `hd_addresses` \
|
|
INNER JOIN `addresses` ON `hd_addresses`.`addrID` = `addresses`.`addrID` \
|
|
INNER JOIN `hd` ON `hd_addresses`.`hdID` = `hd`.`hdID` \
|
|
INNER JOIN `outputs` ON `outputs`.`addrID` = `addresses`.`addrID` \
|
|
LEFT JOIN `inputs` ON `outputs`.`outID` = `inputs`.`outID` \
|
|
WHERE `inputs`.`outID` IS NULL \
|
|
AND `hd`.`hdXpub` = ?'
|
|
|
|
const params = xpub
|
|
const query = mysql.format(sqlQuery, params)
|
|
const results = await this._query(query)
|
|
|
|
if (results.length == 1)
|
|
return (results[0].balance == null) ? 0 : results[0].balance
|
|
|
|
return null
|
|
}
|
|
|
|
/**
|
|
* Get next unused address indices for each HD chain of an account
|
|
* @param {string} xpub - xpub
|
|
* @returns {integer[]} returns an array of unused indices
|
|
* [M/0/X, M/1/Y] -> [X,Y]
|
|
*/
|
|
async getHDAccountNextUnusedIndices(xpub) {
|
|
const sqlQuery = 'SELECT \
|
|
`hd_addresses`.`hdAddrChain`, \
|
|
MAX(`hd_addresses`.`hdAddrIndex`) + 1 AS `nextUnusedIndex` \
|
|
FROM `hd_addresses` \
|
|
INNER JOIN `addresses` ON `hd_addresses`.`addrID` = `addresses`.`addrID` \
|
|
INNER JOIN `hd` ON `hd_addresses`.`hdID` = `hd`.`hdID` \
|
|
INNER JOIN `outputs` ON `outputs`.`addrID` = `addresses`.`addrID` \
|
|
WHERE `hd`.`hdXpub` = ? \
|
|
GROUP BY `hd_addresses`.`hdAddrChain`'
|
|
|
|
const params = xpub
|
|
const query = mysql.format(sqlQuery, params)
|
|
const results = await this._query(query)
|
|
|
|
const ret = [0, 0]
|
|
|
|
for (let r of results)
|
|
if ([0,1].indexOf(r.hdAddrChain) > -1)
|
|
ret[r.hdAddrChain] = r.nextUnusedIndex
|
|
|
|
return ret
|
|
}
|
|
|
|
/**
|
|
* Get the maximum derived address index for each HD chain of an account
|
|
* @param {string} xpub - xpub
|
|
* @returns {integer[]} returns an array of derived indices
|
|
* [M/0/X, M/1/Y] -> [X,Y]
|
|
*/
|
|
async getHDAccountDerivedIndices(xpub) {
|
|
const sqlQuery = 'SELECT \
|
|
`hd_addresses`.`hdAddrChain`, \
|
|
MAX(`hd_addresses`.`hdAddrIndex`) AS `maxDerivedIndex` \
|
|
FROM `hd_addresses` \
|
|
INNER JOIN `addresses` ON `hd_addresses`.`addrID` = `addresses`.`addrID` \
|
|
INNER JOIN `hd` ON `hd_addresses`.`hdID` = `hd`.`hdID` \
|
|
WHERE `hd`.`hdXpub` = ? \
|
|
GROUP BY `hd_addresses`.`hdAddrChain`'
|
|
|
|
const params = xpub
|
|
const query = mysql.format(sqlQuery, params)
|
|
const results = await this._query(query)
|
|
|
|
const ret = [-1, -1]
|
|
|
|
for (let r of results)
|
|
if ([0,1].indexOf(r.hdAddrChain) > -1)
|
|
ret[r.hdAddrChain] = r.maxDerivedIndex
|
|
|
|
return ret
|
|
}
|
|
|
|
/**
|
|
* Get the number of indices derived in an interval for a HD chain
|
|
* @param {string} xpub - xpub
|
|
* @param {integer} chain - HD chain (0 or 1)
|
|
* @param {integer} minIdx - min index of derivation
|
|
* @param {integer} maxIdx - max index of derivation
|
|
* @returns {integer[]} returns an array of number of derived indices
|
|
*/
|
|
async getHDAccountNbDerivedIndices(xpub, chain, minIdx, maxIdx) {
|
|
const sqlQuery = 'SELECT \
|
|
COUNT(`hd_addresses`.`hdAddrIndex`) AS `nbDerivedIndices` \
|
|
FROM `hd_addresses` \
|
|
INNER JOIN `hd` ON `hd_addresses`.`hdID` = `hd`.`hdID` \
|
|
WHERE `hd`.`hdXpub` = ? \
|
|
AND `hd_addresses`.`hdAddrChain` = ? \
|
|
AND `hd_addresses`.`hdAddrIndex` >= ? \
|
|
AND `hd_addresses`.`hdAddrIndex` <= ?'
|
|
|
|
const params = [xpub, chain, minIdx, maxIdx]
|
|
const query = mysql.format(sqlQuery, params)
|
|
const results = await this._query(query)
|
|
|
|
if (results.length == 1) {
|
|
const nbDerivedIndices = results[0].nbDerivedIndices
|
|
return (nbDerivedIndices == null) ? 0 : nbDerivedIndices
|
|
}
|
|
|
|
return 0
|
|
}
|
|
|
|
/**
|
|
* Get the number of transactions for an HD account
|
|
* @param {string} xpub - xpub
|
|
* @returns {integer} returns the balance of the hd account
|
|
*/
|
|
async getHDAccountNbTransactions(xpub) {
|
|
const sqlQuery = 'SELECT COUNT(DISTINCT `r`.`txnTxid`) AS nbTxs \
|
|
FROM ( \
|
|
( \
|
|
SELECT `transactions`.`txnTxid` AS txnTxid \
|
|
FROM `outputs` \
|
|
INNER JOIN `transactions` ON `transactions`.`txnID` = `outputs`.`txnID` \
|
|
INNER JOIN `hd_addresses` ON `hd_addresses`.`addrID` = `outputs`.`addrID` \
|
|
INNER JOIN `hd` ON `hd`.`hdID` = `hd_addresses`.`hdID` \
|
|
WHERE `hd`.`hdXpub` = ? \
|
|
) UNION ( \
|
|
SELECT `transactions`.`txnTxid` AS txnTxid \
|
|
FROM `inputs` \
|
|
INNER JOIN `transactions` ON `transactions`.`txnID` = `inputs`.`txnID` \
|
|
INNER JOIN `outputs` ON `outputs`.`outID` = `inputs`.`outID` \
|
|
INNER JOIN `hd_addresses` ON `hd_addresses`.`addrID` = `outputs`.`addrID` \
|
|
INNER JOIN `hd` ON `hd`.`hdID` = `hd_addresses`.`hdID` \
|
|
WHERE `hd`.`hdXpub` = ? \
|
|
) \
|
|
) AS `r`'
|
|
|
|
const params = [xpub, xpub]
|
|
const query = mysql.format(sqlQuery, params)
|
|
const results = await this._query(query)
|
|
|
|
if (results.length == 1)
|
|
return (results[0].nbTxs == null) ? 0 : results[0].nbTxs
|
|
|
|
return null
|
|
}
|
|
|
|
/**
|
|
* Get the number of transactions for a list of addresses and HD accounts
|
|
* @param {string[]} addresses - array of bitcoin addresses
|
|
* @param {string[]} xpubs - array of xpubs
|
|
* @returns {int} returns the number of transactions
|
|
*/
|
|
async getAddrAndXpubsNbTransactions(addresses, xpubs) {
|
|
if (
|
|
(!addresses || addresses.length == 0)
|
|
&& (!xpubs || xpubs.length == 0)
|
|
) {
|
|
return []
|
|
}
|
|
|
|
// Prepares subqueries for the query
|
|
// retrieving txs of interest
|
|
let subQuery = ''
|
|
let subQueries = []
|
|
|
|
if (addresses && addresses.length > 0) {
|
|
const params = [addresses, addresses]
|
|
subQuery = mysql.format(SUBQUERY_TXIDS_ADDR, params)
|
|
subQueries.push(subQuery)
|
|
}
|
|
|
|
if (xpubs && xpubs.length > 0) {
|
|
const params = [xpubs, xpubs]
|
|
subQuery = mysql.format(SUBQUERY_TXIDS_XPUBS, params)
|
|
subQueries.push(subQuery)
|
|
}
|
|
|
|
subQuery = subQueries.join(' UNION ')
|
|
|
|
const sqlQuery = 'SELECT COUNT(DISTINCT `r`.`txnTxid`) AS nbTxs \
|
|
FROM (' + subQuery + ') AS `r`'
|
|
|
|
let query = mysql.format(sqlQuery)
|
|
const results = await this._query(query)
|
|
|
|
if (results.length == 1)
|
|
return (results[0].nbTxs == null) ? 0 : results[0].nbTxs
|
|
|
|
return null
|
|
}
|
|
|
|
/**
|
|
* Get the transactions for a list of addresses and HD accounts
|
|
* @param {string[]} addresses - array of bitcoin addresses
|
|
* @param {string[]} xpubs - array of xpubs
|
|
* @returns {object[]} returns an array of transactions
|
|
*/
|
|
async getTxsByAddrAndXpubs(addresses, xpubs, page, nbTxsPerPage) {
|
|
if (
|
|
(!addresses || addresses.length == 0)
|
|
&& (!xpubs || xpubs.length == 0)
|
|
) {
|
|
return []
|
|
}
|
|
|
|
// Manages the paging
|
|
if (page == null)
|
|
page = 0
|
|
|
|
if (nbTxsPerPage == null)
|
|
nbTxsPerPage = keys.multiaddr.transactions
|
|
|
|
const skip = page * nbTxsPerPage
|
|
|
|
// Prepares subqueries for the query
|
|
// retrieving txs of interest
|
|
let subQuery = ''
|
|
let subQueries = []
|
|
|
|
if (addresses && addresses.length > 0) {
|
|
const params = [addresses, addresses]
|
|
subQuery = mysql.format(SUBQUERY_TXS_ADDR, params)
|
|
subQueries.push(subQuery)
|
|
}
|
|
|
|
if (xpubs && xpubs.length > 0) {
|
|
const params = [xpubs, xpubs]
|
|
subQuery = mysql.format(SUBQUERY_TXS_XPUB, params)
|
|
subQueries.push(subQuery)
|
|
}
|
|
|
|
subQuery = subQueries.join(' UNION DISTINCT ')
|
|
|
|
// Get a page of transactions
|
|
const sqlQuery = 'SELECT \
|
|
`txs`.`txnID`, \
|
|
`txs`.`txnTxid`, \
|
|
`txs`.`txnVersion`, \
|
|
`txs`.`txnLocktime`, \
|
|
`txs`.`blockHeight`, \
|
|
`txs`.`time` \
|
|
FROM (' + subQuery + ') AS txs \
|
|
ORDER BY `txs`.`time` DESC, `txs`.`txnID` DESC \
|
|
LIMIT ?,?'
|
|
const params = [skip, nbTxsPerPage]
|
|
let query = mysql.format(sqlQuery, params)
|
|
const txs = await this._query(query)
|
|
|
|
const txsIds = txs.map(t => t.txnID)
|
|
|
|
if (txsIds.length == 0)
|
|
return []
|
|
|
|
// Prepares subqueries for
|
|
// the query retrieving utxos of interest
|
|
let subQuery2 = ''
|
|
let subQueries2 = []
|
|
|
|
if (addresses && addresses.length > 0) {
|
|
const params2 = [txsIds, addresses, txsIds, addresses]
|
|
subQuery2 = mysql.format(SUBQUERY_UTXOS_ADDR, params2)
|
|
subQueries2.push(subQuery2)
|
|
}
|
|
|
|
if (xpubs && xpubs.length > 0) {
|
|
const params2 = [txsIds, xpubs, txsIds, xpubs]
|
|
subQuery2 = mysql.format(SUBQUERY_UTXOS_XPUB, params2)
|
|
subQueries2.push(subQuery2)
|
|
}
|
|
|
|
subQuery2 = subQueries2.join(' UNION DISTINCT ')
|
|
|
|
// Get inputs and outputs of interest
|
|
const sqlQuery2 = 'SELECT * \
|
|
FROM (' + subQuery2 + ') AS `utxos` \
|
|
ORDER BY `utxos`.`outIndex` ASC, `utxos`.`inIndex` ASC'
|
|
|
|
let query2 = mysql.format(sqlQuery2)
|
|
const utxos = await this._query(query2)
|
|
|
|
return this.assembleTransactions(txs, utxos)
|
|
}
|
|
|
|
/**
|
|
* Initialize a transaction object returned as response to queries
|
|
* @param {object} tx - transaction data retrieved from db
|
|
* @returns {object} returns the transaction stub
|
|
*/
|
|
_transactionStub(tx) {
|
|
let ret = {
|
|
hash: tx.txnTxid,
|
|
time: (tx.time < 32503680000) ? tx.time : 0,
|
|
version: tx.txnVersion,
|
|
locktime: tx.txnLocktime,
|
|
result: 0,
|
|
inputs: [],
|
|
out: []
|
|
}
|
|
|
|
if (tx.blockHeight != null)
|
|
ret.block_height = tx.blockHeight
|
|
|
|
return ret
|
|
}
|
|
|
|
/**
|
|
* Initialize an input object returned as part of a response
|
|
* @param {object} input - input data retrieved from db
|
|
* @returns {object} returns the input stub
|
|
*/
|
|
_inputStub(input) {
|
|
let ret = {
|
|
vin: input.inIndex,
|
|
sequence: input.inSequence,
|
|
prev_out: {
|
|
txid: input.inOutTxid,
|
|
vout: input.inOutIndex,
|
|
value: input.inOutAmount,
|
|
addr: input.inOutAddress
|
|
}
|
|
}
|
|
|
|
if (input.hdXpub && input.hdXpub !== null) {
|
|
ret.prev_out.xpub = {
|
|
m: input.hdXpub,
|
|
path: ['M', input.hdAddrChain, input.hdAddrIndex].join('/')
|
|
}
|
|
}
|
|
|
|
return ret
|
|
}
|
|
|
|
/**
|
|
* Initialize an output object returned as part of a response
|
|
* @param {object} output - output data retrieved from db
|
|
* @returns {object} returns the output stub
|
|
*/
|
|
_outputStub(output) {
|
|
let ret = {
|
|
n: output.outIndex,
|
|
value: output.outAmount,
|
|
addr: output.outAddress
|
|
}
|
|
|
|
if (output.hdXpub && output.hdXpub !== null) {
|
|
ret.xpub = {
|
|
m: output.hdXpub,
|
|
path: ['M', output.hdAddrChain, output.hdAddrIndex].join('/')
|
|
}
|
|
}
|
|
|
|
return ret
|
|
}
|
|
|
|
/**
|
|
* Take query results for txs and utxos and combine into transaction data
|
|
* @param {object[]} txs - array of transaction data retrieved from db
|
|
* @param {object[]} utxos - array of utxos data retrieved from db
|
|
* @returns {object[]} returns an array of transaction objects
|
|
*/
|
|
assembleTransactions(txs, utxos) {
|
|
const txns = {}
|
|
|
|
for (let tx of txs) {
|
|
if (!txns[tx.txnID])
|
|
txns[tx.txnID] = this._transactionStub(tx)
|
|
}
|
|
|
|
for (let u of utxos) {
|
|
if (u.txnID != null && txns[u.txnID]) {
|
|
if (u.inIndex != null) {
|
|
txns[u.txnID].result -= u.inOutAmount
|
|
txns[u.txnID].inputs.push(this._inputStub(u))
|
|
} else if (u.outIndex != null) {
|
|
txns[u.txnID].result += u.outAmount
|
|
txns[u.txnID].out.push(this._outputStub(u))
|
|
}
|
|
}
|
|
}
|
|
|
|
// Return transactions in descending time order, most recent first
|
|
const ret = Object.keys(txns).map(key => txns[key])
|
|
ret.sort((a,b) => b.time - a.time)
|
|
return ret
|
|
}
|
|
|
|
/**
|
|
* Get a list of unspent outputs for given hd account
|
|
* @param {string} xpub - xpub
|
|
* @returns {object[]} returns an array of utxos objects
|
|
* {txnTxid, txnVersion, txnLocktime, outIndex, outAmount, outScript, addrAddress}
|
|
*/
|
|
async getHDAccountUnspentOutputs(xpub) {
|
|
const sqlQuery = 'SELECT \
|
|
`txnTxid`, \
|
|
`txnVersion`, \
|
|
`txnLocktime`, \
|
|
`blockHeight`, \
|
|
`outIndex`, \
|
|
`outAmount`, \
|
|
`outScript`, \
|
|
`addrAddress`, \
|
|
`hdAddrChain`, \
|
|
`hdAddrIndex` \
|
|
FROM `outputs` \
|
|
INNER JOIN `addresses` ON `outputs`.`addrID` = `addresses`.`addrID` \
|
|
INNER JOIN `hd_addresses` ON `outputs`.`addrID` = `hd_addresses`.`addrID` \
|
|
INNER JOIN `hd` ON `hd_addresses`.`hdID` = `hd`.`hdID` \
|
|
INNER JOIN `transactions` ON `outputs`.`txnID` = `transactions`.`txnID` \
|
|
LEFT JOIN `blocks` ON `transactions`.`blockID` = `blocks`.`blockID` \
|
|
LEFT JOIN `inputs` ON `outputs`.`outID` = `inputs`.`outID` \
|
|
WHERE `inputs`.`outID` IS NULL \
|
|
AND `hd`.`hdXpub` = ?'
|
|
|
|
const params = xpub
|
|
const query = mysql.format(sqlQuery, params)
|
|
return this._query(query)
|
|
}
|
|
|
|
/**
|
|
* Get addresses that belong to a given hd account
|
|
* @param {string[]} addresses - array of bitcoin addresses
|
|
* @returns {object} returns a dictionary {[address]: hdXpub, ...}
|
|
*/
|
|
async getXpubByAddresses(addresses) {
|
|
const ret = {}
|
|
|
|
if (addresses.length == 0)
|
|
return ret
|
|
|
|
const sqlQuery = 'SELECT `hd`.`hdXpub`, `addresses`.`addrAddress` \
|
|
FROM `addresses` \
|
|
INNER JOIN `hd_addresses` ON `hd_addresses`.`addrID` = `addresses`.`addrID` \
|
|
INNER JOIN `hd` ON `hd_addresses`.`hdID` = `hd`.`hdID` \
|
|
WHERE `addresses`.`addrAddress` IN (?)'
|
|
|
|
const params = [addresses]
|
|
const query = mysql.format(sqlQuery, params)
|
|
const results = await this._query(query)
|
|
|
|
for (let r of results)
|
|
ret[r.addrAddress] = r.hdXpub
|
|
|
|
return ret
|
|
}
|
|
|
|
/**
|
|
* Get the mysql ID of a transaction. Ensures that the transaction exists.
|
|
* @param {string} txid - txid of a transaction
|
|
* @returns {integer} returns the transaction id (mysql id)
|
|
*/
|
|
async ensureTransactionId(txid) {
|
|
const sqlQuery = 'INSERT IGNORE INTO `transactions` SET ?'
|
|
const params = {
|
|
txnTxid: txid,
|
|
txnCreated: util.unix()
|
|
}
|
|
const query = mysql.format(sqlQuery, params)
|
|
const result = await this._query(query)
|
|
|
|
// Successful insertion
|
|
if (result.insertId > 0)
|
|
return result.insertId
|
|
|
|
// Transaction already in db
|
|
const sqlQuery2 = 'SELECT `txnID` FROM `transactions` WHERE `txnTxid` = ?'
|
|
const params2 = txid
|
|
const query2 = mysql.format(sqlQuery2, params2)
|
|
const result2 = await this._query(query2)
|
|
|
|
if (result2.length > 0)
|
|
return result2[0].txnID
|
|
|
|
throw 'Problem met while trying to insert a new transaction'
|
|
}
|
|
|
|
/**
|
|
* Get the mysql ID of a transaction
|
|
* @param {string} txid - txid of a transaction
|
|
* @returns {integer} returns the transaction id (mysql id)
|
|
*/
|
|
async getTransactionId(txid) {
|
|
const sqlQuery = 'SELECT `txnID` FROM `transactions` WHERE `txnTxid` = ?'
|
|
const params = txid
|
|
const query = mysql.format(sqlQuery, params)
|
|
const result = await this._query(query)
|
|
return (result.length == 0) ? null : result[0].txnID
|
|
}
|
|
|
|
/**
|
|
* Get the mysql IDs of a collection of transactions
|
|
* @param {string[]} txids - txids of the transactions
|
|
* @returns {object[]} returns an array of {txnTxid: txnId}
|
|
*/
|
|
async getTransactionsIds(txids) {
|
|
if (txids.length == 0)
|
|
return []
|
|
|
|
const sqlQuery = 'SELECT `txnID`, `txnTxid` FROM `transactions` WHERE `txnTxid` IN (?)'
|
|
const params = [txids]
|
|
const query = mysql.format(sqlQuery, params)
|
|
const results = await this._query(query)
|
|
|
|
const ret = {}
|
|
for (let r of results)
|
|
ret[r.txnTxid] = r.txnID
|
|
return ret
|
|
}
|
|
|
|
/**
|
|
* Get the mysql IDs of a set of transactions
|
|
* @param {string[]} txid - array of transactions txids
|
|
* @returns {integer[]} returns an array of transaction ids (mysql ids)
|
|
*/
|
|
async getTransactionsById(txnIDs) {
|
|
if (txnIDs.length == 0)
|
|
return []
|
|
|
|
const sqlQuery = 'SELECT * FROM `transactions` WHERE `txnID` IN (?)'
|
|
const params = [txnIDs]
|
|
const query = mysql.format(sqlQuery, params)
|
|
return this._query(query)
|
|
}
|
|
|
|
/**
|
|
* Insert a new transaction in db
|
|
* @param {object} tx - {txid, version, locktime}
|
|
*/
|
|
async addTransaction(tx) {
|
|
if (!tx.created)
|
|
tx.created = util.unix()
|
|
|
|
const sqlQuery = 'INSERT INTO `transactions` \
|
|
(txnTxid, txnCreated, txnVersion, txnLocktime) VALUES (?) \
|
|
ON DUPLICATE KEY UPDATE txnVersion = VALUES(txnVersion)'
|
|
|
|
const params = [[
|
|
tx.txid,
|
|
tx.created,
|
|
tx.version,
|
|
tx.locktime
|
|
]]
|
|
|
|
const query = mysql.format(sqlQuery, params)
|
|
return this._query(query)
|
|
}
|
|
|
|
/**
|
|
* Insert a collection of transactions in db
|
|
* @param {object[]} txs - array of {txid, version, locktime}
|
|
*/
|
|
async addTransactions(txs) {
|
|
if (txs.length == 0)
|
|
return
|
|
|
|
const sqlQuery = 'INSERT INTO `transactions` \
|
|
(txnTxid, txnCreated, txnVersion, txnLocktime) VALUES ? \
|
|
ON DUPLICATE KEY UPDATE txnVersion = VALUES(txnVersion)'
|
|
|
|
const params = [txs.map(tx => [
|
|
tx.txid,
|
|
tx.created ? tx.created : util.unix(),
|
|
tx.version,
|
|
tx.locktime
|
|
])]
|
|
|
|
const query = mysql.format(sqlQuery, params)
|
|
return this._query(query)
|
|
}
|
|
|
|
/**
|
|
* Get a transaction for a given txid
|
|
* @param {string} txid - txid of the transaction
|
|
*/
|
|
async getTransaction(txid) {
|
|
// Get transaction outputs
|
|
const outputsQuery = mysql.format(SUBQUERY_GET_TX_OUTS, txid)
|
|
// Get transaction inputs
|
|
const inputsQuery = mysql.format(SUBQUERY_GET_TX_INS, txid)
|
|
|
|
const results = await Promise.all([
|
|
this._query(outputsQuery),
|
|
this._query(inputsQuery)
|
|
])
|
|
|
|
const tx = {
|
|
hash: txid,
|
|
time: Infinity,
|
|
version: 0,
|
|
locktime: 0,
|
|
inputs: [],
|
|
out: [],
|
|
block_height: null,
|
|
}
|
|
|
|
// Process the outputs
|
|
for (let output of results[0]) {
|
|
tx.version = output.txnVersion
|
|
tx.locktime = output.txnLocktime
|
|
|
|
if (output.blockTime != null)
|
|
tx.time = Math.min(tx.time, output.blockTime)
|
|
|
|
tx.time = Math.min(tx.time, output.txnCreated)
|
|
|
|
if (output.blockHeight != null)
|
|
tx.block_height = output.blockHeight
|
|
|
|
const fmt = {
|
|
n: output.outIndex,
|
|
value: output.outAmount,
|
|
addr: output.addrAddress,
|
|
script: output.outScript,
|
|
}
|
|
|
|
if (output.hdXpub) {
|
|
fmt.xpub = {
|
|
m: output.hdXpub,
|
|
path: ['M', output.hdAddrChain, output.hdAddrIndex].join('/')
|
|
}
|
|
}
|
|
|
|
tx.out.push(fmt)
|
|
}
|
|
|
|
// Process the inputs
|
|
for (let input of results[1]) {
|
|
tx.version = input.txnVersion
|
|
tx.locktime = input.txnLocktime
|
|
|
|
if (input.blockTime != null)
|
|
tx.time = Math.min(tx.time, input.blockTime)
|
|
|
|
tx.time = Math.min(tx.time, input.txnCreated)
|
|
|
|
if (input.blockHeight != null)
|
|
tx.block_height = input.blockHeight
|
|
|
|
const fmt = {
|
|
vin: input.inIndex,
|
|
prev_out: {
|
|
txid: input.prevOutTxid,
|
|
vout: input.outIndex,
|
|
value: input.outAmount,
|
|
addr: input.addrAddress,
|
|
script: input.outScript,
|
|
},
|
|
sequence: input.inSequence
|
|
}
|
|
|
|
if (input.hdXpub) {
|
|
fmt.prev_out.xpub = {
|
|
m: input.hdXpub,
|
|
path: ['M', input.hdAddrChain, input.hdAddrIndex].join('/')
|
|
}
|
|
}
|
|
|
|
tx.inputs.push(fmt)
|
|
}
|
|
|
|
// Remove block height if null
|
|
if (tx.block_height == null)
|
|
delete tx.block_height
|
|
|
|
return tx
|
|
}
|
|
|
|
/**
|
|
* Get the unconfirmed transactions
|
|
* @returns {object[]} returns an array of transactions data
|
|
*/
|
|
async getUnconfirmedTransactions() {
|
|
const query = 'SELECT * FROM `transactions` WHERE blockID IS NULL'
|
|
return this._query(query)
|
|
}
|
|
|
|
/**
|
|
* Get all transactions
|
|
* @returns {object[]} returns an array of transactions data
|
|
*/
|
|
async getTransactions() {
|
|
const query = 'SELECT * FROM `transactions`'
|
|
return this._query(query)
|
|
}
|
|
|
|
/**
|
|
* Get the inputs of a transaction
|
|
* @param {string} txnID - mysql id of the transaction
|
|
* @returns {object[]} returns an array of inputs
|
|
*/
|
|
async getTxInputs(txnID) {
|
|
const sqlQuery = 'SELECT * FROM `inputs` WHERE `txnID` = ?'
|
|
const params = txnID
|
|
const query = mysql.format(sqlQuery, params)
|
|
return this._query(query)
|
|
}
|
|
|
|
/**
|
|
* Batch confirm txids in a block
|
|
* @param {string[]} txnTxidArray - array of transaction txids
|
|
* @param {integer} blockID - mysql id of the blck
|
|
*/
|
|
async confirmTransactions(txnTxidArray, blockID) {
|
|
if (txnTxidArray.length == 0)
|
|
return
|
|
|
|
const sqlQuery = 'UPDATE `transactions` SET `blockID` = ? WHERE `txnTxid` IN (?)'
|
|
const params = [blockID, txnTxidArray]
|
|
const query = mysql.format(sqlQuery, params)
|
|
return this._query(query)
|
|
}
|
|
|
|
/**
|
|
* Get the transactions confirmed after a given height
|
|
* @param {integer]} height - block height
|
|
* @param {object[]} returns an array of transactions
|
|
*/
|
|
async getTransactionsConfirmedAfterHeight(height) {
|
|
const sqlQuery = 'SELECT `transactions`.* FROM `transactions` \
|
|
INNER JOIN `blocks` ON `blocks`.`blockID` = `transactions`.`blockID` \
|
|
WHERE `blocks`.`blockHeight` > ?'
|
|
const params = height
|
|
const query = mysql.format(sqlQuery, params)
|
|
return this._query(query)
|
|
}
|
|
|
|
/**
|
|
* Delete the transactions confirmed after a given height
|
|
* @param {integer]} height - block height
|
|
*/
|
|
async deleteTransactionsConfirmedAfterHeight(height) {
|
|
const sqlQuery = 'DELETE `transactions`.* FROM `transactions` \
|
|
INNER JOIN `blocks` ON `blocks`.`blockID` = `transactions`.`blockID` \
|
|
WHERE `blocks`.`blockHeight` > ?'
|
|
const params = height
|
|
const query = mysql.format(sqlQuery, params)
|
|
return this._query(query)
|
|
}
|
|
|
|
/**
|
|
* Batch unconfirm a set of transactions
|
|
* @param {string[]} txnTxidArray - array of transaction txids
|
|
*/
|
|
async unconfirmTransactions(txnTxidArray) {
|
|
if (txnTxidArray.length == 0)
|
|
return
|
|
|
|
const sqlQuery = 'UPDATE `transactions` SET `blockID` = NULL WHERE `txnTxid` IN (?)'
|
|
const params = [txnTxidArray]
|
|
const query = mysql.format(sqlQuery, params)
|
|
return this._query(query)
|
|
}
|
|
|
|
/**
|
|
* Delete a transaction identified by its txid
|
|
* @param {string} txid - txid of the transaction
|
|
*/
|
|
async deleteTransaction(txid) {
|
|
const sqlQuery = 'DELETE `transactions`.* FROM `transactions` WHERE `transactions`.`txnTxid` = ?'
|
|
const params = txid
|
|
const query = mysql.format(sqlQuery, params)
|
|
return this._query(query)
|
|
}
|
|
|
|
/**
|
|
* Delete a set of transactions identified by their mysql ids
|
|
* @param {integer[]} txnIDs - mysql ids of the transactions
|
|
*/
|
|
async deleteTransactionsByID(txnIDs) {
|
|
if (txnIDs.length == 0)
|
|
return []
|
|
|
|
const sqlQuery = 'DELETE `transactions`.* FROM `transactions` WHERE `transactions`.`txnID` in (?)'
|
|
const params = [txnIDs]
|
|
const query = mysql.format(sqlQuery, params)
|
|
return this._query(query)
|
|
}
|
|
|
|
/**
|
|
* Insert transaction outputs associated with known addresses
|
|
* @param {object[]} outputs - array of {txnID, addrID, outIndex, outAmount, outScript}
|
|
*/
|
|
async addOutputs(outputs) {
|
|
if (outputs.length == 0)
|
|
return
|
|
|
|
const sqlQuery = 'INSERT IGNORE INTO `outputs` \
|
|
(txnID, addrID, outIndex, outAmount, outScript) VALUES ?'
|
|
|
|
const params = [outputs.map(o => [o.txnID, o.addrID, o.outIndex, o.outAmount, o.outScript])]
|
|
const query = mysql.format(sqlQuery, params)
|
|
return this._query(query)
|
|
}
|
|
|
|
/**
|
|
* Get a list of outputs identified by their txid and index.
|
|
* The presence of spendingTxnID and spendingInID not null indicate that an
|
|
* input spending the transaction output index is already in the database and
|
|
* may indicate a DOUBLE SPEND.
|
|
* @param {object[]} spends - array of {txid,index}
|
|
* @returns {object[]} returns a array of output objects
|
|
* {addrAddress, outID, outAmount, txnTxid, outIndex, spendingTxnID/null, spendingInID/null}
|
|
*/
|
|
async getOutputSpends(spends) {
|
|
if (spends.length == 0)
|
|
return []
|
|
|
|
const whereClauses =
|
|
spends.map(s => '(`txnTxid`=' + this.pool.escape(s.txid) + ' AND `outIndex`=' + this.pool.escape(s.index) + ')')
|
|
|
|
const whereClause = whereClauses.join(' OR ')
|
|
|
|
const sqlQuery = 'SELECT \
|
|
`addrAddress`, \
|
|
`outputs`.`outID`, \
|
|
`outAmount`, \
|
|
`txnTxid`, \
|
|
`outIndex`, \
|
|
`inputs`.`txnID` AS `spendingTxnID` \
|
|
FROM `outputs` \
|
|
INNER JOIN `addresses` ON `outputs`.`addrID` = `addresses`.`addrID` \
|
|
INNER JOIN `transactions` ON `outputs`.`txnID` = `transactions`.`txnID` \
|
|
LEFT JOIN `inputs` ON `inputs`.`outID` = `outputs`.outID \
|
|
WHERE ' + whereClause
|
|
|
|
const query = mysql.format(sqlQuery)
|
|
return this._query(query)
|
|
}
|
|
|
|
/**
|
|
* Get a list of mysql ids for outputs identified by their txid and index.
|
|
* @param {object[]} spends - array of {txid,vout}
|
|
* @returns {object[]} returns a array of output objects
|
|
* {outID, txnTxid, outIndex}
|
|
*/
|
|
async getOutputIds(spends) {
|
|
if (spends.length == 0)
|
|
return []
|
|
|
|
const whereClauses =
|
|
spends.map((s) => '(`txnTxid`=' + this.pool.escape(s.txid) + ' AND `outIndex`=' + this.pool.escape(s.vout) + ')')
|
|
|
|
const whereClause = whereClauses.join(' OR ')
|
|
|
|
const sqlQuery = 'SELECT \
|
|
`outID`, \
|
|
`txnTxid`, \
|
|
`outIndex` \
|
|
FROM `outputs` \
|
|
INNER JOIN `transactions` ON `outputs`.`txnID` = `transactions`.`txnID` \
|
|
WHERE ' + whereClause
|
|
|
|
const query = mysql.format(sqlQuery)
|
|
return this._query(query)
|
|
}
|
|
|
|
/**
|
|
* Get a list of unspent outputs for a list of addresses
|
|
* @param {string[]} addresses - array of bitcoin addresses
|
|
* @returns {object[]} returns a array of output objects
|
|
* {txnTxid, outIndex, outAmount, outScript, addrAddress}
|
|
*/
|
|
async getUnspentOutputs(addresses) {
|
|
if (addresses.length == 0)
|
|
return []
|
|
|
|
const sqlQuery = 'SELECT \
|
|
`txnTxid`, \
|
|
`txnVersion`, \
|
|
`txnLocktime`, \
|
|
`blockHeight`, \
|
|
`outIndex`, \
|
|
`outAmount`, \
|
|
`outScript`, \
|
|
`addrAddress` \
|
|
FROM `outputs` \
|
|
INNER JOIN `addresses` ON `outputs`.`addrID` = `addresses`.`addrID` \
|
|
INNER JOIN `transactions` ON `outputs`.`txnID` = `transactions`.`txnID` \
|
|
LEFT JOIN `blocks` ON `transactions`.`blockID` = `blocks`.`blockID` \
|
|
LEFT JOIN `inputs` ON `outputs`.`outID` = `inputs`.`outID` \
|
|
WHERE `inputs`.`outID` IS NULL \
|
|
AND (`addrAddress`) IN (?)'
|
|
|
|
const params = [addresses]
|
|
const query = mysql.format(sqlQuery, params)
|
|
return this._query(query)
|
|
}
|
|
|
|
/**
|
|
* Insert transaction inputs that spend known Outputs
|
|
* @param {object[]} inputs - array of input objects
|
|
* {txnID, outID, inIndex, inSequence}
|
|
*/
|
|
async addInputs(inputs) {
|
|
if (inputs.length == 0)
|
|
return
|
|
|
|
const sqlQuery = 'INSERT INTO `inputs` \
|
|
(txnID, outID, inIndex, inSequence) VALUES ? \
|
|
ON DUPLICATE KEY UPDATE outID = VALUES(outID)'
|
|
|
|
const params = [inputs.map((i) => [i.txnID, i.outID, i.inIndex, i.inSequence])]
|
|
const query = mysql.format(sqlQuery, params)
|
|
return this._query(query)
|
|
}
|
|
|
|
/**
|
|
* Insert a new block
|
|
* @param {object} block - block object
|
|
* {blockHash, blockParent, blockHeight, blockTime}
|
|
* block.blockParent is an ID, which may be obtained with DB.getBlockByHash
|
|
*/
|
|
async addBlock(block) {
|
|
const sqlQuery = 'INSERT IGNORE INTO `blocks` SET ?'
|
|
const params = block
|
|
const query = mysql.format(sqlQuery, params)
|
|
const result = await this._query(query)
|
|
|
|
// Successful insertion
|
|
if (result.insertId > 0)
|
|
return result.insertId
|
|
|
|
// Block already in db
|
|
const sqlQuery2 = 'SELECT `blockID` FROM `blocks` WHERE `blockHash` = ?'
|
|
const params2 = block.blockHash
|
|
const query2 = mysql.format(sqlQuery2, params2)
|
|
const result2 = await this._query(query2)
|
|
|
|
if (result2.length > 0)
|
|
return result2[0].blockID
|
|
|
|
throw 'Problem met while trying to insert a new block'
|
|
}
|
|
|
|
/**
|
|
* Get a block identified by the block hash
|
|
* @param {string} hash - block hash
|
|
* @returns {object} returns the block
|
|
*/
|
|
async getBlockByHash(hash) {
|
|
const sqlQuery = 'SELECT * FROM `blocks` WHERE `blockHash` = ?'
|
|
const params = hash
|
|
const query = mysql.format(sqlQuery, params)
|
|
const result = await this._query(query)
|
|
return (result.length == 1) ? result[0] : null
|
|
}
|
|
|
|
/**
|
|
* Get a collection of blocks identified by the blocks hashes
|
|
* @param {string[]} hashes - blocks hashes
|
|
* @returns {object[]} returns the blocks
|
|
*/
|
|
async getBlocksByHashes(hashes) {
|
|
if (hashes.length == 0)
|
|
return []
|
|
|
|
const sqlQuery = 'SELECT * FROM `blocks` WHERE `blockHash` IN (?)'
|
|
const params = [hashes]
|
|
const query = mysql.format(sqlQuery, params)
|
|
return await this._query(query)
|
|
}
|
|
|
|
/**
|
|
* Get details about all blocks at a given block height
|
|
* @param {integer} height - block height
|
|
* @returns {object[]} returns an array of blocks
|
|
*/
|
|
async getBlocksByHeight(height) {
|
|
const sqlQuery = 'SELECT * FROM `blocks` WHERE `blockHeight` = ?'
|
|
const params = height
|
|
const query = mysql.format(sqlQuery, params)
|
|
return this._query(query)
|
|
}
|
|
|
|
/**
|
|
* Delete the blocks after a given block height
|
|
* @param {integer} height - block height
|
|
*/
|
|
async deleteBlocksAfterHeight(height) {
|
|
const sqlQuery = 'DELETE FROM `blocks` WHERE `blockHeight` > ?'
|
|
const params = height
|
|
const query = mysql.format(sqlQuery, params)
|
|
return this._query(query)
|
|
}
|
|
|
|
/**
|
|
* Gets the last block
|
|
* @returns {object} returns the last block
|
|
*/
|
|
async getHighestBlock() {
|
|
try {
|
|
const results = await this.getLastBlocks(1)
|
|
if (results == null || results.length == 0)
|
|
return {
|
|
blockID: null,
|
|
blockHeight: 0
|
|
}
|
|
else
|
|
return results[0]
|
|
} catch(err) {
|
|
return null
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Gets the N last blocks
|
|
* @param {integer} n - number of blocks to be retrieved
|
|
* @returns {object[]} returns an array of the n last blocks
|
|
*/
|
|
async getLastBlocks(n) {
|
|
const sqlQuery = 'SELECT * FROM `blocks` ORDER BY `blockHeight` DESC LIMIT ?'
|
|
const params = n
|
|
let query = mysql.format(sqlQuery, n)
|
|
return this._query(query)
|
|
}
|
|
|
|
|
|
/**
|
|
* Get all scheduled transactions
|
|
* @returns {object[]} returns an array of scheduled transactions
|
|
*/
|
|
async getScheduledTransactions() {
|
|
const sqlQuery = 'SELECT * FROM `scheduled_transactions` ORDER BY `schTrigger`, `schCreated`'
|
|
return this._query(sqlQuery)
|
|
}
|
|
|
|
/**
|
|
* Get the mysql ID of a scheduled transaction
|
|
* @param {string} txid - txid of a scheduled transaction
|
|
* @returns {integer} returns the scheduled transaction id (mysql id)
|
|
*/
|
|
async getScheduledTransactionId(txid) {
|
|
const sqlQuery = 'SELECT `schID` FROM `scheduled_transactions` WHERE `schTxid` = ?'
|
|
const params = txid
|
|
const query = mysql.format(sqlQuery, params)
|
|
const result = await this._query(query)
|
|
return (result.length == 0) ? null : result[0].txnID
|
|
}
|
|
|
|
/**
|
|
* Insert a new scheduled transaction in db
|
|
* @param {object} tx - {txid, created, rawTx, parentId, delay, trigger}
|
|
*/
|
|
async addScheduledTransaction(tx) {
|
|
if (!tx.created)
|
|
tx.created = util.unix()
|
|
|
|
const sqlQuery = 'INSERT INTO `scheduled_transactions` \
|
|
(schTxid, schCreated, schRaw, schParentID, schParentTxid, schDelay, schTrigger) VALUES (?)'
|
|
|
|
const params = [[
|
|
tx.txid,
|
|
tx.created,
|
|
tx.rawTx,
|
|
tx.parentId,
|
|
tx.parentTxid,
|
|
tx.delay,
|
|
tx.trigger
|
|
]]
|
|
|
|
const query = mysql.format(sqlQuery, params)
|
|
const result = await this._query(query)
|
|
|
|
if (result.insertId > 0)
|
|
return result.insertId
|
|
|
|
throw 'Problem met while trying to insert a new scheduled transaction'
|
|
}
|
|
|
|
/**
|
|
* Delete a scheduled transaction
|
|
* @param {string} txid - scheduled transaction txid
|
|
*/
|
|
async deleteScheduledTransaction(txid) {
|
|
const sqlQuery = 'DELETE `scheduled_transactions`.* \
|
|
FROM `scheduled_transactions` \
|
|
WHERE `scheduled_transactions`.`schTxid` = ?'
|
|
const params = txid
|
|
const query = mysql.format(sqlQuery, params)
|
|
return this._query(query)
|
|
}
|
|
|
|
/**
|
|
* Get scheduled transactions
|
|
* with a trigger lower than a given block height
|
|
* @param {integer} height - block height
|
|
*/
|
|
async getActivatedScheduledTransactions(height) {
|
|
const sqlQuery = 'SELECT * FROM `scheduled_transactions` \
|
|
WHERE `schTrigger` <= ? AND `schParentID` IS NULL'
|
|
const params = height
|
|
const query = mysql.format(sqlQuery, params)
|
|
return this._query(query)
|
|
}
|
|
|
|
/**
|
|
* Get the scheduled transaction having a given parentID
|
|
* @param {integer} parentId - parent ID
|
|
* @returns {object[]} returns an array of scheduled transactions
|
|
*/
|
|
async getNextScheduledTransactions(parentId) {
|
|
const sqlQuery = 'SELECT * FROM `scheduled_transactions` \
|
|
WHERE `schParentID` = ?'
|
|
const params = parentId
|
|
const query = mysql.format(sqlQuery, params)
|
|
return this._query(query)
|
|
}
|
|
|
|
/**
|
|
* Update the trigger of a scheduled transaction
|
|
* identified by its ID
|
|
* @param {integer} id - id of the scheduled transaction
|
|
* @param {integer} trigger - new trigger
|
|
*/
|
|
async updateTriggerScheduledTransaction(id, trigger) {
|
|
const sqlQuery = 'UPDATE `scheduled_transactions` \
|
|
SET `schTrigger` = ? \
|
|
WHERE `schID` = ?'
|
|
const params = [trigger, id]
|
|
const query = mysql.format(sqlQuery, params)
|
|
return this._query(query)
|
|
}
|
|
|
|
|
|
/**
|
|
* MAINTENANCE FUNCTIONS
|
|
*/
|
|
|
|
async getInvalidAccountTimes() {
|
|
const sqlQuery = 'SELECT \
|
|
`hd`.`hdID`, \
|
|
`hdCreated`, \
|
|
min(`txnCreated`) as `earliest` \
|
|
FROM `hd` \
|
|
INNER JOIN `hd_addresses` ON `hd_addresses`.`hdID` = `hd`.`hdID` \
|
|
INNER JOIN `addresses` ON `hd_addresses`.`addrID` = `addresses`.`addrID` \
|
|
INNER JOIN `outputs` ON `outputs`.`addrID` = `hd_addresses`.`addrID` \
|
|
INNER JOIN `transactions` ON `outputs`.`txnID` = `transactions`.`txnID` \
|
|
WHERE `hd`.`hdCreated` > `transactions`.`txnCreated` \
|
|
GROUP BY `hd`.`hdID` LIMIT 100'
|
|
|
|
return this._query(sqlQuery)
|
|
}
|
|
|
|
async getInvalidTxTimes() {
|
|
const sqlQuery = 'SELECT \
|
|
`txnID`, \
|
|
`txnCreated`, \
|
|
`blockTime` \
|
|
FROM `transactions` \
|
|
INNER JOIN `blocks` ON `transactions`.`blockID` = `blocks`.`blockID` \
|
|
WHERE `transactions`.`txnCreated` > `blocks`.`blockTime` \
|
|
LIMIT 100'
|
|
|
|
return this._query(sqlQuery)
|
|
}
|
|
|
|
async setHDTime(hdID, hdCreated) {
|
|
const sqlQuery = 'UPDATE `hd` SET `hdCreated` = ? WHERE `hdID` = ?'
|
|
const params = [hdCreated, hdID]
|
|
const query = mysql.format(sqlQuery, params)
|
|
return this._query(query)
|
|
}
|
|
|
|
async setTransactionTime(txnID, txnCreated) {
|
|
const sqlQuery = 'UPDATE `transactions` SET `txnCreated` = ? WHERE `txnID` = ?'
|
|
const params = [txnCreated, txnID]
|
|
const query = mysql.format(sqlQuery, params)
|
|
return this._query(query)
|
|
}
|
|
|
|
async updateInputSequence(inID, inSequence) {
|
|
const sqlQuery = 'UPDATE `inputs` SET `inSequence` = ? WHERE `inID` = ?'
|
|
const params = [inSequence, inID]
|
|
const query = mysql.format(sqlQuery, params)
|
|
return this._query(query)
|
|
}
|
|
|
|
async getOutputsWithoutScript() {
|
|
const sqlQuery = 'SELECT \
|
|
`txnTxid`, \
|
|
`outIndex`, \
|
|
`outId` \
|
|
FROM `outputs` \
|
|
INNER JOIN `transactions` ON `outputs`.`txnID` = `transactions`.`txnID` \
|
|
WHERE length(`outputs`.`outScript`) = 0 \
|
|
ORDER BY `outId` \
|
|
LIMIT 100'
|
|
|
|
return this._query(sqlQuery)
|
|
}
|
|
|
|
async updateOutputScript(outID, outScript) {
|
|
const sqlQuery = 'UPDATE `outputs` SET `outScript` = ? WHERE `outID` = ?'
|
|
const params = [outScript, outID]
|
|
const query = mysql.format(sqlQuery, params)
|
|
return this._query(query)
|
|
}
|
|
|
|
async setBlockParent(hash, blockID) {
|
|
const sqlQuery = 'UPDATE `blocks` SET `blockParent` = ? WHERE `blockHash` = ?'
|
|
const params = [blockID, hash]
|
|
const query = mysql.format(sqlQuery, params)
|
|
return this._query(query)
|
|
}
|
|
|
|
}
|
|
|
|
module.exports = new MySqlDbWrapper()
|
|
|