Skip to content

Commit bcb316b

Browse files
committed
Merge bitcoin/bitcoin#32050: test: avoid treating hash results as integers
a82829f test: simplify (w)txid checks by avoiding .calc_sha256 calls (Sebastian Falbesoner) 346a099 test: avoid unneeded hash -> uint256 -> hash roundtrips (Sebastian Falbesoner) Pull request description: In the functional test framework we currently have a strong tendency to treat and store identifiers that result from hash functions (e.g. (w)txids, block hashes) as integers, which seems an unnatural and confusing choice. Hashes are just pseudo-random sequences of bytes, and there is usually no need to apply integer operations on them; the only exceptions I could think of is PoW-verification of block hashes with the less-than (`<`) operator, or interpreting the byte-string as scalar in the EC-context for e.g. key derivation. I'd hence argue that most uses of `ser_uint256`/`uint256_from_str` and txid conversions via `int(txid/blockhash, 16)` are potential code smells and should be reduced to a minimum long-term if possible. This PR is a first step into this direction, intentionally kept small with (what I think) uncontroversial changes for demonstration purposes, to check out if other contributors are interested in this. A next step could be to change the classes of primitives (CTransaction, CBlock etc.) and network messages (msg_) to store hash results as actual bytes (maybe in a class wrapping the bytes that offers conversion from/to human-readable strings [1], for easier interaction with RPC calls and debug outputs) rather than ints. But that would of course need larger, potentially more controversial changes, and its questionable if its really worth the effort. [1] unfortunately, txids and block hashes are shown to user in reverse byte order, so e.g. a txid_bytes->txid_str conversion is not just a simple `txid_bytes.hex()`, but a `txid_bytes[::-1].hex()` ACKs for top commit: maflcko: review ACK a82829f 🐘 rkrux: Concept and utACK a82829f ryanofsky: Code review ACK a82829f. Nice changes, and sorry about the false bug report Tree-SHA512: bb0465802d743a495207800f922b65f49ed0d20552f95bb0bee764944664092aad74812e29df6e01ef40bcb8f9bc6c84c7e9cbbe6f008ee1a14d94ed88e698b4
2 parents a54baa8 + a82829f commit bcb316b

File tree

5 files changed

+22
-28
lines changed

5 files changed

+22
-28
lines changed

test/functional/feature_segwit.py

+3-3
Original file line numberDiff line numberDiff line change
@@ -267,7 +267,7 @@ def run_test(self):
267267
tx1 = tx_from_hex(tx1_hex)
268268

269269
# Check that wtxid is properly reported in mempool entry (txid1)
270-
assert_equal(int(self.nodes[0].getmempoolentry(txid1)["wtxid"], 16), tx1.calc_sha256(True))
270+
assert_equal(self.nodes[0].getmempoolentry(txid1)["wtxid"], tx1.getwtxid())
271271

272272
# Check that weight and vsize are properly reported in mempool entry (txid1)
273273
assert_equal(self.nodes[0].getmempoolentry(txid1)["vsize"], tx1.get_vsize())
@@ -283,7 +283,7 @@ def run_test(self):
283283
assert not tx.wit.is_null()
284284

285285
# Check that wtxid is properly reported in mempool entry (txid2)
286-
assert_equal(int(self.nodes[0].getmempoolentry(txid2)["wtxid"], 16), tx.calc_sha256(True))
286+
assert_equal(self.nodes[0].getmempoolentry(txid2)["wtxid"], tx.getwtxid())
287287

288288
# Check that weight and vsize are properly reported in mempool entry (txid2)
289289
assert_equal(self.nodes[0].getmempoolentry(txid2)["vsize"], tx.get_vsize())
@@ -306,7 +306,7 @@ def run_test(self):
306306
assert txid3 in template_txids
307307

308308
# Check that wtxid is properly reported in mempool entry (txid3)
309-
assert_equal(int(self.nodes[0].getmempoolentry(txid3)["wtxid"], 16), tx.calc_sha256(True))
309+
assert_equal(self.nodes[0].getmempoolentry(txid3)["wtxid"], tx.getwtxid())
310310

311311
# Check that weight and vsize are properly reported in mempool entry (txid3)
312312
assert_equal(self.nodes[0].getmempoolentry(txid3)["vsize"], tx.get_vsize())

test/functional/p2p_compactblocks.py

+4-7
Original file line numberDiff line numberDiff line change
@@ -347,13 +347,11 @@ def check_compactblock_construction_from_block(self, header_and_shortids, block_
347347

348348
# Check that all prefilled_txn entries match what's in the block.
349349
for entry in header_and_shortids.prefilled_txn:
350-
entry.tx.calc_sha256()
351350
# This checks the non-witness parts of the tx agree
352-
assert_equal(entry.tx.sha256, block.vtx[entry.index].sha256)
351+
assert_equal(entry.tx.rehash(), block.vtx[entry.index].rehash())
353352

354353
# And this checks the witness
355-
wtxid = entry.tx.calc_sha256(True)
356-
assert_equal(wtxid, block.vtx[entry.index].calc_sha256(True))
354+
assert_equal(entry.tx.getwtxid(), block.vtx[entry.index].getwtxid())
357355

358356
# Check that the cmpctblock message announced all the transactions.
359357
assert_equal(len(header_and_shortids.prefilled_txn) + len(header_and_shortids.shortids), len(block.vtx))
@@ -590,10 +588,9 @@ def test_getblocktxn_handler(self, test_node):
590588
all_indices = msg.block_txn_request.to_absolute()
591589
for index in all_indices:
592590
tx = test_node.last_message["blocktxn"].block_transactions.transactions.pop(0)
593-
tx.calc_sha256()
594-
assert_equal(tx.sha256, block.vtx[index].sha256)
591+
assert_equal(tx.rehash(), block.vtx[index].rehash())
595592
# Check that the witness matches
596-
assert_equal(tx.calc_sha256(True), block.vtx[index].calc_sha256(True))
593+
assert_equal(tx.getwtxid(), block.vtx[index].getwtxid())
597594
test_node.last_message.pop("blocktxn", None)
598595
current_height -= 1
599596

test/functional/p2p_segwit.py

+2-3
Original file line numberDiff line numberDiff line change
@@ -336,8 +336,7 @@ def test_unnecessary_witness_before_segwit_activation(self):
336336

337337
# Verify the hash with witness differs from the txid
338338
# (otherwise our testing framework must be broken!)
339-
tx.rehash()
340-
assert tx.sha256 != tx.calc_sha256(with_witness=True)
339+
assert tx.rehash() != tx.getwtxid()
341340

342341
# Construct a block that includes the transaction.
343342
block = self.build_next_block()
@@ -1293,7 +1292,7 @@ def test_tx_relay_after_segwit_activation(self):
12931292
# Test that getrawtransaction returns correct witness information
12941293
# hash, size, vsize
12951294
raw_tx = self.nodes[0].getrawtransaction(tx3.hash, 1)
1296-
assert_equal(int(raw_tx["hash"], 16), tx3.calc_sha256(True))
1295+
assert_equal(raw_tx["hash"], tx3.getwtxid())
12971296
assert_equal(raw_tx["size"], len(tx3.serialize_with_witness()))
12981297
vsize = tx3.get_vsize()
12991298
assert_equal(raw_tx["vsize"], vsize)

test/functional/test_framework/blocktools.py

+2-3
Original file line numberDiff line numberDiff line change
@@ -28,7 +28,6 @@
2828
ser_uint256,
2929
tx_from_hex,
3030
uint256_from_compact,
31-
uint256_from_str,
3231
WITNESS_SCALE_FACTOR,
3332
)
3433
from .script import (
@@ -111,8 +110,8 @@ def create_block(hashprev=None, coinbase=None, ntime=None, *, version=None, tmpl
111110
return block
112111

113112
def get_witness_script(witness_root, witness_nonce):
114-
witness_commitment = uint256_from_str(hash256(ser_uint256(witness_root) + ser_uint256(witness_nonce)))
115-
output_data = WITNESS_COMMITMENT_HEADER + ser_uint256(witness_commitment)
113+
witness_commitment = hash256(ser_uint256(witness_root) + ser_uint256(witness_nonce))
114+
output_data = WITNESS_COMMITMENT_HEADER + witness_commitment
116115
return CScript([OP_RETURN, output_data])
117116

118117
def add_witness_commitment(block, nonce=0):

test/functional/test_framework/script.py

+11-12
Original file line numberDiff line numberDiff line change
@@ -17,9 +17,7 @@
1717
CTxOut,
1818
hash256,
1919
ser_string,
20-
ser_uint256,
2120
sha256,
22-
uint256_from_str,
2321
)
2422

2523
from .crypto.ripemd160 import ripemd160
@@ -711,41 +709,42 @@ def sign_input_segwitv0(tx, input_index, input_scriptpubkey, input_amount, privk
711709
# Note that this corresponds to sigversion == 1 in EvalScript, which is used
712710
# for version 0 witnesses.
713711
def SegwitV0SignatureMsg(script, txTo, inIdx, hashtype, amount):
712+
ZERO_HASH = bytes([0]*32)
714713

715-
hashPrevouts = 0
716-
hashSequence = 0
717-
hashOutputs = 0
714+
hashPrevouts = ZERO_HASH
715+
hashSequence = ZERO_HASH
716+
hashOutputs = ZERO_HASH
718717

719718
if not (hashtype & SIGHASH_ANYONECANPAY):
720719
serialize_prevouts = bytes()
721720
for i in txTo.vin:
722721
serialize_prevouts += i.prevout.serialize()
723-
hashPrevouts = uint256_from_str(hash256(serialize_prevouts))
722+
hashPrevouts = hash256(serialize_prevouts)
724723

725724
if (not (hashtype & SIGHASH_ANYONECANPAY) and (hashtype & 0x1f) != SIGHASH_SINGLE and (hashtype & 0x1f) != SIGHASH_NONE):
726725
serialize_sequence = bytes()
727726
for i in txTo.vin:
728727
serialize_sequence += i.nSequence.to_bytes(4, "little")
729-
hashSequence = uint256_from_str(hash256(serialize_sequence))
728+
hashSequence = hash256(serialize_sequence)
730729

731730
if ((hashtype & 0x1f) != SIGHASH_SINGLE and (hashtype & 0x1f) != SIGHASH_NONE):
732731
serialize_outputs = bytes()
733732
for o in txTo.vout:
734733
serialize_outputs += o.serialize()
735-
hashOutputs = uint256_from_str(hash256(serialize_outputs))
734+
hashOutputs = hash256(serialize_outputs)
736735
elif ((hashtype & 0x1f) == SIGHASH_SINGLE and inIdx < len(txTo.vout)):
737736
serialize_outputs = txTo.vout[inIdx].serialize()
738-
hashOutputs = uint256_from_str(hash256(serialize_outputs))
737+
hashOutputs = hash256(serialize_outputs)
739738

740739
ss = bytes()
741740
ss += txTo.version.to_bytes(4, "little")
742-
ss += ser_uint256(hashPrevouts)
743-
ss += ser_uint256(hashSequence)
741+
ss += hashPrevouts
742+
ss += hashSequence
744743
ss += txTo.vin[inIdx].prevout.serialize()
745744
ss += ser_string(script)
746745
ss += amount.to_bytes(8, "little", signed=True)
747746
ss += txTo.vin[inIdx].nSequence.to_bytes(4, "little")
748-
ss += ser_uint256(hashOutputs)
747+
ss += hashOutputs
749748
ss += txTo.nLockTime.to_bytes(4, "little")
750749
ss += hashtype.to_bytes(4, "little")
751750
return ss

0 commit comments

Comments
 (0)