forked from bitcoin/bitcoin
-
Notifications
You must be signed in to change notification settings - Fork 40
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Merge pull request #6 from braydonf/0.12-bitcore-addressindex
Address and additional indexes
- Loading branch information
Showing
25 changed files
with
1,891 additions
and
3 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,283 @@ | ||
#!/usr/bin/env python2 | ||
# Copyright (c) 2014-2015 The Bitcoin Core developers | ||
# Distributed under the MIT software license, see the accompanying | ||
# file COPYING or http://www.opensource.org/licenses/mit-license.php. | ||
|
||
# | ||
# Test addressindex generation and fetching | ||
# | ||
|
||
import time | ||
from test_framework.test_framework import BitcoinTestFramework | ||
from test_framework.util import * | ||
from test_framework.script import * | ||
from test_framework.mininode import * | ||
import binascii | ||
|
||
class AddressIndexTest(BitcoinTestFramework): | ||
|
||
def setup_chain(self): | ||
print("Initializing test directory "+self.options.tmpdir) | ||
initialize_chain_clean(self.options.tmpdir, 4) | ||
|
||
def setup_network(self): | ||
self.nodes = [] | ||
# Nodes 0/1 are "wallet" nodes | ||
self.nodes.append(start_node(0, self.options.tmpdir, ["-debug"])) | ||
self.nodes.append(start_node(1, self.options.tmpdir, ["-debug", "-addressindex"])) | ||
# Nodes 2/3 are used for testing | ||
self.nodes.append(start_node(2, self.options.tmpdir, ["-debug", "-addressindex"])) | ||
self.nodes.append(start_node(3, self.options.tmpdir, ["-debug", "-addressindex"])) | ||
connect_nodes(self.nodes[0], 1) | ||
connect_nodes(self.nodes[0], 2) | ||
connect_nodes(self.nodes[0], 3) | ||
|
||
self.is_network_split = False | ||
self.sync_all() | ||
|
||
def run_test(self): | ||
print "Mining blocks..." | ||
self.nodes[0].generate(105) | ||
self.sync_all() | ||
|
||
chain_height = self.nodes[1].getblockcount() | ||
assert_equal(chain_height, 105) | ||
assert_equal(self.nodes[1].getbalance(), 0) | ||
assert_equal(self.nodes[2].getbalance(), 0) | ||
|
||
# Check that balances are correct | ||
balance0 = self.nodes[1].getaddressbalance("2N2JD6wb56AfK4tfmM6PwdVmoYk2dCKf4Br") | ||
assert_equal(balance0["balance"], 0) | ||
|
||
# Check p2pkh and p2sh address indexes | ||
print "Testing p2pkh and p2sh address index..." | ||
|
||
txid0 = self.nodes[0].sendtoaddress("mo9ncXisMeAoXwqcV5EWuyncbmCcQN4rVs", 10) | ||
self.nodes[0].generate(1) | ||
|
||
txidb0 = self.nodes[0].sendtoaddress("2N2JD6wb56AfK4tfmM6PwdVmoYk2dCKf4Br", 10) | ||
self.nodes[0].generate(1) | ||
|
||
txid1 = self.nodes[0].sendtoaddress("mo9ncXisMeAoXwqcV5EWuyncbmCcQN4rVs", 15) | ||
self.nodes[0].generate(1) | ||
|
||
txidb1 = self.nodes[0].sendtoaddress("2N2JD6wb56AfK4tfmM6PwdVmoYk2dCKf4Br", 15) | ||
self.nodes[0].generate(1) | ||
|
||
txid2 = self.nodes[0].sendtoaddress("mo9ncXisMeAoXwqcV5EWuyncbmCcQN4rVs", 20) | ||
self.nodes[0].generate(1) | ||
|
||
txidb2 = self.nodes[0].sendtoaddress("2N2JD6wb56AfK4tfmM6PwdVmoYk2dCKf4Br", 20) | ||
self.nodes[0].generate(1) | ||
|
||
self.sync_all() | ||
|
||
txids = self.nodes[1].getaddresstxids("mo9ncXisMeAoXwqcV5EWuyncbmCcQN4rVs") | ||
assert_equal(len(txids), 3) | ||
assert_equal(txids[0], txid0) | ||
assert_equal(txids[1], txid1) | ||
assert_equal(txids[2], txid2) | ||
|
||
txidsb = self.nodes[1].getaddresstxids("2N2JD6wb56AfK4tfmM6PwdVmoYk2dCKf4Br") | ||
assert_equal(len(txidsb), 3) | ||
assert_equal(txidsb[0], txidb0) | ||
assert_equal(txidsb[1], txidb1) | ||
assert_equal(txidsb[2], txidb2) | ||
|
||
# Check that limiting by height works | ||
print "Testing querying txids by range of block heights.." | ||
height_txids = self.nodes[1].getaddresstxids({ | ||
"addresses": ["2N2JD6wb56AfK4tfmM6PwdVmoYk2dCKf4Br"], | ||
"start": 105, | ||
"end": 110 | ||
}) | ||
assert_equal(len(height_txids), 2) | ||
assert_equal(height_txids[0], txidb0) | ||
assert_equal(height_txids[1], txidb1) | ||
|
||
# Check that multiple addresses works | ||
multitxids = self.nodes[1].getaddresstxids({"addresses": ["2N2JD6wb56AfK4tfmM6PwdVmoYk2dCKf4Br", "mo9ncXisMeAoXwqcV5EWuyncbmCcQN4rVs"]}) | ||
assert_equal(len(multitxids), 6) | ||
assert_equal(multitxids[0], txid0) | ||
assert_equal(multitxids[1], txidb0) | ||
assert_equal(multitxids[2], txid1) | ||
assert_equal(multitxids[3], txidb1) | ||
assert_equal(multitxids[4], txid2) | ||
assert_equal(multitxids[5], txidb2) | ||
|
||
# Check that balances are correct | ||
balance0 = self.nodes[1].getaddressbalance("2N2JD6wb56AfK4tfmM6PwdVmoYk2dCKf4Br") | ||
assert_equal(balance0["balance"], 45 * 100000000) | ||
|
||
# Check that outputs with the same address will only return one txid | ||
print "Testing for txid uniqueness..." | ||
addressHash = "6349a418fc4578d10a372b54b45c280cc8c4382f".decode("hex") | ||
scriptPubKey = CScript([OP_HASH160, addressHash, OP_EQUAL]) | ||
unspent = self.nodes[0].listunspent() | ||
tx = CTransaction() | ||
tx.vin = [CTxIn(COutPoint(int(unspent[0]["txid"], 16), unspent[0]["vout"]))] | ||
tx.vout = [CTxOut(10, scriptPubKey), CTxOut(11, scriptPubKey)] | ||
tx.rehash() | ||
|
||
signed_tx = self.nodes[0].signrawtransaction(binascii.hexlify(tx.serialize()).decode("utf-8")) | ||
sent_txid = self.nodes[0].sendrawtransaction(signed_tx["hex"], True) | ||
|
||
self.nodes[0].generate(1) | ||
self.sync_all() | ||
|
||
txidsmany = self.nodes[1].getaddresstxids("2N2JD6wb56AfK4tfmM6PwdVmoYk2dCKf4Br") | ||
assert_equal(len(txidsmany), 4) | ||
assert_equal(txidsmany[3], sent_txid) | ||
|
||
# Check that balances are correct | ||
print "Testing balances..." | ||
balance0 = self.nodes[1].getaddressbalance("2N2JD6wb56AfK4tfmM6PwdVmoYk2dCKf4Br") | ||
assert_equal(balance0["balance"], 45 * 100000000 + 21) | ||
|
||
# Check that balances are correct after spending | ||
print "Testing balances after spending..." | ||
privkey2 = "cSdkPxkAjA4HDr5VHgsebAPDEh9Gyub4HK8UJr2DFGGqKKy4K5sG" | ||
address2 = "mgY65WSfEmsyYaYPQaXhmXMeBhwp4EcsQW" | ||
addressHash2 = "0b2f0a0c31bfe0406b0ccc1381fdbe311946dadc".decode("hex") | ||
scriptPubKey2 = CScript([OP_DUP, OP_HASH160, addressHash2, OP_EQUALVERIFY, OP_CHECKSIG]) | ||
self.nodes[0].importprivkey(privkey2) | ||
|
||
unspent = self.nodes[0].listunspent() | ||
tx = CTransaction() | ||
tx.vin = [CTxIn(COutPoint(int(unspent[0]["txid"], 16), unspent[0]["vout"]))] | ||
amount = unspent[0]["amount"] * 100000000 | ||
tx.vout = [CTxOut(amount, scriptPubKey2)] | ||
tx.rehash() | ||
signed_tx = self.nodes[0].signrawtransaction(binascii.hexlify(tx.serialize()).decode("utf-8")) | ||
spending_txid = self.nodes[0].sendrawtransaction(signed_tx["hex"], True) | ||
self.nodes[0].generate(1) | ||
self.sync_all() | ||
balance1 = self.nodes[1].getaddressbalance(address2) | ||
assert_equal(balance1["balance"], amount) | ||
|
||
tx = CTransaction() | ||
tx.vin = [CTxIn(COutPoint(int(spending_txid, 16), 0))] | ||
send_amount = 1 * 100000000 + 12840 | ||
change_amount = amount - send_amount - 10000 | ||
tx.vout = [CTxOut(change_amount, scriptPubKey2), CTxOut(send_amount, scriptPubKey)] | ||
tx.rehash() | ||
|
||
signed_tx = self.nodes[0].signrawtransaction(binascii.hexlify(tx.serialize()).decode("utf-8")) | ||
sent_txid = self.nodes[0].sendrawtransaction(signed_tx["hex"], True) | ||
self.nodes[0].generate(1) | ||
self.sync_all() | ||
|
||
balance2 = self.nodes[1].getaddressbalance(address2) | ||
assert_equal(balance2["balance"], change_amount) | ||
|
||
# Check that deltas are returned correctly | ||
deltas = self.nodes[1].getaddressdeltas({"addresses": [address2], "start": 0, "end": 200}) | ||
balance3 = 0 | ||
for delta in deltas: | ||
balance3 += delta["satoshis"] | ||
assert_equal(balance3, change_amount) | ||
assert_equal(deltas[0]["address"], address2) | ||
|
||
# Check that entire range will be queried | ||
deltasAll = self.nodes[1].getaddressdeltas({"addresses": [address2]}) | ||
assert_equal(len(deltasAll), len(deltas)) | ||
|
||
# Check that deltas can be returned from range of block heights | ||
deltas = self.nodes[1].getaddressdeltas({"addresses": [address2], "start": 113, "end": 113}) | ||
assert_equal(len(deltas), 1) | ||
|
||
# Check that unspent outputs can be queried | ||
print "Testing utxos..." | ||
utxos = self.nodes[1].getaddressutxos({"addresses": [address2]}) | ||
assert_equal(len(utxos), 1) | ||
assert_equal(utxos[0]["satoshis"], change_amount) | ||
|
||
# Check that indexes will be updated with a reorg | ||
print "Testing reorg..." | ||
|
||
best_hash = self.nodes[0].getbestblockhash() | ||
self.nodes[0].invalidateblock(best_hash) | ||
self.nodes[1].invalidateblock(best_hash) | ||
self.nodes[2].invalidateblock(best_hash) | ||
self.nodes[3].invalidateblock(best_hash) | ||
self.sync_all() | ||
|
||
balance4 = self.nodes[1].getaddressbalance(address2) | ||
assert_equal(balance4, balance1) | ||
|
||
utxos2 = self.nodes[1].getaddressutxos({"addresses": [address2]}) | ||
assert_equal(len(utxos2), 1) | ||
assert_equal(utxos2[0]["satoshis"], 5000000000) | ||
|
||
# Check sorting of utxos | ||
self.nodes[2].generate(150) | ||
|
||
txidsort1 = self.nodes[2].sendtoaddress(address2, 50) | ||
self.nodes[2].generate(1) | ||
txidsort2 = self.nodes[2].sendtoaddress(address2, 50) | ||
self.nodes[2].generate(1) | ||
self.sync_all() | ||
|
||
utxos3 = self.nodes[1].getaddressutxos({"addresses": [address2]}) | ||
assert_equal(len(utxos3), 3) | ||
assert_equal(utxos3[0]["height"], 114) | ||
assert_equal(utxos3[1]["height"], 264) | ||
assert_equal(utxos3[2]["height"], 265) | ||
|
||
# Check mempool indexing | ||
print "Testing mempool indexing..." | ||
|
||
privKey3 = "cVfUn53hAbRrDEuMexyfgDpZPhF7KqXpS8UZevsyTDaugB7HZ3CD" | ||
address3 = "mw4ynwhS7MmrQ27hr82kgqu7zryNDK26JB" | ||
addressHash3 = "aa9872b5bbcdb511d89e0e11aa27da73fd2c3f50".decode("hex") | ||
scriptPubKey3 = CScript([OP_DUP, OP_HASH160, addressHash3, OP_EQUALVERIFY, OP_CHECKSIG]) | ||
unspent = self.nodes[2].listunspent() | ||
|
||
tx = CTransaction() | ||
tx.vin = [CTxIn(COutPoint(int(unspent[0]["txid"], 16), unspent[0]["vout"]))] | ||
amount = unspent[0]["amount"] * 100000000 | ||
tx.vout = [CTxOut(amount, scriptPubKey3)] | ||
tx.rehash() | ||
signed_tx = self.nodes[2].signrawtransaction(binascii.hexlify(tx.serialize()).decode("utf-8")) | ||
memtxid1 = self.nodes[2].sendrawtransaction(signed_tx["hex"], True) | ||
time.sleep(2) | ||
|
||
tx2 = CTransaction() | ||
tx2.vin = [CTxIn(COutPoint(int(unspent[1]["txid"], 16), unspent[1]["vout"]))] | ||
amount = unspent[1]["amount"] * 100000000 | ||
tx2.vout = [CTxOut(amount, scriptPubKey3)] | ||
tx2.rehash() | ||
signed_tx2 = self.nodes[2].signrawtransaction(binascii.hexlify(tx2.serialize()).decode("utf-8")) | ||
memtxid2 = self.nodes[2].sendrawtransaction(signed_tx2["hex"], True) | ||
time.sleep(2) | ||
|
||
mempool = self.nodes[2].getaddressmempool({"addresses": [address3]}) | ||
assert_equal(len(mempool), 2) | ||
assert_equal(mempool[0]["txid"], memtxid1) | ||
assert_equal(mempool[1]["txid"], memtxid2) | ||
assert_equal(mempool[0]["address"], address3) | ||
|
||
self.nodes[2].generate(1); | ||
self.sync_all(); | ||
mempool2 = self.nodes[2].getaddressmempool({"addresses": [address3]}) | ||
assert_equal(len(mempool2), 0) | ||
|
||
tx = CTransaction() | ||
tx.vin = [CTxIn(COutPoint(int(memtxid2, 16), 0))] | ||
tx.vout = [CTxOut(amount - 10000, scriptPubKey2)] | ||
tx.rehash() | ||
self.nodes[2].importprivkey(privKey3) | ||
signed_tx3 = self.nodes[2].signrawtransaction(binascii.hexlify(tx.serialize()).decode("utf-8")) | ||
memtxid3 = self.nodes[2].sendrawtransaction(signed_tx3["hex"], True) | ||
time.sleep(2) | ||
|
||
mempool3 = self.nodes[2].getaddressmempool({"addresses": [address3]}) | ||
assert_equal(len(mempool3), 1) | ||
assert_equal(mempool3[0]["prevtxid"], memtxid2) | ||
assert_equal(mempool3[0]["prevout"], 0) | ||
|
||
print "Passed\n" | ||
|
||
|
||
if __name__ == '__main__': | ||
AddressIndexTest().main() |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,81 @@ | ||
#!/usr/bin/env python2 | ||
# Copyright (c) 2014-2015 The Bitcoin Core developers | ||
# Distributed under the MIT software license, see the accompanying | ||
# file COPYING or http://www.opensource.org/licenses/mit-license.php. | ||
|
||
# | ||
# Test addressindex generation and fetching | ||
# | ||
|
||
import time | ||
from test_framework.test_framework import BitcoinTestFramework | ||
from test_framework.util import * | ||
from test_framework.script import * | ||
from test_framework.mininode import * | ||
import binascii | ||
|
||
class SpentIndexTest(BitcoinTestFramework): | ||
|
||
def setup_chain(self): | ||
print("Initializing test directory "+self.options.tmpdir) | ||
initialize_chain_clean(self.options.tmpdir, 4) | ||
|
||
def setup_network(self): | ||
self.nodes = [] | ||
# Nodes 0/1 are "wallet" nodes | ||
self.nodes.append(start_node(0, self.options.tmpdir, ["-debug"])) | ||
self.nodes.append(start_node(1, self.options.tmpdir, ["-debug", "-spentindex"])) | ||
# Nodes 2/3 are used for testing | ||
self.nodes.append(start_node(2, self.options.tmpdir, ["-debug", "-spentindex"])) | ||
self.nodes.append(start_node(3, self.options.tmpdir, ["-debug", "-spentindex", "-txindex"])) | ||
connect_nodes(self.nodes[0], 1) | ||
connect_nodes(self.nodes[0], 2) | ||
connect_nodes(self.nodes[0], 3) | ||
|
||
self.is_network_split = False | ||
self.sync_all() | ||
|
||
def run_test(self): | ||
print "Mining blocks..." | ||
self.nodes[0].generate(105) | ||
self.sync_all() | ||
|
||
chain_height = self.nodes[1].getblockcount() | ||
assert_equal(chain_height, 105) | ||
|
||
# Check that | ||
print "Testing spent index..." | ||
|
||
privkey = "cSdkPxkAjA4HDr5VHgsebAPDEh9Gyub4HK8UJr2DFGGqKKy4K5sG" | ||
address = "mgY65WSfEmsyYaYPQaXhmXMeBhwp4EcsQW" | ||
addressHash = "0b2f0a0c31bfe0406b0ccc1381fdbe311946dadc".decode("hex") | ||
scriptPubKey = CScript([OP_DUP, OP_HASH160, addressHash, OP_EQUALVERIFY, OP_CHECKSIG]) | ||
unspent = self.nodes[0].listunspent() | ||
tx = CTransaction() | ||
amount = unspent[0]["amount"] * 100000000 | ||
tx.vin = [CTxIn(COutPoint(int(unspent[0]["txid"], 16), unspent[0]["vout"]))] | ||
tx.vout = [CTxOut(amount, scriptPubKey)] | ||
tx.rehash() | ||
|
||
signed_tx = self.nodes[0].signrawtransaction(binascii.hexlify(tx.serialize()).decode("utf-8")) | ||
txid = self.nodes[0].sendrawtransaction(signed_tx["hex"], True) | ||
self.nodes[0].generate(1) | ||
self.sync_all() | ||
|
||
# Check that the spentinfo works standalone | ||
info = self.nodes[1].getspentinfo({"txid": unspent[0]["txid"], "index": unspent[0]["vout"]}) | ||
assert_equal(info["txid"], txid) | ||
assert_equal(info["index"], 0) | ||
assert_equal(info["height"], 106) | ||
|
||
# Check that verbose raw transaction includes spent info | ||
txVerbose = self.nodes[3].getrawtransaction(unspent[0]["txid"], 1) | ||
assert_equal(txVerbose["vout"][unspent[0]["vout"]]["spentTxId"], txid) | ||
assert_equal(txVerbose["vout"][unspent[0]["vout"]]["spentIndex"], 0) | ||
assert_equal(txVerbose["vout"][unspent[0]["vout"]]["spentHeight"], 106) | ||
|
||
print "Passed\n" | ||
|
||
|
||
if __name__ == '__main__': | ||
SpentIndexTest().main() |
Oops, something went wrong.