2019-12-04 07:29:16 -08:00
|
|
|
#!/usr/bin/env python3
|
2018-02-22 23:27:38 -08:00
|
|
|
# Copyright (c) 2018 The Zcash developers
|
|
|
|
# Distributed under the MIT software license, see the accompanying
|
2019-07-18 07:16:09 -07:00
|
|
|
# file COPYING or https://www.opensource.org/licenses/mit-license.php .
|
2018-02-22 23:27:38 -08:00
|
|
|
|
|
|
|
#
|
2019-07-19 05:10:13 -07:00
|
|
|
# Test proper expiry for transactions >= version 4
|
2018-02-22 23:27:38 -08:00
|
|
|
#
|
|
|
|
|
2018-11-05 10:24:20 -08:00
|
|
|
from test_framework.authproxy import JSONRPCException
|
2018-02-22 23:27:38 -08:00
|
|
|
from test_framework.test_framework import BitcoinTestFramework
|
2018-03-12 07:15:25 -07:00
|
|
|
from test_framework.util import assert_equal, \
|
|
|
|
connect_nodes_bi, sync_blocks, start_nodes, \
|
2018-02-22 23:27:38 -08:00
|
|
|
wait_and_assert_operationid_status
|
|
|
|
|
|
|
|
from decimal import Decimal
|
|
|
|
|
2018-11-05 10:24:20 -08:00
|
|
|
TX_EXPIRING_SOON_THRESHOLD = 3
|
|
|
|
TX_EXPIRY_DELTA = 10
|
|
|
|
|
2018-02-22 23:27:38 -08:00
|
|
|
class MempoolTxExpiryTest(BitcoinTestFramework):
|
|
|
|
|
|
|
|
def setup_nodes(self):
|
2018-11-05 10:24:20 -08:00
|
|
|
return start_nodes(4, self.options.tmpdir,
|
|
|
|
[[
|
|
|
|
"-txexpirydelta=%d" % TX_EXPIRY_DELTA,
|
|
|
|
"-debug=mempool"
|
|
|
|
]] * 4)
|
2018-02-22 23:27:38 -08:00
|
|
|
|
|
|
|
# Test before, at, and after expiry block
|
|
|
|
# chain is at block height 199 when run_test executes
|
|
|
|
def run_test(self):
|
2018-03-08 10:05:40 -08:00
|
|
|
alice = self.nodes[0].getnewaddress()
|
2018-11-05 10:24:20 -08:00
|
|
|
z_alice = self.nodes[0].z_getnewaddress('sapling')
|
2018-02-22 23:27:38 -08:00
|
|
|
bob = self.nodes[2].getnewaddress()
|
2018-11-05 10:24:20 -08:00
|
|
|
z_bob = self.nodes[2].z_getnewaddress('sapling')
|
2018-03-12 07:15:25 -07:00
|
|
|
|
2019-12-04 07:29:16 -08:00
|
|
|
print("Splitting network...")
|
2018-03-08 10:05:40 -08:00
|
|
|
self.split_network()
|
|
|
|
|
2019-07-19 05:10:13 -07:00
|
|
|
# Test dependent txs
|
2018-03-08 10:05:40 -08:00
|
|
|
firstTx = self.nodes[0].sendtoaddress(alice, 0.1)
|
|
|
|
firstTxInfo = self.nodes[0].getrawtransaction(firstTx, 1)
|
2019-07-19 05:10:13 -07:00
|
|
|
assert_equal(firstTxInfo["version"], 4)
|
2018-11-05 10:24:20 -08:00
|
|
|
assert_equal(firstTxInfo["overwintered"], True)
|
|
|
|
assert("expiryheight" in firstTxInfo)
|
2019-12-04 07:29:16 -08:00
|
|
|
print("First tx expiry height:", firstTxInfo['expiryheight'])
|
2018-03-08 10:05:40 -08:00
|
|
|
# Mine first transaction
|
|
|
|
self.nodes[0].generate(1)
|
|
|
|
for outpoint in firstTxInfo['vout']:
|
|
|
|
if outpoint['value'] == Decimal('0.10000000'):
|
|
|
|
vout = outpoint
|
|
|
|
break
|
|
|
|
inputs = [{'txid': firstTx, 'vout': vout['n'], 'scriptPubKey': vout['scriptPubKey']['hex']}]
|
|
|
|
outputs = {alice: 0.1}
|
|
|
|
rawTx = self.nodes[0].createrawtransaction(inputs, outputs)
|
|
|
|
rawTxSigned = self.nodes[0].signrawtransaction(rawTx)
|
|
|
|
assert(rawTxSigned['complete'])
|
|
|
|
secondTx = self.nodes[0].sendrawtransaction(rawTxSigned['hex'])
|
|
|
|
secondTxInfo = self.nodes[0].getrawtransaction(secondTx, 1)
|
2019-12-04 07:29:16 -08:00
|
|
|
print("Second tx expiry height:", secondTxInfo['expiryheight'])
|
2018-03-08 10:05:40 -08:00
|
|
|
# Mine second, dependent transaction
|
|
|
|
self.nodes[0].generate(1)
|
2019-12-04 07:29:16 -08:00
|
|
|
print("Mine %d competing blocks on Node 2..." % (2 + TX_EXPIRY_DELTA))
|
2018-11-05 10:24:20 -08:00
|
|
|
blocks = self.nodes[2].generate(2 + TX_EXPIRY_DELTA)
|
2019-12-04 07:29:16 -08:00
|
|
|
print("Connect nodes to force a reorg")
|
2018-03-08 10:05:40 -08:00
|
|
|
connect_nodes_bi(self.nodes,0,2)
|
|
|
|
self.is_network_split = False
|
2019-12-04 07:29:16 -08:00
|
|
|
print("Syncing blocks")
|
2018-03-08 10:05:40 -08:00
|
|
|
sync_blocks(self.nodes)
|
2019-12-04 07:29:16 -08:00
|
|
|
print("Ensure that both txs are dropped from mempool of node 0")
|
|
|
|
print("Blockheight node 0:", self.nodes[0].getblockchaininfo()['blocks'])
|
|
|
|
print("Blockheight node 2:", self.nodes[2].getblockchaininfo()['blocks'])
|
2018-03-08 10:05:40 -08:00
|
|
|
assert_equal(set(self.nodes[0].getrawmempool()), set())
|
|
|
|
assert_equal(set(self.nodes[2].getrawmempool()), set())
|
|
|
|
|
2018-02-22 23:27:38 -08:00
|
|
|
## Shield one of Alice's coinbase funds to her zaddr
|
|
|
|
res = self.nodes[0].z_shieldcoinbase("*", z_alice, 0.0001, 1)
|
|
|
|
wait_and_assert_operationid_status(self.nodes[0], res['opid'])
|
|
|
|
self.nodes[0].generate(1)
|
|
|
|
self.sync_all()
|
|
|
|
|
|
|
|
# Get balance on node 0
|
|
|
|
bal = self.nodes[0].z_gettotalbalance()
|
2019-12-04 07:29:16 -08:00
|
|
|
print("Balance before zsend, after shielding 10: ", bal)
|
2018-02-22 23:27:38 -08:00
|
|
|
assert_equal(Decimal(bal["private"]), Decimal("9.9999"))
|
|
|
|
|
2019-12-04 07:29:16 -08:00
|
|
|
print("Splitting network...")
|
2018-02-22 23:27:38 -08:00
|
|
|
self.split_network()
|
|
|
|
|
|
|
|
# Create transactions
|
2018-03-08 10:05:40 -08:00
|
|
|
blockheight = self.nodes[0].getblockchaininfo()['blocks']
|
2018-02-22 23:27:38 -08:00
|
|
|
zsendamount = Decimal('1.0') - Decimal('0.0001')
|
|
|
|
recipients = []
|
|
|
|
recipients.append({"address": z_bob, "amount": zsendamount})
|
|
|
|
myopid = self.nodes[0].z_sendmany(z_alice, recipients)
|
|
|
|
persist_shielded = wait_and_assert_operationid_status(self.nodes[0], myopid)
|
|
|
|
persist_transparent = self.nodes[0].sendtoaddress(bob, 0.01)
|
2018-11-05 10:24:20 -08:00
|
|
|
# Verify transparent transaction is version 4 intended for Sapling branch
|
2018-02-22 23:27:38 -08:00
|
|
|
rawtx = self.nodes[0].getrawtransaction(persist_transparent, 1)
|
2018-11-05 10:24:20 -08:00
|
|
|
assert_equal(rawtx["version"], 4)
|
2018-02-22 23:27:38 -08:00
|
|
|
assert_equal(rawtx["overwintered"], True)
|
2018-11-05 10:24:20 -08:00
|
|
|
assert_equal(rawtx["expiryheight"], blockheight + 1 + TX_EXPIRY_DELTA)
|
2019-12-04 07:29:16 -08:00
|
|
|
print("Blockheight at persist_transparent & persist_shielded creation:", self.nodes[0].getblockchaininfo()['blocks'])
|
|
|
|
print("Expiryheight of persist_transparent:", rawtx['expiryheight'])
|
2018-11-05 10:24:20 -08:00
|
|
|
# Verify shielded transaction is version 4 intended for Sapling branch
|
2018-02-22 23:27:38 -08:00
|
|
|
rawtx = self.nodes[0].getrawtransaction(persist_shielded, 1)
|
2019-12-04 07:29:16 -08:00
|
|
|
print("Expiryheight of persist_shielded", rawtx['expiryheight'])
|
2018-11-05 10:24:20 -08:00
|
|
|
assert_equal(rawtx["version"], 4)
|
2018-02-22 23:27:38 -08:00
|
|
|
assert_equal(rawtx["overwintered"], True)
|
2018-11-05 10:24:20 -08:00
|
|
|
assert_equal(rawtx["expiryheight"], blockheight + 1 + TX_EXPIRY_DELTA)
|
2018-02-22 23:27:38 -08:00
|
|
|
|
2020-03-07 04:26:25 -08:00
|
|
|
print("\nBlockheight advances to less than expiry block height. After reorg, txs should persist in mempool")
|
2018-02-22 23:27:38 -08:00
|
|
|
assert(persist_transparent in self.nodes[0].getrawmempool())
|
|
|
|
assert(persist_shielded in self.nodes[0].getrawmempool())
|
|
|
|
assert_equal(set(self.nodes[2].getrawmempool()), set())
|
2019-12-04 07:29:16 -08:00
|
|
|
print("mempool node 0:", self.nodes[0].getrawmempool())
|
|
|
|
print("mempool node 2:", self.nodes[2].getrawmempool())
|
2018-02-22 23:27:38 -08:00
|
|
|
bal = self.nodes[0].z_gettotalbalance()
|
2019-12-04 07:29:16 -08:00
|
|
|
print("Printing balance before persist_shielded & persist_transparent are initially mined from mempool", bal)
|
2018-02-22 23:27:38 -08:00
|
|
|
# Txs are mined on node 0; will later be rolled back
|
|
|
|
self.nodes[0].generate(1)
|
2019-12-04 07:29:16 -08:00
|
|
|
print("Node 0 generated 1 block")
|
|
|
|
print("Node 0 height:", self.nodes[0].getblockchaininfo()['blocks'])
|
|
|
|
print("Node 2 height:", self.nodes[2].getblockchaininfo()['blocks'])
|
2018-02-22 23:27:38 -08:00
|
|
|
bal = self.nodes[0].z_gettotalbalance()
|
2019-12-04 07:29:16 -08:00
|
|
|
print("Printing balance after persist_shielded & persist_transparent are mined:", bal)
|
2018-02-22 23:27:38 -08:00
|
|
|
assert_equal(set(self.nodes[0].getrawmempool()), set())
|
|
|
|
|
2019-12-04 07:29:16 -08:00
|
|
|
print("Mine 2 competing blocks on Node 2...")
|
2018-02-22 23:27:38 -08:00
|
|
|
blocks = self.nodes[2].generate(2)
|
|
|
|
for block in blocks:
|
|
|
|
blk = self.nodes[2].getblock(block)
|
2019-12-04 07:29:16 -08:00
|
|
|
print("Height: {0}, Mined block txs: {1}".format(blk["height"], blk["tx"]))
|
|
|
|
print("Connect nodes to force a reorg")
|
2018-02-22 23:27:38 -08:00
|
|
|
connect_nodes_bi(self.nodes,0,2)
|
|
|
|
self.is_network_split = False
|
|
|
|
|
2019-12-04 07:29:16 -08:00
|
|
|
print("Syncing blocks")
|
2018-02-22 23:27:38 -08:00
|
|
|
sync_blocks(self.nodes)
|
|
|
|
|
2019-12-04 07:29:16 -08:00
|
|
|
print("Ensure that txs are back in mempool of node 0")
|
|
|
|
print("Blockheight node 0:", self.nodes[0].getblockchaininfo()['blocks'])
|
|
|
|
print("Blockheight node 2:", self.nodes[2].getblockchaininfo()['blocks'])
|
|
|
|
print("mempool node 0: ", self.nodes[0].getrawmempool())
|
|
|
|
print("mempool node 2: ", self.nodes[2].getrawmempool())
|
2018-02-22 23:27:38 -08:00
|
|
|
assert(persist_transparent in self.nodes[0].getrawmempool())
|
|
|
|
assert(persist_shielded in self.nodes[0].getrawmempool())
|
|
|
|
bal = self.nodes[0].z_gettotalbalance()
|
|
|
|
# Mine txs to get them out of the way of mempool sync in split_network()
|
2019-12-04 07:29:16 -08:00
|
|
|
print("Generating another block on node 0 to clear txs from mempool")
|
2018-02-22 23:27:38 -08:00
|
|
|
self.nodes[0].generate(1)
|
|
|
|
assert_equal(set(self.nodes[0].getrawmempool()), set())
|
|
|
|
sync_blocks(self.nodes)
|
|
|
|
|
2019-12-04 07:29:16 -08:00
|
|
|
print("Splitting network...")
|
2018-02-22 23:27:38 -08:00
|
|
|
self.split_network()
|
|
|
|
|
2020-03-07 04:26:25 -08:00
|
|
|
print("\nBlockheight advances to equal expiry block height. After reorg, txs should persist in mempool")
|
2018-02-22 23:27:38 -08:00
|
|
|
myopid = self.nodes[0].z_sendmany(z_alice, recipients)
|
|
|
|
persist_shielded_2 = wait_and_assert_operationid_status(self.nodes[0], myopid)
|
|
|
|
persist_transparent_2 = self.nodes[0].sendtoaddress(bob, 0.01)
|
|
|
|
rawtx_trans = self.nodes[0].getrawtransaction(persist_transparent_2, 1)
|
|
|
|
rawtx_shield = self.nodes[0].getrawtransaction(persist_shielded_2, 1)
|
2019-12-04 07:29:16 -08:00
|
|
|
print("Blockheight node 0 at persist_transparent_2 creation:", self.nodes[0].getblockchaininfo()['blocks'])
|
|
|
|
print("Blockheight node 2 at persist_transparent_2 creation:", self.nodes[2].getblockchaininfo()['blocks'])
|
|
|
|
print("Expiryheight of persist_transparent_2:", rawtx_trans['expiryheight'])
|
|
|
|
print("Expiryheight of persist_shielded_2:", rawtx_shield['expiryheight'])
|
2018-02-22 23:27:38 -08:00
|
|
|
blocks = self.nodes[2].generate(4)
|
|
|
|
for block in blocks:
|
|
|
|
blk = self.nodes[2].getblock(block)
|
2019-12-04 07:29:16 -08:00
|
|
|
print("Height: {0}, Mined block txs: {1}".format(blk["height"], blk["tx"]))
|
|
|
|
print("Connect nodes to force a reorg")
|
2018-02-22 23:27:38 -08:00
|
|
|
connect_nodes_bi(self.nodes, 0, 2)
|
|
|
|
self.is_network_split = False
|
|
|
|
sync_blocks(self.nodes)
|
2019-12-04 07:29:16 -08:00
|
|
|
print("Ensure that persist_transparent_2 & persist_shielded_2 are in mempool at expiry block height")
|
|
|
|
print("Blockheight node 0:", self.nodes[0].getblockchaininfo()['blocks'])
|
|
|
|
print("Blockheight node 2:", self.nodes[2].getblockchaininfo()['blocks'])
|
|
|
|
print("mempool node 0: ", self.nodes[0].getrawmempool())
|
|
|
|
print("mempool node 2: ", self.nodes[2].getrawmempool())
|
2018-02-22 23:27:38 -08:00
|
|
|
assert(persist_transparent_2 in self.nodes[0].getrawmempool())
|
|
|
|
assert(persist_shielded_2 in self.nodes[0].getrawmempool())
|
|
|
|
# Mine persist txs to get them out of the way of mempool sync in split_network()
|
|
|
|
self.nodes[0].generate(1)
|
|
|
|
assert_equal(set(self.nodes[0].getrawmempool()), set())
|
|
|
|
sync_blocks(self.nodes)
|
2019-12-04 07:29:16 -08:00
|
|
|
print("Balance after persist_shielded_2 is mined to remove from mempool: ", self.nodes[0].z_gettotalbalance())
|
2018-02-22 23:27:38 -08:00
|
|
|
|
2019-12-04 07:29:16 -08:00
|
|
|
print("Splitting network...")
|
2018-02-22 23:27:38 -08:00
|
|
|
self.split_network()
|
|
|
|
|
2020-03-07 04:26:25 -08:00
|
|
|
print("\nBlockheight advances to greater than expiry block height. After reorg, txs should expire from mempool")
|
2019-12-04 07:29:16 -08:00
|
|
|
print("Balance before expire_shielded is sent: ", self.nodes[0].z_gettotalbalance())
|
2018-02-22 23:27:38 -08:00
|
|
|
myopid = self.nodes[0].z_sendmany(z_alice, recipients)
|
|
|
|
expire_shielded = wait_and_assert_operationid_status(self.nodes[0], myopid)
|
|
|
|
expire_transparent = self.nodes[0].sendtoaddress(bob, 0.01)
|
2019-12-04 07:29:16 -08:00
|
|
|
print("Blockheight node 0 at expire_transparent creation:", self.nodes[0].getblockchaininfo()['blocks'])
|
|
|
|
print("Blockheight node 2 at expire_shielded creation:", self.nodes[2].getblockchaininfo()['blocks'])
|
|
|
|
print("Expiryheight of expire_transparent:", self.nodes[0].getrawtransaction(expire_transparent, 1)['expiryheight'])
|
|
|
|
print("Expiryheight of expire_shielded:", self.nodes[0].getrawtransaction(expire_shielded, 1)['expiryheight'])
|
2018-02-22 23:27:38 -08:00
|
|
|
assert(expire_transparent in self.nodes[0].getrawmempool())
|
|
|
|
assert(expire_shielded in self.nodes[0].getrawmempool())
|
2018-11-05 10:24:20 -08:00
|
|
|
blocks = self.nodes[2].generate(1 + TX_EXPIRY_DELTA + 1)
|
2018-02-22 23:27:38 -08:00
|
|
|
for block in blocks:
|
|
|
|
blk = self.nodes[2].getblock(block)
|
2019-12-04 07:29:16 -08:00
|
|
|
print("Height: {0}, Mined block txs: {1}".format(blk["height"], blk["tx"]))
|
|
|
|
print("Connect nodes to force a reorg")
|
2018-02-22 23:27:38 -08:00
|
|
|
connect_nodes_bi(self.nodes, 0, 2)
|
|
|
|
self.is_network_split = False
|
|
|
|
sync_blocks(self.nodes)
|
2019-12-04 07:29:16 -08:00
|
|
|
print("Ensure that expire_transparent & expire_shielded are not in mempool after expiry block height")
|
|
|
|
print("mempool node 0: ", self.nodes[0].getrawmempool())
|
|
|
|
print("mempool node 2: ", self.nodes[2].getrawmempool())
|
2018-02-22 23:27:38 -08:00
|
|
|
assert_equal(set(self.nodes[0].getrawmempool()), set())
|
2019-12-04 07:29:16 -08:00
|
|
|
print("Ensure balance of node 0 is correct")
|
2018-02-22 23:27:38 -08:00
|
|
|
bal = self.nodes[0].z_gettotalbalance()
|
2019-12-04 07:29:16 -08:00
|
|
|
print("Balance after expire_shielded has expired: ", bal)
|
2018-02-22 23:27:38 -08:00
|
|
|
assert_equal(Decimal(bal["private"]), Decimal("7.9999"))
|
|
|
|
|
2019-12-04 07:29:16 -08:00
|
|
|
print("Splitting network...")
|
2018-11-05 10:24:20 -08:00
|
|
|
self.split_network()
|
|
|
|
|
2020-03-07 04:26:25 -08:00
|
|
|
print("\nBlockheight advances to just before expiring soon threshold. Txs should be rejected from entering mempool.")
|
2019-12-04 07:29:16 -08:00
|
|
|
print("Balance before expire_shielded is sent: ", self.nodes[0].z_gettotalbalance())
|
2018-11-05 10:24:20 -08:00
|
|
|
myopid = self.nodes[0].z_sendmany(z_alice, recipients)
|
|
|
|
expire_shielded = wait_and_assert_operationid_status(self.nodes[0], myopid)
|
|
|
|
expire_transparent = self.nodes[0].sendtoaddress(bob, 0.01)
|
2019-12-04 07:29:16 -08:00
|
|
|
print("Blockheight node 0 at expire_transparent creation:", self.nodes[0].getblockchaininfo()['blocks'])
|
|
|
|
print("Blockheight node 2 at expire_shielded creation:", self.nodes[2].getblockchaininfo()['blocks'])
|
|
|
|
print("Expiryheight of expire_transparent:", self.nodes[0].getrawtransaction(expire_transparent, 1)['expiryheight'])
|
|
|
|
print("Expiryheight of expire_shielded:", self.nodes[0].getrawtransaction(expire_shielded, 1)['expiryheight'])
|
2018-11-05 10:24:20 -08:00
|
|
|
assert(expire_transparent in self.nodes[0].getrawmempool())
|
|
|
|
assert(expire_shielded in self.nodes[0].getrawmempool())
|
|
|
|
blocks = self.nodes[2].generate(1 + TX_EXPIRY_DELTA - TX_EXPIRING_SOON_THRESHOLD - 1)
|
|
|
|
for block in blocks:
|
|
|
|
blk = self.nodes[2].getblock(block)
|
2019-12-04 07:29:16 -08:00
|
|
|
print("Height: {0}, Mined block txs: {1}".format(blk["height"], blk["tx"]))
|
|
|
|
print("Connect nodes to force a reorg")
|
2018-11-05 10:24:20 -08:00
|
|
|
connect_nodes_bi(self.nodes, 0, 2)
|
|
|
|
self.is_network_split = False
|
|
|
|
sync_blocks(self.nodes)
|
2019-12-04 07:29:16 -08:00
|
|
|
print("Ensure that expire_transparent & expire_shielded are in node 0 mempool but not node 2 mempool")
|
|
|
|
print("mempool node 0: ", self.nodes[0].getrawmempool())
|
|
|
|
print("mempool node 2: ", self.nodes[2].getrawmempool())
|
2018-11-05 10:24:20 -08:00
|
|
|
assert(expire_transparent in self.nodes[0].getrawmempool())
|
|
|
|
assert(expire_shielded in self.nodes[0].getrawmempool())
|
|
|
|
assert(expire_transparent not in self.nodes[2].getrawmempool())
|
|
|
|
assert(expire_shielded not in self.nodes[2].getrawmempool())
|
|
|
|
|
|
|
|
# Now try to add the transactions to Node 2 mempool.
|
|
|
|
# Node 2 mempool will accept a transaction since block height has not reached the expiring soon threshold.
|
|
|
|
rawtx = self.nodes[0].getrawtransaction(expire_transparent)
|
|
|
|
self.nodes[2].sendrawtransaction(rawtx)
|
|
|
|
|
|
|
|
# Generate a block and enter the expiring soon threshold.
|
|
|
|
self.nodes[2].generate(1)
|
|
|
|
# Node 2 mempool will reject a transaction which is expiring soon.
|
|
|
|
try:
|
|
|
|
rawtx = self.nodes[0].getrawtransaction(expire_shielded)
|
|
|
|
self.nodes[2].sendrawtransaction(rawtx)
|
|
|
|
assert(False)
|
|
|
|
except JSONRPCException as e:
|
|
|
|
errorString = e.error['message']
|
|
|
|
assert("tx-expiring-soon" in errorString)
|
2018-02-22 23:27:38 -08:00
|
|
|
|
|
|
|
if __name__ == '__main__':
|
|
|
|
MempoolTxExpiryTest().main()
|