mirror of
https://github.com/dashpay/dash.git
synced 2024-12-25 20:12:57 +01:00
b07a7b810c
* Merge #11796: [tests] Functional test naming convention5fecd84
[tests] Remove redundant import in blocktools.py test (Anthony Towns)9b20bb4
[tests] Check tests conform to naming convention (Anthony Towns)7250b4e
[tests] README.md nit fixes (Anthony Towns)82b2712
[tests] move witness util functions to blocktools.py (John Newbery)1e10854
[tests] [docs] update README for new test naming scheme (John Newbery) Pull request description: Splitting #11774 into two parts -- this part updates the README with the proposed naming convention, and adds some checks to test_runner.py that the number of tests violating the naming convention doesn't increase too much. Idea is this part of the change should not introduce merge conflicts or require much rebasing, so reviews of the complicated bits won't become invalidated too often; while the second part will just be file renames, which will require regular rebasing and will introduce merge conflicts with pending PRs, but can be merged later, and should also be much easier to review, since it will only include relatively trivial changes. Tree-SHA512: b96557d41714addbbfe2aed62fb5a48639eaeb1eb3aba30ac1b3a86bb3cb8d796c6247f9c414c4695c4bf54c0ec9968ac88e2f88fb62483bc1a2f89368f7fc80 * update violation count Signed-off-by: pasta <pasta@dashboost.org> * Merge #11774: [tests] Rename functional tests6f881cc880
[tests] Remove EXPECTED_VIOLATION_COUNT (Anthony Towns)3150b3fea7
[tests] Rename misc functional tests. (Anthony Towns)81b79f2c39
[tests] Rename rpc_* functional tests. (Anthony Towns)61b8f7f273
[tests] Rename p2p_* functional tests. (Anthony Towns)90600bc7db
[tests] Rename wallet_* functional tests. (Anthony Towns)ca6523d0c8
[tests] Rename feature_* functional tests. (Anthony Towns) Pull request description: This PR changes the functional tests to have a consistent naming scheme: tests for individual RPC methods are named rpc_... tests for interfaces (REST, ZMQ, RPC features) are named interface_... tests that explicitly test the p2p interface are named p2p_... tests for wallet features are named wallet_... tests for mining features are named mining_... tests for mempool behaviour are named mempool_... tests for full features that aren't wallet/mining/mempool are named feature_... Rationale: it's sometimes difficult for new contributors to know what's already covered by existing tests and where new tests should be added. Naming in a consistent fashion makes it easier to see what's already covered at a glance. Tree-SHA512: 4246790552d42bbd95f6d5bdf67702b81b3b2c583ce7eaf1fe6d8e254721279b47315973c6e9ae82dad6e4c747f12188160764bf2624c0f8f3b4d39330ec8b16 * rename tests and edit associated strings to align test-suite with test name standards Signed-off-by: pasta <pasta@dashboost.org> * fix grammar in test/functional/test_runner.py Co-authored-by: dustinface <35775977+xdustinface@users.noreply.github.com> * ci: Fix excluded test names * rename feature_privatesend.py to rpc_privatesend.py Signed-off-by: pasta <pasta@dashboost.org> Co-authored-by: Wladimir J. van der Laan <laanwj@gmail.com> Co-authored-by: MarcoFalke <falke.marco@gmail.com> Co-authored-by: dustinface <35775977+xdustinface@users.noreply.github.com> Co-authored-by: xdustinface <xdustinfacex@gmail.com>
499 lines
25 KiB
Python
Executable File
499 lines
25 KiB
Python
Executable File
#!/usr/bin/env python3
|
|
# Copyright (c) 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 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 61 blocks to transition from DEFINED to STARTED
|
|
mine 144 blocks only 100 of which are signaling readiness in order to fail to change state this period
|
|
mine 144 blocks with 108 signaling and verify STARTED->LOCKED_IN
|
|
mine 140 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
|
|
from io import BytesIO
|
|
|
|
from test_framework.blocktools import create_coinbase, create_block
|
|
from test_framework.mininode import ToHex, CTransaction
|
|
from test_framework.mininode import network_thread_start, P2PDataStore
|
|
from test_framework.script import (
|
|
CScript,
|
|
OP_CHECKSEQUENCEVERIFY,
|
|
OP_DROP,
|
|
)
|
|
from test_framework.test_framework import (BitcoinTestFramework, GENESISTIME)
|
|
from test_framework.util import (
|
|
assert_equal,
|
|
get_bip9_status,
|
|
hex_str_to_bytes,
|
|
)
|
|
|
|
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 create_transaction(node, txid, to_address, amount):
|
|
inputs = [{"txid": txid, "vout": 0}]
|
|
outputs = {to_address: amount}
|
|
rawtx = node.createrawtransaction(inputs, outputs)
|
|
tx = CTransaction()
|
|
f = BytesIO(hex_str_to_bytes(rawtx))
|
|
tx.deserialize(f)
|
|
return tx
|
|
|
|
def sign_transaction(node, unsignedtx):
|
|
rawtx = ToHex(unsignedtx)
|
|
signresult = node.signrawtransaction(rawtx)
|
|
tx = CTransaction()
|
|
f = BytesIO(hex_str_to_bytes(signresult['hex']))
|
|
tx.deserialize(f)
|
|
return tx
|
|
|
|
def create_bip112special(node, input, txversion, address):
|
|
tx = create_transaction(node, input, address, Decimal("499.98"))
|
|
tx.nVersion = txversion
|
|
signtx = sign_transaction(node, tx)
|
|
signtx.vin[0].scriptSig = CScript([-1, OP_CHECKSEQUENCEVERIFY, OP_DROP] + list(CScript(signtx.vin[0].scriptSig)))
|
|
return signtx
|
|
|
|
def send_generic_input_tx(node, coinbases, address):
|
|
amount = Decimal("499.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("499.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 bip68 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("499.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
|
|
signtx = sign_transaction(node, tx)
|
|
if (varyOP_CSV):
|
|
signtx.vin[0].scriptSig = CScript([locktime, OP_CHECKSEQUENCEVERIFY, OP_DROP] + list(CScript(signtx.vin[0].scriptSig)))
|
|
else:
|
|
signtx.vin[0].scriptSig = CScript([BASE_RELATIVE_LOCKTIME, OP_CHECKSEQUENCEVERIFY, OP_DROP] + list(CScript(signtx.vin[0].scriptSig)))
|
|
tx.rehash()
|
|
txs.append({'tx': signtx, 'sdf': sdf, 'stf': stf})
|
|
return txs
|
|
|
|
class BIP68_112_113Test(BitcoinTestFramework):
|
|
def set_test_params(self):
|
|
self.num_nodes = 1
|
|
self.setup_clean_chain = True
|
|
# Must also set '-maxtipage=600100' to allow syncing from very old blocks
|
|
# and '-dip3params=2000:2000' to create pre-dip3 blocks only
|
|
self.extra_args = [['-whitelist=127.0.0.1', '-blockversion=4', '-maxtipage=600100', '-dip3params=2000:2000']]
|
|
|
|
def setup_network(self):
|
|
self.setup_nodes()
|
|
|
|
def generate_blocks(self, number, version, test_blocks=None):
|
|
if test_blocks is None:
|
|
test_blocks = []
|
|
for i in range(number):
|
|
block = self.create_test_block([], version)
|
|
test_blocks.append(block)
|
|
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)
|
|
block.hashMerkleRoot = block.calc_merkle_root()
|
|
block.rehash()
|
|
block.solve()
|
|
return block
|
|
|
|
def send_blocks(self, blocks, success=True, reject_code=None, reject_reason=None, request_block=True):
|
|
"""Sends blocks to test node. Syncs and verifies that tip has advanced to most recent block.
|
|
|
|
Call with success = False if the tip shouldn't advance to the most recent block."""
|
|
self.nodes[0].p2p.send_blocks_and_test(blocks, self.nodes[0], success=success, reject_code=reject_code, reject_reason=reject_reason, request_block=request_block)
|
|
|
|
def run_test(self):
|
|
self.nodes[0].add_p2p_connection(P2PDataStore())
|
|
network_thread_start()
|
|
self.nodes[0].p2p.wait_for_verack()
|
|
|
|
self.log.info("Generate blocks in the past for coinbase outputs.")
|
|
self.coinbase_blocks = self.nodes[0].generate(1 + 16 + 2 * 32 + 1) # 82 blocks generated for inputs
|
|
# set time so that there was enough time to build up to 1000 blocks 10 minutes apart on top of the last one
|
|
# without worrying about getting into the future
|
|
self.nodes[0].setmocktime(GENESISTIME + 600 * 1000 + 100)
|
|
self.tipheight = 82 # height of the next block to build
|
|
self.last_block_time = GENESISTIME
|
|
self.tip = int(self.nodes[0].getbestblockhash(), 16)
|
|
self.nodeaddress = self.nodes[0].getnewaddress()
|
|
|
|
self.log.info("Test that the csv softfork is DEFINED")
|
|
assert_equal(get_bip9_status(self.nodes[0], 'csv')['status'], 'defined')
|
|
test_blocks = self.generate_blocks(61, 4)
|
|
self.send_blocks(test_blocks)
|
|
|
|
self.log.info("Advance from DEFINED to STARTED, height = 143")
|
|
assert_equal(get_bip9_status(self.nodes[0], 'csv')['status'], 'started')
|
|
|
|
self.log.info("Fail to achieve LOCKED_IN")
|
|
# 100 out of 144 signal bit 0. Use a variety of bits to simulate multiple parallel softforks
|
|
|
|
test_blocks = self.generate_blocks(50, 536870913) # 0x20000001 (signalling ready)
|
|
test_blocks = self.generate_blocks(20, 4, test_blocks) # 0x00000004 (signalling not)
|
|
test_blocks = self.generate_blocks(50, 536871169, test_blocks) # 0x20000101 (signalling ready)
|
|
test_blocks = self.generate_blocks(24, 536936448, test_blocks) # 0x20010000 (signalling not)
|
|
self.send_blocks(test_blocks)
|
|
|
|
self.log.info("Failed to advance past STARTED, height = 287")
|
|
assert_equal(get_bip9_status(self.nodes[0], 'csv')['status'], 'started')
|
|
|
|
self.log.info("Generate blocks to achieve LOCK-IN")
|
|
# 108 out of 144 signal bit 0 to achieve lock-in
|
|
# using a variety of bits to simulate multiple parallel softforks
|
|
test_blocks = self.generate_blocks(58, 536870913) # 0x20000001 (signalling ready)
|
|
test_blocks = self.generate_blocks(26, 4, test_blocks) # 0x00000004 (signalling not)
|
|
test_blocks = self.generate_blocks(50, 536871169, test_blocks) # 0x20000101 (signalling ready)
|
|
test_blocks = self.generate_blocks(10, 536936448, test_blocks) # 0x20010000 (signalling not)
|
|
self.send_blocks(test_blocks)
|
|
|
|
self.log.info("Advanced from STARTED to LOCKED_IN, height = 431")
|
|
assert_equal(get_bip9_status(self.nodes[0], 'csv')['status'], 'locked_in')
|
|
|
|
# Generate 140 more version 4 blocks
|
|
test_blocks = self.generate_blocks(140, 4)
|
|
self.send_blocks(test_blocks)
|
|
|
|
# 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(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(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(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 = send_generic_input_tx(self.nodes[0], self.coinbase_blocks, self.nodeaddress)
|
|
|
|
# 1 normal input
|
|
bip113input = send_generic_input_tx(self.nodes[0], self.coinbase_blocks, self.nodeaddress)
|
|
|
|
self.nodes[0].setmocktime(self.last_block_time + 600)
|
|
inputblockhash = self.nodes[0].generate(1)[0] # 1 block generated for inputs to be in chain at height 572
|
|
self.nodes[0].setmocktime(GENESISTIME + 600 * 1000 + 100)
|
|
self.tip = int(inputblockhash, 16)
|
|
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, 4)
|
|
self.send_blocks(test_blocks)
|
|
|
|
self.log.info("Not yet advanced to ACTIVE, height = 574 (will activate for block 576, not 575)")
|
|
assert_equal(get_bip9_status(self.nodes[0], 'csv')['status'], 'locked_in')
|
|
|
|
# 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 = create_transaction(self.nodes[0], bip113input, self.nodeaddress, Decimal("499.98"))
|
|
bip113tx_v1.vin[0].nSequence = 0xFFFFFFFE
|
|
bip113tx_v1.nVersion = 1
|
|
bip113tx_v2 = create_transaction(self.nodes[0], bip113input, self.nodeaddress, Decimal("499.98"))
|
|
bip113tx_v2.vin[0].nSequence = 0xFFFFFFFE
|
|
bip113tx_v2.nVersion = 2
|
|
|
|
# For BIP68 test all 16 relative sequence locktimes
|
|
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 = 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 = 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 = 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 = 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 = 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
|
|
bip113tx_v1.nLockTime = self.last_block_time - 600 * 5 # = MTP of prior block (not <) but < time put on current block
|
|
bip113signed1 = sign_transaction(self.nodes[0], bip113tx_v1)
|
|
success_txs.append(bip113signed1)
|
|
success_txs.append(bip112tx_special_v1)
|
|
# 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(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(all_rlt_txs(bip112txs_vary_OP_CSV_9_v1))
|
|
self.send_blocks([self.create_test_block(success_txs)])
|
|
self.nodes[0].invalidateblock(self.nodes[0].getbestblockhash())
|
|
|
|
self.log.info("Test version 2 txs")
|
|
|
|
success_txs = []
|
|
# add BIP113 tx and -1 CSV tx
|
|
bip113tx_v2.nLockTime = self.last_block_time - 600 * 5 # = MTP of prior block (not <) but < time put on current block
|
|
bip113signed2 = sign_transaction(self.nodes[0], bip113tx_v2)
|
|
success_txs.append(bip113signed2)
|
|
success_txs.append(bip112tx_special_v2)
|
|
# 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(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(all_rlt_txs(bip112txs_vary_OP_CSV_9_v2))
|
|
self.send_blocks([self.create_test_block(success_txs)])
|
|
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, 4)
|
|
self.send_blocks(test_blocks)
|
|
assert_equal(get_bip9_status(self.nodes[0], 'csv')['status'], 'active')
|
|
|
|
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
|
|
bip113tx_v1.nLockTime = self.last_block_time - 600 * 5 # = MTP of prior block (not <) but < time put on current block
|
|
bip113signed1 = sign_transaction(self.nodes[0], bip113tx_v1)
|
|
bip113tx_v2.nLockTime = self.last_block_time - 600 * 5 # = MTP of prior block (not <) but < time put on current block
|
|
bip113signed2 = sign_transaction(self.nodes[0], bip113tx_v2)
|
|
for bip113tx in [bip113signed1, bip113signed2]:
|
|
self.send_blocks([self.create_test_block([bip113tx])], success=False)
|
|
# BIP 113 tests should now pass if the locktime is < MTP
|
|
bip113tx_v1.nLockTime = self.last_block_time - 600 * 5 - 1 # < MTP of prior block
|
|
bip113signed1 = sign_transaction(self.nodes[0], bip113tx_v1)
|
|
bip113tx_v2.nLockTime = self.last_block_time - 600 * 5 - 1 # < MTP of prior block
|
|
bip113signed2 = sign_transaction(self.nodes[0], bip113tx_v2)
|
|
for bip113tx in [bip113signed1, bip113signed2]:
|
|
self.send_blocks([self.create_test_block([bip113tx])])
|
|
self.nodes[0].invalidateblock(self.nodes[0].getbestblockhash())
|
|
|
|
# Next block height = 580 after 4 blocks of random version
|
|
test_blocks = self.generate_blocks(4, 1234)
|
|
self.send_blocks(test_blocks)
|
|
|
|
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))
|
|
self.send_blocks([self.create_test_block(success_txs)])
|
|
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']]
|
|
self.send_blocks([self.create_test_block(bip68success_txs)])
|
|
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:
|
|
self.send_blocks([self.create_test_block([tx])], success=False)
|
|
|
|
bip68heighttxs = [tx['tx'] for tx in bip68txs_v2 if not tx['sdf'] and not tx['stf']]
|
|
for tx in bip68heighttxs:
|
|
self.send_blocks([self.create_test_block([tx])], success=False)
|
|
|
|
# Advance one block to 581
|
|
test_blocks = self.generate_blocks(1, 1234)
|
|
self.send_blocks(test_blocks)
|
|
|
|
# Height txs should fail and time txs should now pass 9 * 600 > 10 * 512
|
|
bip68success_txs.extend(bip68timetxs)
|
|
self.send_blocks([self.create_test_block(bip68success_txs)])
|
|
self.nodes[0].invalidateblock(self.nodes[0].getbestblockhash())
|
|
for tx in bip68heighttxs:
|
|
self.send_blocks([self.create_test_block([tx])], success=False)
|
|
|
|
# Advance one block to 582
|
|
test_blocks = self.generate_blocks(1, 1234)
|
|
self.send_blocks(test_blocks)
|
|
|
|
# All BIP 68 txs should pass
|
|
bip68success_txs.extend(bip68heighttxs)
|
|
self.send_blocks([self.create_test_block(bip68success_txs)])
|
|
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
|
|
self.send_blocks([self.create_test_block([bip112tx_special_v1])], success=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']]
|
|
self.send_blocks([self.create_test_block(success_txs)])
|
|
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:
|
|
self.send_blocks([self.create_test_block([tx])], success=False)
|
|
|
|
self.log.info("Test version 2 txs")
|
|
|
|
# -1 OP_CSV tx should fail
|
|
self.send_blocks([self.create_test_block([bip112tx_special_v2])], success=False)
|
|
|
|
# If SEQUENCE_LOCKTIME_DISABLE_FLAG is set in argument to OP_CSV, version 2 txs should pass (all sequence locks are met)
|
|
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']]
|
|
|
|
self.send_blocks([self.create_test_block(success_txs)])
|
|
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:
|
|
self.send_blocks([self.create_test_block([tx])], success=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:
|
|
self.send_blocks([self.create_test_block([tx])], success=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:
|
|
self.send_blocks([self.create_test_block([tx])], success=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']]
|
|
self.send_blocks([self.create_test_block(success_txs)])
|
|
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']]:
|
|
tx.vin[0].nSequence = BASE_RELATIVE_LOCKTIME | SEQ_TYPE_FLAG
|
|
signtx = sign_transaction(self.nodes[0], tx)
|
|
time_txs.append(signtx)
|
|
|
|
self.send_blocks([self.create_test_block(time_txs)])
|
|
self.nodes[0].invalidateblock(self.nodes[0].getbestblockhash())
|
|
|
|
# TODO: Test empty stack fails
|
|
|
|
if __name__ == '__main__':
|
|
BIP68_112_113Test().main()
|