140 lines
5.0 KiB
TypeScript
140 lines
5.0 KiB
TypeScript
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 bitcoinBaseApi from './bitcoin-base.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';
|
|
|
|
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.ELECTRS.HOST}:${config.ELECTRS.PORT} (${JSON.stringify(versionInfo)})`); },
|
|
onClose: (client) => { logger.info(`Disconnected from Electrum Server at ${config.ELECTRS.HOST}:${config.ELECTRS.PORT}`); },
|
|
onError: (err) => { logger.err(`Electrum error: ${JSON.stringify(err)}`); },
|
|
onLog: (str) => { logger.debug(str); },
|
|
};
|
|
|
|
this.electrumClient = new ElectrumClient(
|
|
config.ELECTRS.PORT,
|
|
config.ELECTRS.HOST,
|
|
config.ELECTRS.PORT === 50001 ? 'tcp' : 'tls',
|
|
null,
|
|
electrumCallbacks
|
|
);
|
|
|
|
this.electrumClient.initElectrum(electrumConfig, electrumPersistencePolicy)
|
|
.then(() => {})
|
|
.catch((err) => {
|
|
logger.err(`Error connecting to Electrum Server at ${config.ELECTRS.HOST}:${config.ELECTRS.PORT}`);
|
|
});
|
|
}
|
|
|
|
async $getRawTransaction(txId: string, skipConversion = false, addPrevout = false): Promise<IEsploraApi.Transaction> {
|
|
const txInMempool = mempool.getMempool()[txId];
|
|
if (txInMempool && addPrevout) {
|
|
return this.$addPrevouts(txInMempool);
|
|
}
|
|
const transaction: IBitcoinApi.Transaction = await this.electrumClient.blockchainTransaction_get(txId, true);
|
|
if (!transaction) {
|
|
throw new Error('Unable to get transaction: ' + txId);
|
|
}
|
|
if (skipConversion) {
|
|
// @ts-ignore
|
|
return transaction;
|
|
}
|
|
return this.$convertTransaction(transaction, addPrevout);
|
|
}
|
|
|
|
async $getAddress(address: string): Promise<IEsploraApi.Address> {
|
|
const addressInfo = await bitcoinBaseApi.$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
|
|
}
|
|
});
|
|
}
|
|
|
|
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,
|
|
}
|
|
};
|
|
|
|
}
|
|
|
|
async $getAddressTransactions(address: string, lastSeenTxId: string): Promise<IEsploraApi.Transaction[]> {
|
|
const addressInfo = await bitcoinBaseApi.$validateAddress(address);
|
|
if (!addressInfo || !addressInfo.isvalid) {
|
|
return [];
|
|
}
|
|
const history = await this.$getScriptHashHistory(addressInfo.scriptPubKey);
|
|
const transactions: IEsploraApi.Transaction[] = [];
|
|
for (const h of history) {
|
|
const tx = await this.$getRawTransaction(h.tx_hash, false, true);
|
|
if (tx) {
|
|
transactions.push(tx);
|
|
}
|
|
}
|
|
return transactions;
|
|
}
|
|
|
|
private $getScriptHashBalance(scriptHash: string): Promise<IElectrumApi.ScriptHashBalance> {
|
|
return this.electrumClient.blockchainScripthash_getBalance(this.encodeScriptHash(scriptHash));
|
|
}
|
|
|
|
private $getScriptHashHistory(scriptHash: string): Promise<IElectrumApi.ScriptHashHistory[]> {
|
|
return this.electrumClient.blockchainScripthash_getHistory(this.encodeScriptHash(scriptHash));
|
|
}
|
|
|
|
private encodeScriptHash(scriptPubKey: string): string {
|
|
const addrScripthash = hexEnc.stringify(sha256(hexEnc.parse(scriptPubKey)));
|
|
return addrScripthash.match(/.{2}/g).reverse().join('');
|
|
}
|
|
|
|
}
|
|
|
|
export default BitcoindElectrsApi;
|