Page MenuHomePhabricator

No OneTemporary

diff --git a/test/functional/feature_csv_activation.py b/test/functional/feature_csv_activation.py
index 3869718437..dc6267084b 100755
--- a/test/functional/feature_csv_activation.py
+++ b/test/functional/feature_csv_activation.py
@@ -1,638 +1,651 @@
#!/usr/bin/env python3
# Copyright (c) 2015-2016 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 activation of the first version bits soft fork.
This soft fork will activate the following BIPS:
BIP 68 - nSequence relative lock times
BIP 112 - CHECKSEQUENCEVERIFY
BIP 113 - MedianTimePast semantics for nLockTime
regtest lock-in with 108/144 block signalling
activation after a further 144 blocks
mine 82 blocks whose coinbases will be used to generate inputs for our tests
mine 489 blocks and seed block chain with the 82 inputs will use for our tests at height 572
mine 3 blocks and verify still at LOCKED_IN and test that enforcement has not triggered
mine 1 block and test that enforcement has triggered (which triggers ACTIVE)
Test BIP 113 is enforced
Mine 4 blocks so next height is 580 and test BIP 68 is enforced for time and height
Mine 1 block so next height is 581 and test BIP 68 now passes time but not height
Mine 1 block so next height is 582 and test BIP 68 now passes time and height
Test that BIP 112 is enforced
Various transactions will be used to test that the BIPs rules are not enforced before the soft fork activates
And that after the soft fork activates transactions pass and fail as they should according to the rules.
For each BIP, transactions of versions 1 and 2 will be tested.
----------------
BIP 113:
bip113tx - modify the nLocktime variable
BIP 68:
bip68txs - 16 txs with nSequence relative locktime of 10 with various bits set as per the relative_locktimes below
BIP 112:
bip112txs_vary_nSequence - 16 txs with nSequence relative_locktimes of 10 evaluated against 10 OP_CSV OP_DROP
bip112txs_vary_nSequence_9 - 16 txs with nSequence relative_locktimes of 9 evaluated against 10 OP_CSV OP_DROP
bip112txs_vary_OP_CSV - 16 txs with nSequence = 10 evaluated against varying {relative_locktimes of 10} OP_CSV OP_DROP
bip112txs_vary_OP_CSV_9 - 16 txs with nSequence = 9 evaluated against varying {relative_locktimes of 10} OP_CSV OP_DROP
bip112tx_special - test negative argument to OP_CSV
"""
from decimal import Decimal
from itertools import product
import time
from test_framework.blocktools import (
create_block,
create_coinbase,
make_conform_to_ctor,
)
from test_framework.comptool import TestInstance, TestManager
from test_framework.messages import COIN, CTransaction, FromHex, ToHex
from test_framework.mininode import network_thread_start
from test_framework.script import (
CScript,
OP_CHECKSEQUENCEVERIFY,
OP_DROP,
OP_TRUE,
)
from test_framework.test_framework import ComparisonTestFramework
from test_framework.txtools import pad_tx
from test_framework.util import assert_equal
BASE_RELATIVE_LOCKTIME = 10
SEQ_DISABLE_FLAG = 1 << 31
SEQ_RANDOM_HIGH_BIT = 1 << 25
SEQ_TYPE_FLAG = 1 << 22
SEQ_RANDOM_LOW_BIT = 1 << 18
def relative_locktime(sdf, srhb, stf, srlb):
"""Returns a locktime with certain bits set."""
locktime = BASE_RELATIVE_LOCKTIME
if sdf:
locktime |= SEQ_DISABLE_FLAG
if srhb:
locktime |= SEQ_RANDOM_HIGH_BIT
if stf:
locktime |= SEQ_TYPE_FLAG
if srlb:
locktime |= SEQ_RANDOM_LOW_BIT
return locktime
def all_rlt_txs(txs):
return [tx['tx'] for tx in txs]
def get_csv_status(node):
height = node.getblockchaininfo()['blocks']
return height >= 576
+def create_transaction(node, txid, to_address, amount):
+ inputs = [{"txid": txid, "vout": 0}]
+ outputs = {to_address: amount}
+ rawtx = node.createrawtransaction(inputs, outputs)
+ tx = FromHex(CTransaction(), rawtx)
+ return tx
+
+
+def sign_transaction(node, unsignedtx):
+ rawtx = ToHex(unsignedtx)
+ signresult = node.signrawtransactionwithwallet(rawtx)
+ tx = FromHex(CTransaction(), signresult['hex'])
+ return tx
+
+
+def spend_tx(node, prev_tx, address):
+ spendtx = create_transaction(
+ node, prev_tx.hash, address, (prev_tx.vout[0].nValue - 1000) / COIN)
+ spendtx.nVersion = prev_tx.nVersion
+ pad_tx(spendtx)
+ spendtx.rehash()
+ return spendtx
+
+
+def create_bip112special(node, input, txversion, address):
+ tx = create_transaction(
+ node, input, address, Decimal("49.98"))
+ tx.nVersion = txversion
+ tx.vout[0].scriptPubKey = CScript(
+ [-1, OP_CHECKSEQUENCEVERIFY, OP_DROP, OP_TRUE])
+ tx.rehash()
+ signtx = sign_transaction(node, tx)
+ signtx.rehash()
+
+ return signtx
+
+
+def send_generic_input_tx(node, coinbases, address):
+ amount = Decimal("49.99")
+ return node.sendrawtransaction(ToHex(sign_transaction(node, create_transaction(node, node.getblock(coinbases.pop())['tx'][0], address, amount))))
+
+
+def create_bip68txs(node, bip68inputs, txversion, address, locktime_delta=0):
+ """Returns a list of bip68 transactions with different bits set."""
+ txs = []
+ assert(len(bip68inputs) >= 16)
+ for i, (sdf, srhb, stf, srlb) in enumerate(product(*[[True, False]] * 4)):
+ locktime = relative_locktime(sdf, srhb, stf, srlb)
+ tx = create_transaction(
+ node, bip68inputs[i], address, Decimal("49.98"))
+ tx.nVersion = txversion
+ tx.vin[0].nSequence = locktime + locktime_delta
+ tx = sign_transaction(node, tx)
+ tx.rehash()
+ txs.append({'tx': tx, 'sdf': sdf, 'stf': stf})
+
+ return txs
+
+
+def create_bip112txs(node, bip112inputs, varyOP_CSV, txversion, address, locktime_delta=0):
+ """Returns a list of bip112 transactions with different bits set."""
+ txs = []
+ assert(len(bip112inputs) >= 16)
+ for i, (sdf, srhb, stf, srlb) in enumerate(product(*[[True, False]] * 4)):
+ locktime = relative_locktime(sdf, srhb, stf, srlb)
+ tx = create_transaction(
+ node, bip112inputs[i], address, Decimal("49.98"))
+ if (varyOP_CSV): # if varying OP_CSV, nSequence is fixed
+ tx.vin[0].nSequence = BASE_RELATIVE_LOCKTIME + locktime_delta
+ else: # vary nSequence instead, OP_CSV is fixed
+ tx.vin[0].nSequence = locktime + locktime_delta
+ tx.nVersion = txversion
+ if (varyOP_CSV):
+ tx.vout[0].scriptPubKey = CScript(
+ [locktime, OP_CHECKSEQUENCEVERIFY, OP_DROP, OP_TRUE])
+ else:
+ tx.vout[0].scriptPubKey = CScript(
+ [BASE_RELATIVE_LOCKTIME, OP_CHECKSEQUENCEVERIFY, OP_DROP, OP_TRUE])
+ tx.rehash()
+ signtx = sign_transaction(node, tx)
+ signtx.rehash()
+ txs.append({'tx': signtx, 'sdf': sdf, 'stf': stf})
+ return txs
+
+
class BIP68_112_113Test(ComparisonTestFramework):
def set_test_params(self):
self.num_nodes = 1
self.setup_clean_chain = True
self.extra_args = [['-whitelist=127.0.0.1', '-blockversion=4']]
def run_test(self):
test = TestManager(self, self.options.tmpdir)
test.add_all_connections(self.nodes)
network_thread_start()
test.run()
- def send_generic_input_tx(self, node, coinbases):
- amount = Decimal("49.99")
- return node.sendrawtransaction(ToHex(self.sign_transaction(node, self.create_transaction(node, node.getblock(coinbases.pop())['tx'][0], self.nodeaddress, amount))))
-
- def create_transaction(self, node, txid, to_address, amount):
- inputs = [{"txid": txid, "vout": 0}]
- outputs = {to_address: amount}
- rawtx = node.createrawtransaction(inputs, outputs)
- tx = FromHex(CTransaction(), rawtx)
- return tx
-
- def sign_transaction(self, node, unsignedtx):
- rawtx = ToHex(unsignedtx)
- signresult = node.signrawtransactionwithwallet(rawtx)
- tx = FromHex(CTransaction(), signresult['hex'])
- return tx
-
- def spend_tx(self, node, prev_tx):
- spendtx = self.create_transaction(
- node, prev_tx.hash, self.nodeaddress, (prev_tx.vout[0].nValue - 1000) / COIN)
- spendtx.nVersion = prev_tx.nVersion
- pad_tx(spendtx)
- spendtx.rehash()
- return spendtx
-
def generate_blocks(self, number):
test_blocks = []
for i in range(number):
block = self.create_test_block([])
test_blocks.append([block, True])
self.last_block_time += 600
self.tip = block.sha256
self.tipheight += 1
return test_blocks
def create_test_block(self, txs, version=536870912):
block = create_block(self.tip, create_coinbase(
self.tipheight + 1), self.last_block_time + 600)
block.nVersion = version
block.vtx.extend(txs)
make_conform_to_ctor(block)
block.hashMerkleRoot = block.calc_merkle_root()
block.rehash()
block.solve()
return block
# Create a block with given txs, and spend these txs in the same block.
# Spending utxos in the same block is OK as long as nSequence is not enforced.
# Otherwise a number of intermediate blocks should be generated, and this
# method should not be used.
def create_test_block_spend_utxos(self, node, txs, version=536870912):
block = self.create_test_block(txs, version)
- block.vtx.extend([self.spend_tx(node, tx) for tx in txs])
+ block.vtx.extend([spend_tx(node, tx, self.nodeaddress) for tx in txs])
make_conform_to_ctor(block)
block.hashMerkleRoot = block.calc_merkle_root()
block.rehash()
block.solve()
return block
- def create_bip68txs(self, bip68inputs, txversion, locktime_delta=0):
- """Returns a list of bip68 transactions with different bits set."""
- txs = []
- assert(len(bip68inputs) >= 16)
- for i, (sdf, srhb, stf, srlb) in enumerate(product(*[[True, False]] * 4)):
- locktime = relative_locktime(sdf, srhb, stf, srlb)
- tx = self.create_transaction(
- self.nodes[0], bip68inputs[i], self.nodeaddress, Decimal("49.98"))
- tx.nVersion = txversion
- tx.vin[0].nSequence = locktime + locktime_delta
- tx = self.sign_transaction(self.nodes[0], tx)
- tx.rehash()
- txs.append({'tx': tx, 'sdf': sdf, 'stf': stf})
-
- return txs
-
- def create_bip112special(self, input, txversion):
- tx = self.create_transaction(
- self.nodes[0], input, self.nodeaddress, Decimal("49.98"))
- tx.nVersion = txversion
- tx.vout[0].scriptPubKey = CScript(
- [-1, OP_CHECKSEQUENCEVERIFY, OP_DROP, OP_TRUE])
- tx.rehash()
- signtx = self.sign_transaction(self.nodes[0], tx)
- signtx.rehash()
-
- return signtx
-
- def create_bip112txs(self, bip112inputs, varyOP_CSV, txversion, locktime_delta=0):
- """Returns a list of bip112 transactions with different bits set."""
- txs = []
- assert(len(bip112inputs) >= 16)
- for i, (sdf, srhb, stf, srlb) in enumerate(product(*[[True, False]] * 4)):
- locktime = relative_locktime(sdf, srhb, stf, srlb)
- tx = self.create_transaction(
- self.nodes[0], bip112inputs[i], self.nodeaddress, Decimal("49.98"))
- if (varyOP_CSV): # if varying OP_CSV, nSequence is fixed
- tx.vin[0].nSequence = BASE_RELATIVE_LOCKTIME + locktime_delta
- else: # vary nSequence instead, OP_CSV is fixed
- tx.vin[0].nSequence = locktime + locktime_delta
- tx.nVersion = txversion
- if (varyOP_CSV):
- tx.vout[0].scriptPubKey = CScript(
- [locktime, OP_CHECKSEQUENCEVERIFY, OP_DROP, OP_TRUE])
- else:
- tx.vout[0].scriptPubKey = CScript(
- [BASE_RELATIVE_LOCKTIME, OP_CHECKSEQUENCEVERIFY, OP_DROP, OP_TRUE])
- tx.rehash()
- signtx = self.sign_transaction(self.nodes[0], tx)
- signtx.rehash()
- txs.append({'tx': signtx, 'sdf': sdf, 'stf': stf})
- return txs
-
def get_tests(self):
self.log.info("Generate blocks in the past for coinbase outputs.")
# Enough to build up to 1000 blocks 10 minutes apart without worrying
# about getting into the future
long_past_time = int(time.time()) - 600 * 1000
# Enough so that the generated blocks will still all be before long_past_time
self.nodes[0].setmocktime(long_past_time - 100)
# 82 blocks generated for inputs
self.coinbase_blocks = self.nodes[0].generate(1 + 16 + 2 * 32 + 1)
# Set time back to present so yielded blocks aren't in the future as
# we advance last_block_time
self.nodes[0].setmocktime(0)
# height of the next block to build
self.tipheight = 82
self.last_block_time = long_past_time
self.tip = int("0x" + self.nodes[0].getbestblockhash(), 0)
self.nodeaddress = self.nodes[0].getnewaddress()
# CSV is not activated yet.
assert_equal(get_csv_status(self.nodes[0]), False)
# Generate 489 more version 4 blocks
test_blocks = self.generate_blocks(489)
# Test #1
yield TestInstance(test_blocks, sync_every_block=False)
# Still not activated.
assert_equal(get_csv_status(self.nodes[0]), False)
# Inputs at height = 572
# Put inputs for all tests in the chain at height 572 (tip now = 571) (time increases by 600s per block)
# Note we reuse inputs for v1 and v2 txs so must test these separately
# 16 normal inputs
bip68inputs = []
for i in range(16):
- bip68inputs.append(self.send_generic_input_tx(
- self.nodes[0], self.coinbase_blocks))
+ bip68inputs.append(send_generic_input_tx(
+ self.nodes[0], self.coinbase_blocks, self.nodeaddress))
# 2 sets of 16 inputs with 10 OP_CSV OP_DROP (actually will be prepended to spending scriptSig)
bip112basicinputs = []
for j in range(2):
inputs = []
for i in range(16):
- inputs.append(self.send_generic_input_tx(
- self.nodes[0], self.coinbase_blocks))
+ inputs.append(send_generic_input_tx(
+ self.nodes[0], self.coinbase_blocks, self.nodeaddress))
bip112basicinputs.append(inputs)
# 2 sets of 16 varied inputs with (relative_lock_time) OP_CSV OP_DROP (actually will be prepended to spending scriptSig)
bip112diverseinputs = []
for j in range(2):
inputs = []
for i in range(16):
- inputs.append(self.send_generic_input_tx(
- self.nodes[0], self.coinbase_blocks))
+ inputs.append(send_generic_input_tx(
+ self.nodes[0], self.coinbase_blocks, self.nodeaddress))
bip112diverseinputs.append(inputs)
# 1 special input with -1 OP_CSV OP_DROP (actually will be prepended to spending scriptSig)
- bip112specialinput = self.send_generic_input_tx(
- self.nodes[0], self.coinbase_blocks)
+ bip112specialinput = send_generic_input_tx(
+ self.nodes[0], self.coinbase_blocks, self.nodeaddress)
# 1 normal input
- bip113input = self.send_generic_input_tx(
- self.nodes[0], self.coinbase_blocks)
+ bip113input = send_generic_input_tx(
+ self.nodes[0], self.coinbase_blocks, self.nodeaddress)
self.nodes[0].setmocktime(self.last_block_time + 600)
# 1 block generated for inputs to be in chain at height 572
inputblockhash = self.nodes[0].generate(1)[0]
self.nodes[0].setmocktime(0)
self.tip = int("0x" + inputblockhash, 0)
self.tipheight += 1
self.last_block_time += 600
assert_equal(len(self.nodes[0].getblock(
inputblockhash, True)["tx"]), 82 + 1)
# 2 more version 4 blocks
test_blocks = self.generate_blocks(2)
# Test #2
yield TestInstance(test_blocks, sync_every_block=False)
self.log.info(
"Not yet activated, height = 574 (will activate for block 576, not 575)")
assert_equal(get_csv_status(self.nodes[0]), False)
# Test both version 1 and version 2 transactions for all tests
# BIP113 test transaction will be modified before each use to
# put in appropriate block time
- bip113tx_v1 = self.create_transaction(
+ bip113tx_v1 = create_transaction(
self.nodes[0], bip113input, self.nodeaddress, Decimal("49.98"))
bip113tx_v1.vin[0].nSequence = 0xFFFFFFFE
bip113tx_v1.nVersion = 1
- bip113tx_v2 = self.create_transaction(
+ bip113tx_v2 = create_transaction(
self.nodes[0], bip113input, self.nodeaddress, Decimal("49.98"))
bip113tx_v2.vin[0].nSequence = 0xFFFFFFFE
bip113tx_v2.nVersion = 2
# For BIP68 test all 16 relative sequence locktimes
- bip68txs_v1 = self.create_bip68txs(bip68inputs, 1)
- bip68txs_v2 = self.create_bip68txs(bip68inputs, 2)
+ bip68txs_v1 = create_bip68txs(
+ self.nodes[0], bip68inputs, 1, self.nodeaddress)
+ bip68txs_v2 = create_bip68txs(
+ self.nodes[0], bip68inputs, 2, self.nodeaddress)
# For BIP112 test:
# 16 relative sequence locktimes of 10 against 10 OP_CSV OP_DROP inputs
- bip112txs_vary_nSequence_v1 = self.create_bip112txs(
- bip112basicinputs[0], False, 1)
- bip112txs_vary_nSequence_v2 = self.create_bip112txs(
- bip112basicinputs[0], False, 2)
+ bip112txs_vary_nSequence_v1 = create_bip112txs(
+ self.nodes[0], bip112basicinputs[0], False, 1, self.nodeaddress)
+ bip112txs_vary_nSequence_v2 = create_bip112txs(
+ self.nodes[0], bip112basicinputs[0], False, 2, self.nodeaddress)
# 16 relative sequence locktimes of 9 against 10 OP_CSV OP_DROP inputs
- bip112txs_vary_nSequence_9_v1 = self.create_bip112txs(
- bip112basicinputs[1], False, 1, -1)
- bip112txs_vary_nSequence_9_v2 = self.create_bip112txs(
- bip112basicinputs[1], False, 2, -1)
+ bip112txs_vary_nSequence_9_v1 = create_bip112txs(
+ self.nodes[0], bip112basicinputs[1], False, 1, self.nodeaddress, -1)
+ bip112txs_vary_nSequence_9_v2 = create_bip112txs(
+ self.nodes[0], bip112basicinputs[1], False, 2, self.nodeaddress, -1)
# sequence lock time of 10 against 16 (relative_lock_time) OP_CSV OP_DROP inputs
- bip112txs_vary_OP_CSV_v1 = self.create_bip112txs(
- bip112diverseinputs[0], True, 1)
- bip112txs_vary_OP_CSV_v2 = self.create_bip112txs(
- bip112diverseinputs[0], True, 2)
+ bip112txs_vary_OP_CSV_v1 = create_bip112txs(
+ self.nodes[0], bip112diverseinputs[0], True, 1, self.nodeaddress)
+ bip112txs_vary_OP_CSV_v2 = create_bip112txs(
+ self.nodes[0], bip112diverseinputs[0], True, 2, self.nodeaddress)
# sequence lock time of 9 against 16 (relative_lock_time) OP_CSV OP_DROP inputs
- bip112txs_vary_OP_CSV_9_v1 = self.create_bip112txs(
- bip112diverseinputs[1], True, 1, -1)
- bip112txs_vary_OP_CSV_9_v2 = self.create_bip112txs(
- bip112diverseinputs[1], True, 2, -1)
+ bip112txs_vary_OP_CSV_9_v1 = create_bip112txs(
+ self.nodes[0], bip112diverseinputs[1], True, 1, self.nodeaddress, -1)
+ bip112txs_vary_OP_CSV_9_v2 = create_bip112txs(
+ self.nodes[0], bip112diverseinputs[1], True, 2, self.nodeaddress, -1)
# -1 OP_CSV OP_DROP input
- bip112tx_special_v1 = self.create_bip112special(bip112specialinput, 1)
- bip112tx_special_v2 = self.create_bip112special(bip112specialinput, 2)
+ bip112tx_special_v1 = create_bip112special(
+ self.nodes[0], bip112specialinput, 1, self.nodeaddress)
+ bip112tx_special_v2 = create_bip112special(
+ self.nodes[0], bip112specialinput, 2, self.nodeaddress)
self.log.info("TESTING")
self.log.info("Pre-Soft Fork Tests. All txs should pass.")
self.log.info("Test version 1 txs")
success_txs = []
# add BIP113 tx and -1 CSV tx
# = MTP of prior block (not <) but < time put on current block
bip113tx_v1.nLockTime = self.last_block_time - 600 * 5
- bip113signed1 = self.sign_transaction(self.nodes[0], bip113tx_v1)
+ bip113signed1 = sign_transaction(self.nodes[0], bip113tx_v1)
success_txs.append(bip113signed1)
success_txs.append(bip112tx_special_v1)
- success_txs.append(self.spend_tx(self.nodes[0], bip112tx_special_v1))
+ success_txs.append(
+ spend_tx(self.nodes[0], bip112tx_special_v1, self.nodeaddress))
# add BIP 68 txs
success_txs.extend(all_rlt_txs(bip68txs_v1))
# add BIP 112 with seq=10 txs
success_txs.extend(all_rlt_txs(bip112txs_vary_nSequence_v1))
- success_txs.extend([self.spend_tx(self.nodes[0], tx)
+ success_txs.extend([spend_tx(self.nodes[0], tx, self.nodeaddress)
for tx in all_rlt_txs(bip112txs_vary_nSequence_v1)])
success_txs.extend(all_rlt_txs(bip112txs_vary_OP_CSV_v1))
- success_txs.extend([self.spend_tx(self.nodes[0], tx)
+ success_txs.extend([spend_tx(self.nodes[0], tx, self.nodeaddress)
for tx in all_rlt_txs(bip112txs_vary_OP_CSV_v1)])
# try BIP 112 with seq=9 txs
success_txs.extend(all_rlt_txs(bip112txs_vary_nSequence_9_v1))
- success_txs.extend([self.spend_tx(self.nodes[0], tx)
+ success_txs.extend([spend_tx(self.nodes[0], tx, self.nodeaddress)
for tx in all_rlt_txs(bip112txs_vary_nSequence_9_v1)])
success_txs.extend(all_rlt_txs(bip112txs_vary_OP_CSV_9_v1))
- success_txs.extend([self.spend_tx(self.nodes[0], tx)
+ success_txs.extend([spend_tx(self.nodes[0], tx, self.nodeaddress)
for tx in all_rlt_txs(bip112txs_vary_OP_CSV_9_v1)])
# Test #3
yield TestInstance([[self.create_test_block(success_txs), True]])
self.nodes[0].invalidateblock(self.nodes[0].getbestblockhash())
self.log.info("Test version 2 txs")
success_txs = []
# add BIP113 tx and -1 CSV tx
# = MTP of prior block (not <) but < time put on current block
bip113tx_v2.nLockTime = self.last_block_time - 600 * 5
- bip113signed2 = self.sign_transaction(self.nodes[0], bip113tx_v2)
+ bip113signed2 = sign_transaction(self.nodes[0], bip113tx_v2)
success_txs.append(bip113signed2)
success_txs.append(bip112tx_special_v2)
- success_txs.append(self.spend_tx(self.nodes[0], bip112tx_special_v2))
+ success_txs.append(
+ spend_tx(self.nodes[0], bip112tx_special_v2, self.nodeaddress))
# add BIP 68 txs
success_txs.extend(all_rlt_txs(bip68txs_v2))
# add BIP 112 with seq=10 txs
success_txs.extend(all_rlt_txs(bip112txs_vary_nSequence_v2))
- success_txs.extend([self.spend_tx(self.nodes[0], tx)
+ success_txs.extend([spend_tx(self.nodes[0], tx, self.nodeaddress)
for tx in all_rlt_txs(bip112txs_vary_nSequence_v2)])
success_txs.extend(all_rlt_txs(bip112txs_vary_OP_CSV_v2))
- success_txs.extend([self.spend_tx(self.nodes[0], tx)
+ success_txs.extend([spend_tx(self.nodes[0], tx, self.nodeaddress)
for tx in all_rlt_txs(bip112txs_vary_OP_CSV_v2)])
# try BIP 112 with seq=9 txs
success_txs.extend(all_rlt_txs(bip112txs_vary_nSequence_9_v2))
- success_txs.extend([self.spend_tx(self.nodes[0], tx)
+ success_txs.extend([spend_tx(self.nodes[0], tx, self.nodeaddress)
for tx in all_rlt_txs(bip112txs_vary_nSequence_9_v2)])
success_txs.extend(all_rlt_txs(bip112txs_vary_OP_CSV_9_v2))
- success_txs.extend([self.spend_tx(self.nodes[0], tx)
+ success_txs.extend([spend_tx(self.nodes[0], tx, self.nodeaddress)
for tx in all_rlt_txs(bip112txs_vary_OP_CSV_9_v2)])
# Test #4
yield TestInstance([[self.create_test_block(success_txs), True]])
self.nodes[0].invalidateblock(self.nodes[0].getbestblockhash())
# 1 more version 4 block to get us to height 575 so the fork should
# now be active for the next block
test_blocks = self.generate_blocks(1)
# Test #5
yield TestInstance(test_blocks, sync_every_block=False)
assert_equal(get_csv_status(self.nodes[0]), False)
self.nodes[0].generate(1)
assert_equal(get_csv_status(self.nodes[0]), True)
self.nodes[0].invalidateblock(self.nodes[0].getbestblockhash())
self.log.info("Post-Soft Fork Tests.")
self.log.info("BIP 113 tests")
# BIP 113 tests should now fail regardless of version number
# if nLockTime isn't satisfied by new rules
# = MTP of prior block (not <) but < time put on current block
bip113tx_v1.nLockTime = self.last_block_time - 600 * 5
- bip113signed1 = self.sign_transaction(self.nodes[0], bip113tx_v1)
+ bip113signed1 = sign_transaction(self.nodes[0], bip113tx_v1)
# = MTP of prior block (not <) but < time put on current block
bip113tx_v2.nLockTime = self.last_block_time - 600 * 5
- bip113signed2 = self.sign_transaction(self.nodes[0], bip113tx_v2)
+ bip113signed2 = sign_transaction(self.nodes[0], bip113tx_v2)
for bip113tx in [bip113signed1, bip113signed2]:
# Test #6, Test #7
yield TestInstance([[self.create_test_block([bip113tx]), False]])
# BIP 113 tests should now pass if the locktime is < MTP
# < MTP of prior block
bip113tx_v1.nLockTime = self.last_block_time - 600 * 5 - 1
- bip113signed1 = self.sign_transaction(self.nodes[0], bip113tx_v1)
+ bip113signed1 = sign_transaction(self.nodes[0], bip113tx_v1)
# < MTP of prior block
bip113tx_v2.nLockTime = self.last_block_time - 600 * 5 - 1
- bip113signed2 = self.sign_transaction(self.nodes[0], bip113tx_v2)
+ bip113signed2 = sign_transaction(self.nodes[0], bip113tx_v2)
for bip113tx in [bip113signed1, bip113signed2]:
# Test #8, Test #9
yield TestInstance([[self.create_test_block([bip113tx]), True]])
self.nodes[0].invalidateblock(self.nodes[0].getbestblockhash())
# Next block height = 580 after 4 blocks of random version
test_blocks = self.generate_blocks(4)
# Test #10
yield TestInstance(test_blocks, sync_every_block=False)
self.log.info("BIP 68 tests")
self.log.info("Test version 1 txs - all should still pass")
success_txs = []
success_txs.extend(all_rlt_txs(bip68txs_v1))
# Test #11
yield TestInstance([[self.create_test_block(success_txs), True]])
self.nodes[0].invalidateblock(self.nodes[0].getbestblockhash())
self.log.info("Test version 2 txs")
# All txs with SEQUENCE_LOCKTIME_DISABLE_FLAG set pass
bip68success_txs = [tx['tx'] for tx in bip68txs_v2 if tx['sdf']]
yield TestInstance([[self.create_test_block(bip68success_txs), True]])
self.nodes[0].invalidateblock(self.nodes[0].getbestblockhash())
# All txs without flag fail as we are at delta height = 8 < 10 and
# delta time = 8 * 600 < 10 * 512
bip68timetxs = [tx['tx']
for tx in bip68txs_v2 if not tx['sdf'] and tx['stf']]
for tx in bip68timetxs:
# Test #13 - Test #16
yield TestInstance([[self.create_test_block([tx]), False]])
bip68heighttxs = [tx['tx']
for tx in bip68txs_v2 if not tx['sdf'] and not tx['stf']]
for tx in bip68heighttxs:
# Test #17 - Test #20
yield TestInstance([[self.create_test_block([tx]), False]])
# Advance one block to 581
test_blocks = self.generate_blocks(1)
# Test #21
yield TestInstance(test_blocks, sync_every_block=False)
# Height txs should fail and time txs should now pass 9 * 600 > 10 * 512
bip68success_txs.extend(bip68timetxs)
# Test #22
yield TestInstance([[self.create_test_block(bip68success_txs), True]])
self.nodes[0].invalidateblock(self.nodes[0].getbestblockhash())
for tx in bip68heighttxs:
# Test #23 - Test #26
yield TestInstance([[self.create_test_block([tx]), False]])
# Advance one block to 582
test_blocks = self.generate_blocks(1)
# Test #27
yield TestInstance(test_blocks, sync_every_block=False)
# All BIP 68 txs should pass
bip68success_txs.extend(bip68heighttxs)
# Test #28
yield TestInstance([[self.create_test_block(bip68success_txs), True]])
self.nodes[0].invalidateblock(self.nodes[0].getbestblockhash())
self.log.info("BIP 112 tests")
self.log.info("Test version 1 txs")
# -1 OP_CSV tx should fail
# Test #29
yield TestInstance([[self.create_test_block_spend_utxos(self.nodes[0], [bip112tx_special_v1]), False]])
# If SEQUENCE_LOCKTIME_DISABLE_FLAG is set in argument to OP_CSV,
# version 1 txs should still pass
success_txs = [tx['tx']
for tx in bip112txs_vary_OP_CSV_v1 if tx['sdf']]
success_txs += [tx['tx']
for tx in bip112txs_vary_OP_CSV_9_v1 if tx['sdf']]
# Test #30
yield TestInstance([[self.create_test_block_spend_utxos(self.nodes[0], success_txs), True]])
self.nodes[0].invalidateblock(self.nodes[0].getbestblockhash())
# If SEQUENCE_LOCKTIME_DISABLE_FLAG is unset in argument to OP_CSV,
# version 1 txs should now fail
fail_txs = all_rlt_txs(bip112txs_vary_nSequence_v1)
fail_txs += all_rlt_txs(bip112txs_vary_nSequence_9_v1)
fail_txs += [tx['tx']
for tx in bip112txs_vary_OP_CSV_9_v1 if not tx['sdf']]
fail_txs += [tx['tx']
for tx in bip112txs_vary_OP_CSV_9_v1 if not tx['sdf']]
for tx in fail_txs:
# Test #31 - Test #78
yield TestInstance([[self.create_test_block_spend_utxos(self.nodes[0], [tx]), False]])
self.log.info("Version 2 txs")
# -1 OP_CSV tx should fail
# Test #79
yield TestInstance([[self.create_test_block_spend_utxos(self.nodes[0], [bip112tx_special_v2]), False]])
# If SEQUENCE_LOCKTIME_DISABLE_FLAG is set in argument to OP_CSV,
# version 2 txs should pass
success_txs = [tx['tx']
for tx in bip112txs_vary_OP_CSV_v2 if tx['sdf']]
success_txs += [tx['tx']
for tx in bip112txs_vary_OP_CSV_9_v2 if tx['sdf']]
# Test #80
yield TestInstance([[self.create_test_block_spend_utxos(self.nodes[0], success_txs), True]])
self.nodes[0].invalidateblock(self.nodes[0].getbestblockhash())
## SEQUENCE_LOCKTIME_DISABLE_FLAG is unset in argument to OP_CSV for all remaining txs ##
# All txs with nSequence 9 should fail either due to earlier mismatch
# or failing the CSV check
fail_txs = all_rlt_txs(bip112txs_vary_nSequence_9_v2)
fail_txs += [tx['tx']
for tx in bip112txs_vary_OP_CSV_9_v2 if not tx['sdf']]
for tx in fail_txs:
# Test #81 - Test #104
yield TestInstance([[self.create_test_block_spend_utxos(self.nodes[0], [tx]), False]])
# If SEQUENCE_LOCKTIME_DISABLE_FLAG is set in nSequence, tx should fail
fail_txs = [tx['tx']
for tx in bip112txs_vary_nSequence_v2 if tx['sdf']]
for tx in fail_txs:
# Test #105 - Test #112
yield TestInstance([[self.create_test_block_spend_utxos(self.nodes[0], [tx]), False]])
# If sequencelock types mismatch, tx should fail
fail_txs = [tx['tx']
for tx in bip112txs_vary_nSequence_v2 if not tx['sdf'] and tx['stf']]
fail_txs += [tx['tx']
for tx in bip112txs_vary_OP_CSV_v2 if not tx['sdf'] and tx['stf']]
for tx in fail_txs:
# Test #113 - Test #120
yield TestInstance([[self.create_test_block_spend_utxos(self.nodes[0], [tx]), False]])
# Remaining txs should pass, just test masking works properly
success_txs = [
tx['tx'] for tx in bip112txs_vary_nSequence_v2 if not tx['sdf'] and not tx['stf']]
success_txs += [tx['tx']
for tx in bip112txs_vary_OP_CSV_v2 if not tx['sdf'] and not tx['stf']]
# Test #121
yield TestInstance([[self.create_test_block(success_txs), True]])
# Spending the previous block utxos requires a difference of 10 blocks (nSequence = 10).
# Generate 9 blocks then spend in the 10th
block = self.nodes[0].getbestblockhash()
self.last_block_time += 600
self.tip = int("0x" + block, 0)
self.tipheight += 1
# Test #122
yield TestInstance(self.generate_blocks(9), sync_every_block=False)
spend_txs = []
for tx in success_txs:
- raw_tx = self.spend_tx(self.nodes[0], tx)
+ raw_tx = spend_tx(self.nodes[0], tx, self.nodeaddress)
raw_tx.vin[0].nSequence = BASE_RELATIVE_LOCKTIME
raw_tx.rehash()
spend_txs.append(raw_tx)
# Test #123
yield TestInstance([[self.create_test_block(spend_txs), True]])
self.nodes[0].invalidateblock(self.nodes[0].getbestblockhash())
# Additional test, of checking that comparison of two time types works properly
time_txs = []
for tx in [tx['tx'] for tx in bip112txs_vary_OP_CSV_v2 if not tx['sdf'] and tx['stf']]:
- signtx = self.sign_transaction(self.nodes[0], tx)
+ signtx = sign_transaction(self.nodes[0], tx)
time_txs.append(signtx)
# Test #124
yield TestInstance([[self.create_test_block(time_txs), True]])
# Spending the previous block utxos requires a block time difference of
# at least 10 * 512s (nSequence = 10).
# Generate 8 blocks then spend in the 9th (9 * 600 > 10 * 512)
block = self.nodes[0].getbestblockhash()
self.last_block_time += 600
self.tip = int("0x" + block, 0)
self.tipheight += 1
# Test #125
yield TestInstance(self.generate_blocks(8), sync_every_block=False)
spend_txs = []
for tx in time_txs:
- raw_tx = self.spend_tx(self.nodes[0], tx)
+ raw_tx = spend_tx(self.nodes[0], tx, self.nodeaddress)
raw_tx.vin[0].nSequence = BASE_RELATIVE_LOCKTIME | SEQ_TYPE_FLAG
raw_tx.rehash()
spend_txs.append(raw_tx)
# Test #126
yield TestInstance([[self.create_test_block(spend_txs), True]])
self.nodes[0].invalidateblock(self.nodes[0].getbestblockhash())
# Missing aspects of test
# Testing empty stack fails
if __name__ == '__main__':
BIP68_112_113Test().main()

File Metadata

Mime Type
text/x-diff
Expires
Tue, May 13, 01:35 (1 d, 3 h)
Storage Engine
blob
Storage Format
Raw Data
Storage Handle
5777020
Default Alt Text
(35 KB)

Event Timeline