/*! * accounts/transactions-fees-rest-api.js * Copyright © 2019 – Katana Cryptographic Ltd. All Rights Reserved. */ 'use strict' const validator = require('validator') const Logger = require('../lib/logger') const errors = require('../lib/errors') const rpcTxns = require('../lib/bitcoind-rpc/transactions') const authMgr = require('../lib/auth/authorizations-manager') const HttpServer = require('../lib/http-server/http-server') const walletService = require('../lib/wallet/wallet-service') const network = require('../lib/bitcoin/network') const apiHelper = require('./api-helper') const keys = require('../keys')[network.key] const debugApi = !!(process.argv.indexOf('api-debug') > -1) /** * Transactions API endpoints */ class TransactionsRestApi { /** * Constructor * @param {pushtx.HttpServer} httpServer - HTTP server */ constructor(httpServer) { this.httpServer = httpServer // Establish routes this.httpServer.app.get( '/tx/:txid', authMgr.checkAuthentication.bind(authMgr), this.validateArgsGetTransaction.bind(this), this.getTransaction.bind(this), HttpServer.sendAuthError ) this.httpServer.app.get( '/txs', authMgr.checkAuthentication.bind(authMgr), apiHelper.validateEntitiesParams.bind(apiHelper), this.validateArgsGetTransactions.bind(this), this.getTransactions.bind(this), HttpServer.sendAuthError ) } /** * Retrieve the transaction for a given tiid * @param {object} req - http request object * @param {object} res - http response object */ async getTransaction(req, res) { try { const tx = await rpcTxns.getTransaction(req.params.txid, req.query.fees) const ret = JSON.stringify(tx, null, 2) HttpServer.sendRawData(res, ret) } catch(e) { HttpServer.sendError(res, e) } finally { const strParams = `${req.query.fees ? req.query.fees : ''}` debugApi && Logger.info(`API : Completed GET /tx/${req.params.txid} ${strParams}`) } } /** * Retrieve a page of transactions related to a wallet * @param {object} req - http request object * @param {object} res - http response object */ async getTransactions(req, res) { try { // Check request params if (!apiHelper.checkEntitiesParams(req.query)) return HttpServer.sendError(res, errors.multiaddr.NOACT) // Parse params const active = apiHelper.parseEntities(req.query.active) const page = req.query.page != null ? parseInt(req.query.page) : 0 const count = req.query.count != null ? parseInt(req.query.count) : keys.multiaddr.transactions const result = await walletService.getWalletTransactions(active, page, count) const ret = JSON.stringify(result, null, 2) HttpServer.sendRawData(res, ret) } catch(e) { HttpServer.sendError(res, e) } finally { const strParams = `${req.query.active} \ ${req.query.page ? req.query.page : ''} \ ${req.query.count ? req.query.count : ''}` debugApi && Logger.info(`API : Completed GET /txs ${strParams}`) } } /** * Validate arguments of /tx requests * @param {object} req - http request object * @param {object} res - http response object * @param {function} next - next express middleware */ validateArgsGetTransaction(req, res, next) { const isValidTxid = validator.isHash(req.params.txid, 'sha256') const isValidFees = !req.query.fees || validator.isAlphanumeric(req.query.fees) if (!(isValidTxid && isValidFees)) { HttpServer.sendError(res, errors.body.INVDATA) Logger.error( req.params, 'API : HeadersRestApi.validateArgsGetTransaction() : Invalid arguments' ) Logger.error(req.query, '') } else { next() } } /** * Validate arguments of /txs requests * @param {object} req - http request object * @param {object} res - http response object * @param {function} next - next express middleware */ validateArgsGetTransactions(req, res, next) { const isValidPage = !req.query.page || validator.isInt(req.query.page) const isValidCount = !req.query.count || validator.isInt(req.query.count) if (!(isValidPage && isValidCount)) { HttpServer.sendError(res, errors.body.INVDATA) Logger.error( req.query, 'API : HeadersRestApi.validateArgsGetTransactions() : Invalid arguments' ) } else { next() } } } module.exports = TransactionsRestApi