lamassu-server/lib/plugins/common/json-rpc.js
2022-02-22 00:55:01 +00:00

229 lines
5.9 KiB
JavaScript

// JSON-RPC for bitcoind-like interfaces
const axios = require('axios')
const uuid = require('uuid')
const fs = require('fs')
const _ = require('lodash/fp')
const request = require('request-promise')
const { utils: coinUtils } = require('@lamassu/coins')
const options = require('../../options')
const blockchainDir = options.blockchainDir
module.exports = {
fetch, fetchDigest, createDigestRequest, parseConf, rpcConfig
}
function fetch (account = {}, method, params) {
params = _.defaultTo([], params)
return Promise.resolve(true)
.then(() => {
const data = {
method,
params,
id: uuid.v4()
}
if (_.isNil(account.port)) throw new Error('port attribute required for jsonRpc')
const url = _.defaultTo(`http://localhost:${account.port}`, account.url)
return axios({
method: 'post',
auth: {username: account.username, password: account.password},
url,
data
})
})
.then(r => {
if (r.error) throw r.error
return r.data.result
})
.catch(err => {
throw new Error(_.join(' ', [
'json-rpc::axios error:',
JSON.stringify(_.get('message', err, '')),
JSON.stringify(_.get('response.data.error', err, ''))
]))
})
}
var fetchDigestRequestQueue = []
var fetchDigestResponses = {}
function createDigestRequest (account = {}, method, params = []) {
const requestId = uuid.v4()
fetchDigestRequestQueue.push({
requestId,
options: generateDigestOptions(account, method, params, requestId),
status: 'sendPending'
})
fetchDigestResponses[requestId] = { status: 'receivePending' }
return new Promise((resolve, reject) => {
setTimeout(() => {
const request = _.find(it => it.requestId === requestId, fetchDigestRequestQueue)
const response = fetchDigestResponses[requestId]
if (_.isNil(response) || request.status === 'sendError') {
fetchDigestResponses[requestId].status = 'receiveError'
return reject()
}
fetchDigestResponses[requestId].status = 'finished'
return resolve(response)
}, 750)
})
}
function generateDigestOptions (account = {}, method, params, requestId) {
const headers = {
'Content-Type': 'application/json'
}
const dataString = `{"jsonrpc":"2.0","id":"${requestId}","method":"${method}","params":${JSON.stringify(params)}}`
const options = {
url: `http://localhost:${account.port}/json_rpc`,
method: 'POST',
headers,
body: dataString,
forever: true,
auth: {
user: account.username,
pass: account.password,
sendImmediately: false
}
}
return options
}
function executeRequestQueue (queue) {
return _.reduce(
(acc, value) => acc.then(a => {
const originalReq = _.find(it => it.requestId === value.requestId, queue)
switch (originalReq.status) {
case 'sendPending':
return request(value.options)
.then(res => {
originalReq.status = 'sendComplete'
return Promise.resolve(
_.merge(a, { [value.requestId]: { status: 'receiveComplete', ...JSON.parse(res).result } })
)
})
.catch(err => {
originalReq.status = 'sendError'
throw err
})
case 'sendComplete':
return Promise.resolve({ ...a })
case 'sendError':
return Promise.resolve({ ...a })
default:
break
}
}),
Promise.resolve(fetchDigestResponses),
queue
).then(r => {
fetchDigestResponses = r
return fetchDigestResponses
})
}
function trimReqRes () {
const requestIds = _.keys(fetchDigestResponses)
_.forEach(it => {
const request = _.find(ite => ite.requestId === it, fetchDigestRequestQueue)
const response = fetchDigestResponses[it]
if (request && response && response.status === 'finished' && request.status === 'sendComplete') {
fetchDigestRequestQueue = _.filter(ite => ite.requestId !== it, fetchDigestRequestQueue)
delete fetchDigestResponses[it]
}
}, requestIds)
}
setInterval(
() => {
if (_.size(fetchDigestRequestQueue) === 0 && _.size(_.keys(fetchDigestResponses)) === 0) {
return
}
trimReqRes()
return executeRequestQueue(fetchDigestRequestQueue)
},
500
)
function fetchDigest(account = {}, method, params = []) {
return Promise.resolve(true)
.then(() => {
if (_.isNil(account.port))
throw new Error('port attribute required for jsonRpc')
const headers = {
'Content-Type': 'application/json'
}
const dataString = `{"jsonrpc":"2.0","id":"${uuid.v4()}","method":"${method}","params":${JSON.stringify(params)}}`
const options = {
url: `http://localhost:${account.port}/json_rpc`,
method: 'POST',
headers,
body: dataString,
forever: true,
auth: {
user: account.username,
pass: account.password,
sendImmediately: false
}
}
return request(options)
})
.then((res) => {
const r = JSON.parse(res)
if (r.error) throw r.error
return r.result
})
}
function split (str) {
const i = str.indexOf('=')
if (i === -1) return []
return [str.slice(0, i), str.slice(i + 1)]
}
function parseConf (confPath) {
const conf = fs.readFileSync(confPath)
const lines = conf.toString().split('\n')
const res = {}
for (let i = 0; i < lines.length; i++) {
const keyVal = split(lines[i])
// skip when value is empty
if (!keyVal[1]) continue
res[keyVal[0]] = keyVal[1]
}
return res
}
function rpcConfig (cryptoRec) {
try {
const configPath = coinUtils.configPath(cryptoRec, blockchainDir)
const config = parseConf(configPath)
return {
username: config.rpcuser,
password: config.rpcpassword,
port: config.rpcport || cryptoRec.defaultPort
}
} catch (err) {
throw new Error('Wallet is currently not installed')
}
}