import config from '../../config'; import { AbstractBitcoinApi } from './bitcoin-api-abstract-factory'; import { IBitcoinApi } from './bitcoin-api.interface'; import { IEsploraApi } from './esplora-api.interface'; import { IElectrumApi } from './electrum-api.interface'; import BitcoinApi from './bitcoin-api'; import mempool from '../mempool'; import logger from '../../logger'; import * as ElectrumClient from '@mempool/electrum-client'; import * as sha256 from 'crypto-js/sha256'; import * as hexEnc from 'crypto-js/enc-hex'; import loadingIndicators from '../loading-indicators'; import memoryCache from '../memory-cache'; class BitcoindElectrsApi extends BitcoinApi implements AbstractBitcoinApi { private electrumClient: any; constructor() { super(); const electrumConfig = { client: 'mempool-v2', version: '1.4' }; const electrumPersistencePolicy = { retryPeriod: 10000, maxRetry: 1000, callback: null }; const electrumCallbacks = { onConnect: (client, versionInfo) => { logger.info(`Connected to Electrum Server at ${config.ELECTRUM.HOST}:${config.ELECTRUM.PORT} (${JSON.stringify(versionInfo)})`); }, onClose: (client) => { logger.info(`Disconnected from Electrum Server at ${config.ELECTRUM.HOST}:${config.ELECTRUM.PORT}`); }, onError: (err) => { logger.err(`Electrum error: ${JSON.stringify(err)}`); }, onLog: (str) => { logger.debug(str); }, }; this.electrumClient = new ElectrumClient( config.ELECTRUM.PORT, config.ELECTRUM.HOST, config.ELECTRUM.TLS_ENABLED ? 'tls' : 'tcp', null, electrumCallbacks ); this.electrumClient.initElectrum(electrumConfig, electrumPersistencePolicy) .then(() => {}) .catch((err) => { logger.err(`Error connecting to Electrum Server at ${config.ELECTRUM.HOST}:${config.ELECTRUM.PORT}`); }); } async $getAddress(address: string): Promise { const addressInfo = await this.$validateAddress(address); if (!addressInfo || !addressInfo.isvalid) { return ({ 'address': address, 'chain_stats': { 'funded_txo_count': 0, 'funded_txo_sum': 0, 'spent_txo_count': 0, 'spent_txo_sum': 0, 'tx_count': 0 }, 'mempool_stats': { 'funded_txo_count': 0, 'funded_txo_sum': 0, 'spent_txo_count': 0, 'spent_txo_sum': 0, 'tx_count': 0 } }); } try { const balance = await this.$getScriptHashBalance(addressInfo.scriptPubKey); const history = await this.$getScriptHashHistory(addressInfo.scriptPubKey); const unconfirmed = history.filter((h) => h.fee).length; return { 'address': addressInfo.address, 'chain_stats': { 'funded_txo_count': 0, 'funded_txo_sum': balance.confirmed ? balance.confirmed : 0, 'spent_txo_count': 0, 'spent_txo_sum': balance.confirmed < 0 ? balance.confirmed : 0, 'tx_count': history.length - unconfirmed, }, 'mempool_stats': { 'funded_txo_count': 0, 'funded_txo_sum': balance.unconfirmed > 0 ? balance.unconfirmed : 0, 'spent_txo_count': 0, 'spent_txo_sum': balance.unconfirmed < 0 ? -balance.unconfirmed : 0, 'tx_count': unconfirmed, }, 'electrum': true, }; } catch (e) { if (e === 'failed to get confirmed status') { e = 'The number of transactions on this address exceeds the Electrum server limit'; } throw new Error(e instanceof Error ? e.message : 'Error'); } } async $getAddressTransactions(address: string, lastSeenTxId: string): Promise { const addressInfo = await this.$validateAddress(address); if (!addressInfo || !addressInfo.isvalid) { return []; } try { loadingIndicators.setProgress('address-' + address, 0); const transactions: IEsploraApi.Transaction[] = []; const history = await this.$getScriptHashHistory(addressInfo.scriptPubKey); history.sort((a, b) => (b.height || 9999999) - (a.height || 9999999)); let startingIndex = 0; if (lastSeenTxId) { const pos = history.findIndex((historicalTx) => historicalTx.tx_hash === lastSeenTxId); if (pos) { startingIndex = pos + 1; } } const endIndex = Math.min(startingIndex + 10, history.length); for (let i = startingIndex; i < endIndex; i++) { const tx = await this.$getRawTransaction(history[i].tx_hash, false, true); transactions.push(tx); loadingIndicators.setProgress('address-' + address, (i + 1) / endIndex * 100); } return transactions; } catch (e) { loadingIndicators.setProgress('address-' + address, 100); if (e === 'failed to get confirmed status') { e = 'The number of transactions on this address exceeds the Electrum server limit'; } throw new Error(e instanceof Error ? e.message : 'Error'); } } private $getScriptHashBalance(scriptHash: string): Promise { return this.electrumClient.blockchainScripthash_getBalance(this.encodeScriptHash(scriptHash)); } private $getScriptHashHistory(scriptHash: string): Promise { const fromCache = memoryCache.get('Scripthash_getHistory', scriptHash); if (fromCache) { return Promise.resolve(fromCache); } return this.electrumClient.blockchainScripthash_getHistory(this.encodeScriptHash(scriptHash)) .then((history) => { memoryCache.set('Scripthash_getHistory', scriptHash, history, 2); return history; }); } private encodeScriptHash(scriptPubKey: string): string { const addrScripthash = hexEnc.stringify(sha256(hexEnc.parse(scriptPubKey))); return addrScripthash.match(/.{2}/g).reverse().join(''); } } export default BitcoindElectrsApi;