|
|
|
/*!
|
|
|
|
* pushtx/orchestrator.js
|
|
|
|
* Copyright © 2019 – Katana Cryptographic Ltd. All Rights Reserved.
|
|
|
|
*/
|
|
|
|
'use strict'
|
|
|
|
|
|
|
|
const zmq = require('zeromq')
|
|
|
|
const Sema = require('async-sema')
|
|
|
|
const Logger = require('../lib/logger')
|
|
|
|
const db = require('../lib/db/mysql-db-wrapper')
|
|
|
|
const RpcClient = require('../lib/bitcoind-rpc/rpc-client')
|
|
|
|
const network = require('../lib/bitcoin/network')
|
|
|
|
const keys = require('../keys')[network.key]
|
|
|
|
const pushTxProcessor = require('./pushtx-processor')
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* A class orchestrating the push of scheduled transactions
|
|
|
|
*/
|
|
|
|
class Orchestrator {
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Constructor
|
|
|
|
*/
|
|
|
|
constructor() {
|
|
|
|
// RPC client
|
|
|
|
this.rpcClient = new RpcClient()
|
|
|
|
// ZeroMQ socket for bitcoind blocks messages
|
|
|
|
this.blkSock = null
|
|
|
|
// Initialize a semaphor protecting the onBlockHash() method
|
|
|
|
this._onBlockHashSemaphor = new Sema(1, { capacity: 50 })
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Start processing the blockchain
|
|
|
|
* @returns {Promise}
|
|
|
|
*/
|
|
|
|
start() {
|
|
|
|
this.initSockets()
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Start processing the blockchain
|
|
|
|
*/
|
|
|
|
async stop() {}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Initialiaze ZMQ sockets
|
|
|
|
*/
|
|
|
|
initSockets() {
|
|
|
|
// Socket listening to bitcoind Blocks messages
|
|
|
|
this.blkSock = zmq.socket('sub')
|
|
|
|
this.blkSock.connect(keys.bitcoind.zmqBlk)
|
|
|
|
this.blkSock.subscribe('hashblock')
|
|
|
|
|
|
|
|
this.blkSock.on('message', (topic, message) => {
|
|
|
|
switch (topic.toString()) {
|
|
|
|
case 'hashblock':
|
|
|
|
this.onBlockHash(message)
|
|
|
|
break
|
|
|
|
default:
|
|
|
|
Logger.info(topic.toString())
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
|
|
|
Logger.info('Listening for blocks')
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Push Transactions if triggered by new block
|
|
|
|
* @param {Buffer} buf - block hash
|
|
|
|
*/
|
|
|
|
async onBlockHash(buf) {
|
|
|
|
try {
|
|
|
|
// Acquire the semaphor
|
|
|
|
await this._onBlockHashSemaphor.acquire()
|
|
|
|
|
|
|
|
// Retrieve the block height
|
|
|
|
const blockHash = buf.toString('hex')
|
|
|
|
const header = await this.rpcClient.getblockheader(blockHash, true)
|
|
|
|
const height = header.height
|
|
|
|
|
|
|
|
Logger.info(`Block ${height} ${blockHash}`)
|
|
|
|
|
|
|
|
let nbTxsPushed
|
|
|
|
let rpcConnOk = true
|
|
|
|
|
|
|
|
do {
|
|
|
|
nbTxsPushed = 0
|
|
|
|
|
|
|
|
// Retrieve the transactions triggered by this block
|
|
|
|
let txs = await db.getActivatedScheduledTransactions(height)
|
|
|
|
if (!(txs && txs.length > 0))
|
|
|
|
break
|
|
|
|
|
|
|
|
for (let tx of txs) {
|
|
|
|
let hasParentTx = (tx.schParentTxid != null) && (tx.schParentTxid != '')
|
|
|
|
let parentTx = null
|
|
|
|
|
|
|
|
// Check if previous transaction has been confirmed
|
|
|
|
if (hasParentTx) {
|
|
|
|
try {
|
|
|
|
parentTx = await this.rpcClient.getrawtransaction(tx.schParentTxid, true)
|
|
|
|
} catch(e) {
|
|
|
|
Logger.error(e, 'Transaction.getTransaction()')
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((!hasParentTx) || (parentTx && parentTx.confirmations && (parentTx.confirmations >= tx.schDelay))) {
|
|
|
|
// Push the transaction
|
|
|
|
try {
|
|
|
|
await pushTxProcessor.pushTx(tx.schRaw)
|
|
|
|
Logger.info(`Pushed scheduled transaction ${tx.schTxid}`)
|
|
|
|
} catch(e) {
|
|
|
|
const msg = 'A problem was met while trying to push a scheduled transaction'
|
|
|
|
Logger.error(e, `Orchestrator.onBlockHash() : ${msg}`)
|
|
|
|
// Check if it's an issue with the connection to the RPC API
|
|
|
|
// (=> immediately stop the loop)
|
|
|
|
if (RpcClient.isConnectionError(e)) {
|
|
|
|
Logger.info('Connection issue')
|
|
|
|
rpcConnOk = false
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Update triggers of next transactions if needed
|
|
|
|
if (tx.schTrigger < height) {
|
|
|
|
const shift = height - tx.schTrigger
|
|
|
|
try {
|
|
|
|
await this.updateTriggers(tx.schID, shift)
|
|
|
|
} catch(e) {
|
|
|
|
const msg = 'A problem was met while shifting scheduled transactions'
|
|
|
|
Logger.error(e, `Orchestrator.onBlockHash() : ${msg}`)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Delete the transaction
|
|
|
|
try {
|
|
|
|
await db.deleteScheduledTransaction(tx.schTxid)
|
|
|
|
// Count the transaction as successfully processed
|
|
|
|
nbTxsPushed++
|
|
|
|
} catch(e) {
|
|
|
|
const msg = 'A problem was met while trying to delete a scheduled transaction'
|
|
|
|
Logger.error(e, `Orchestrator.onBlockHash() : ${msg}`)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} while (rpcConnOk && nbTxsPushed > 0)
|
|
|
|
|
|
|
|
} catch(e) {
|
|
|
|
Logger.error(e, 'Orchestrator.onBlockHash() : Error')
|
|
|
|
} finally {
|
|
|
|
// Release the semaphor
|
|
|
|
await this._onBlockHashSemaphor.release()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Update triggers in chain of transactions
|
|
|
|
* following a transaction identified by its txid
|
|
|
|
* @param {integer} parentId - parent id
|
|
|
|
* @param {integer} shift - delta to be added to the triggers
|
|
|
|
*/
|
|
|
|
async updateTriggers(parentId, shift) {
|
|
|
|
if (shift == 0)
|
|
|
|
return
|
|
|
|
|
|
|
|
const txs = await db.getNextScheduledTransactions(parentId)
|
|
|
|
|
|
|
|
for (let tx of txs) {
|
|
|
|
// Update the trigger of the transaction
|
|
|
|
const newTrigger = tx.schTrigger + shift
|
|
|
|
await db.updateTriggerScheduledTransaction(tx.schID, newTrigger)
|
|
|
|
// Update the triggers of next transactions in the chain
|
|
|
|
await this.updateTriggers(tx.schID, shift)
|
|
|
|
Logger.info(`Rescheduled tx ${tx.schTxid} (trigger=${newTrigger})`)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
module.exports = Orchestrator
|