2019-03-30 18:00:34 +01:00
|
|
|
/* global it, describe, afterAll, beforeAll, jasmine */
|
2019-01-29 03:27:07 +01:00
|
|
|
global.net = require('net');
|
|
|
|
let BlueElectrum = require('./BlueElectrum');
|
|
|
|
let assert = require('assert');
|
2019-05-22 01:06:17 +02:00
|
|
|
let bitcoin = require('bitcoinjs-lib');
|
2019-04-01 02:01:11 +02:00
|
|
|
jasmine.DEFAULT_TIMEOUT_INTERVAL = 150 * 1000;
|
2019-03-30 18:00:34 +01:00
|
|
|
|
|
|
|
afterAll(() => {
|
|
|
|
// after all tests we close socket so the test suite can actually terminate
|
2019-05-06 00:17:31 +02:00
|
|
|
BlueElectrum.forceDisconnect();
|
|
|
|
return new Promise(resolve => setTimeout(resolve, 10000)); // simple sleep to wait for all timeouts termination
|
2019-03-30 18:00:34 +01:00
|
|
|
});
|
|
|
|
|
|
|
|
beforeAll(async () => {
|
|
|
|
// awaiting for Electrum to be connected. For RN Electrum would naturally connect
|
|
|
|
// while app starts up, but for tests we need to wait for it
|
|
|
|
try {
|
|
|
|
await BlueElectrum.waitTillConnected();
|
2019-05-02 22:33:03 +02:00
|
|
|
} catch (err) {
|
|
|
|
console.log('failed to connect to Electrum:', err);
|
2019-03-30 18:00:34 +01:00
|
|
|
process.exit(1);
|
|
|
|
}
|
|
|
|
});
|
2019-01-29 03:27:07 +01:00
|
|
|
|
|
|
|
describe('Electrum', () => {
|
|
|
|
it('ElectrumClient can connect and query', async () => {
|
|
|
|
const ElectrumClient = require('electrum-client');
|
|
|
|
|
2019-04-01 02:01:11 +02:00
|
|
|
for (let peer of BlueElectrum.hardcodedPeers) {
|
|
|
|
let mainClient = new ElectrumClient(peer.tcp, peer.host, 'tcp');
|
2019-01-29 03:27:07 +01:00
|
|
|
|
2019-04-01 02:01:11 +02:00
|
|
|
try {
|
|
|
|
await mainClient.connect();
|
2019-04-20 13:52:15 +02:00
|
|
|
await mainClient.server_version('2.7.11', '1.4');
|
2019-04-01 02:01:11 +02:00
|
|
|
} catch (e) {
|
|
|
|
mainClient.reconnect = mainClient.keepAlive = () => {}; // dirty hack to make it stop reconnecting
|
|
|
|
mainClient.close();
|
2019-04-20 13:52:15 +02:00
|
|
|
throw new Error('bad connection: ' + JSON.stringify(peer) + ' ' + e.message);
|
2019-04-01 02:01:11 +02:00
|
|
|
}
|
2019-01-29 03:27:07 +01:00
|
|
|
|
2019-04-01 02:01:11 +02:00
|
|
|
let addr4elect = 'bc1qwqdg6squsna38e46795at95yu9atm8azzmyvckulcc7kytlcckxswvvzej';
|
|
|
|
let script = bitcoin.address.toOutputScript(addr4elect);
|
|
|
|
let hash = bitcoin.crypto.sha256(script);
|
|
|
|
let reversedHash = Buffer.from(hash.reverse());
|
|
|
|
let start = +new Date();
|
|
|
|
let balance = await mainClient.blockchainScripthash_getBalance(reversedHash.toString('hex'));
|
|
|
|
let end = +new Date();
|
|
|
|
console.warn(peer.host, 'took', (end - start) / 1000, 'seconds to fetch balance');
|
|
|
|
assert.ok(balance.confirmed > 0);
|
2019-01-29 03:27:07 +01:00
|
|
|
|
2019-04-01 02:01:11 +02:00
|
|
|
addr4elect = '3GCvDBAktgQQtsbN6x5DYiQCMmgZ9Yk8BK';
|
|
|
|
script = bitcoin.address.toOutputScript(addr4elect);
|
|
|
|
hash = bitcoin.crypto.sha256(script);
|
|
|
|
reversedHash = Buffer.from(hash.reverse());
|
|
|
|
balance = await mainClient.blockchainScripthash_getBalance(reversedHash.toString('hex'));
|
2019-01-29 03:27:07 +01:00
|
|
|
|
2019-04-01 02:01:11 +02:00
|
|
|
// let peers = await mainClient.serverPeers_subscribe();
|
|
|
|
// console.log(peers);
|
|
|
|
mainClient.reconnect = mainClient.keepAlive = () => {}; // dirty hack to make it stop reconnecting
|
|
|
|
mainClient.close();
|
|
|
|
}
|
2019-01-29 03:27:07 +01:00
|
|
|
});
|
|
|
|
|
2019-05-06 00:17:31 +02:00
|
|
|
it('BlueElectrum can do getBalanceByAddress()', async function() {
|
2019-01-29 03:27:07 +01:00
|
|
|
let address = '3GCvDBAktgQQtsbN6x5DYiQCMmgZ9Yk8BK';
|
|
|
|
let balance = await BlueElectrum.getBalanceByAddress(address);
|
|
|
|
assert.strictEqual(balance.confirmed, 51432);
|
|
|
|
assert.strictEqual(balance.unconfirmed, 0);
|
|
|
|
assert.strictEqual(balance.addr, address);
|
2019-05-06 00:17:31 +02:00
|
|
|
});
|
2019-01-29 03:27:07 +01:00
|
|
|
|
2019-05-06 00:17:31 +02:00
|
|
|
it('BlueElectrum can do getTransactionsByAddress()', async function() {
|
|
|
|
let txs = await BlueElectrum.getTransactionsByAddress('bc1qt4t9xl2gmjvxgmp5gev6m8e6s9c85979ta7jeh');
|
2019-01-29 03:27:07 +01:00
|
|
|
assert.strictEqual(txs.length, 1);
|
2019-05-06 00:17:31 +02:00
|
|
|
assert.strictEqual(txs[0].tx_hash, 'ad00a92409d8982a1d7f877056dbed0c4337d2ebab70b30463e2802279fb936d');
|
|
|
|
assert.strictEqual(txs[0].height, 563077);
|
|
|
|
});
|
|
|
|
|
2019-05-12 22:49:43 +02:00
|
|
|
it('BlueElectrum can do getTransactionsFullByAddress()', async function() {
|
2019-05-06 00:17:31 +02:00
|
|
|
let txs = await BlueElectrum.getTransactionsFullByAddress('bc1qt4t9xl2gmjvxgmp5gev6m8e6s9c85979ta7jeh');
|
2019-01-29 03:27:07 +01:00
|
|
|
for (let tx of txs) {
|
2019-05-06 00:17:31 +02:00
|
|
|
assert.ok(tx.address === 'bc1qt4t9xl2gmjvxgmp5gev6m8e6s9c85979ta7jeh');
|
|
|
|
assert.ok(tx.txid);
|
|
|
|
assert.ok(tx.confirmations);
|
2019-05-09 10:35:57 +02:00
|
|
|
assert.ok(!tx.vin);
|
|
|
|
assert.ok(!tx.vout);
|
|
|
|
assert.ok(tx.inputs);
|
|
|
|
assert.ok(tx.inputs[0].addresses.length > 0);
|
|
|
|
assert.ok(tx.inputs[0].value > 0);
|
|
|
|
assert.ok(tx.outputs);
|
|
|
|
assert.ok(tx.outputs[0].value > 0);
|
|
|
|
assert.ok(tx.outputs[0].scriptPubKey);
|
|
|
|
assert.ok(tx.outputs[0].addresses.length > 0);
|
2019-01-29 03:27:07 +01:00
|
|
|
}
|
|
|
|
});
|
2019-05-06 00:17:31 +02:00
|
|
|
|
|
|
|
it('BlueElectrum can do multiGetBalanceByAddress()', async function() {
|
|
|
|
let balances = await BlueElectrum.multiGetBalanceByAddress([
|
|
|
|
'bc1qt4t9xl2gmjvxgmp5gev6m8e6s9c85979ta7jeh',
|
|
|
|
'bc1qvd6w54sydc08z3802svkxr7297ez7cusd6266p',
|
|
|
|
'bc1qwp58x4c9e5cplsnw5096qzdkae036ug7a34x3r',
|
|
|
|
'bc1qcg6e26vtzja0h8up5w2m7utex0fsu4v0e0e7uy',
|
|
|
|
]);
|
|
|
|
|
|
|
|
assert.strictEqual(balances.balance, 200000);
|
|
|
|
assert.strictEqual(balances.unconfirmed_balance, 0);
|
|
|
|
assert.strictEqual(balances.addresses['bc1qt4t9xl2gmjvxgmp5gev6m8e6s9c85979ta7jeh'].confirmed, 50000);
|
|
|
|
assert.strictEqual(balances.addresses['bc1qt4t9xl2gmjvxgmp5gev6m8e6s9c85979ta7jeh'].unconfirmed, 0);
|
|
|
|
assert.strictEqual(balances.addresses['bc1qvd6w54sydc08z3802svkxr7297ez7cusd6266p'].confirmed, 50000);
|
|
|
|
assert.strictEqual(balances.addresses['bc1qvd6w54sydc08z3802svkxr7297ez7cusd6266p'].unconfirmed, 0);
|
|
|
|
assert.strictEqual(balances.addresses['bc1qwp58x4c9e5cplsnw5096qzdkae036ug7a34x3r'].confirmed, 50000);
|
|
|
|
assert.strictEqual(balances.addresses['bc1qwp58x4c9e5cplsnw5096qzdkae036ug7a34x3r'].unconfirmed, 0);
|
|
|
|
assert.strictEqual(balances.addresses['bc1qcg6e26vtzja0h8up5w2m7utex0fsu4v0e0e7uy'].confirmed, 50000);
|
|
|
|
assert.strictEqual(balances.addresses['bc1qcg6e26vtzja0h8up5w2m7utex0fsu4v0e0e7uy'].unconfirmed, 0);
|
|
|
|
});
|
2019-05-23 18:22:23 +02:00
|
|
|
|
|
|
|
it('BlueElectrum can do multiGetUtxoByAddress()', async () => {
|
|
|
|
let utxos = await BlueElectrum.multiGetUtxoByAddress(
|
|
|
|
[
|
|
|
|
'bc1qt4t9xl2gmjvxgmp5gev6m8e6s9c85979ta7jeh',
|
|
|
|
'bc1qvd6w54sydc08z3802svkxr7297ez7cusd6266p',
|
|
|
|
'bc1qwp58x4c9e5cplsnw5096qzdkae036ug7a34x3r',
|
|
|
|
'bc1qcg6e26vtzja0h8up5w2m7utex0fsu4v0e0e7uy',
|
|
|
|
],
|
|
|
|
3,
|
|
|
|
);
|
|
|
|
|
|
|
|
assert.strictEqual(Object.keys(utxos).length, 4);
|
|
|
|
assert.strictEqual(
|
2019-05-24 21:29:21 +02:00
|
|
|
utxos['bc1qt4t9xl2gmjvxgmp5gev6m8e6s9c85979ta7jeh'][0].txId,
|
2019-05-23 18:22:23 +02:00
|
|
|
'ad00a92409d8982a1d7f877056dbed0c4337d2ebab70b30463e2802279fb936d',
|
|
|
|
);
|
2019-05-24 21:29:21 +02:00
|
|
|
assert.strictEqual(utxos['bc1qt4t9xl2gmjvxgmp5gev6m8e6s9c85979ta7jeh'][0].vout, 1);
|
2019-05-23 18:22:23 +02:00
|
|
|
assert.strictEqual(utxos['bc1qt4t9xl2gmjvxgmp5gev6m8e6s9c85979ta7jeh'][0].value, 50000);
|
|
|
|
assert.strictEqual(utxos['bc1qt4t9xl2gmjvxgmp5gev6m8e6s9c85979ta7jeh'][0].address, 'bc1qt4t9xl2gmjvxgmp5gev6m8e6s9c85979ta7jeh');
|
|
|
|
});
|
2019-01-29 03:27:07 +01:00
|
|
|
});
|