forked from bitcoin/bitcoin
-
Notifications
You must be signed in to change notification settings - Fork 7
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Loading branch information
1 parent
6473ec0
commit f506e80
Showing
2 changed files
with
343 additions
and
0 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
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,342 @@ | ||
#!/usr/bin/env python3 | ||
# Copyright (c) 2015-2024 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 (OP_CAT) | ||
""" | ||
|
||
from test_framework.blocktools import ( | ||
create_coinbase, | ||
create_block, | ||
add_witness_commitment, | ||
) | ||
from test_framework.messages import ( | ||
CTransaction, | ||
CTxOut, | ||
CTxIn, | ||
CTxInWitness, | ||
COutPoint, | ||
COIN, | ||
sha256, | ||
) | ||
from test_framework.p2p import P2PInterface | ||
from test_framework.script import ( | ||
CScript, | ||
OP_CAT, | ||
OP_EQUAL, | ||
OP_2, | ||
OP_DUP, | ||
taproot_construct, | ||
) | ||
from test_framework.script_util import script_to_p2sh_script | ||
from test_framework.key import ECKey, compute_xonly_pubkey | ||
from test_framework.test_framework import BitcoinTestFramework | ||
from test_framework.util import assert_equal, assert_raises_rpc_error | ||
from test_framework.wallet import MiniWallet, MiniWalletMode | ||
from decimal import Decimal | ||
import random | ||
from io import BytesIO | ||
from test_framework.address import script_to_p2sh | ||
|
||
DISCOURAGED_ERROR = ( | ||
"non-mandatory-script-verify-flag (NOPx reserved for soft-fork upgrades)" | ||
) | ||
STACK_TOO_SHORT_ERROR = ( | ||
"non-mandatory-script-verify-flag (Operation not valid with the current stack size)" | ||
) | ||
DISABLED_OP_CODE = ( | ||
"mandatory-script-verify-flag-failed (Attempted to use a disabled opcode)" | ||
) | ||
MAX_PUSH_ERROR = ( | ||
"non-mandatory-script-verify-flag (Push value size limit exceeded)" | ||
) | ||
|
||
def random_bytes(n): | ||
return bytes(random.getrandbits(8) for i in range(n)) | ||
|
||
|
||
def random_p2sh(): | ||
return script_to_p2sh_script(random_bytes(20)) | ||
|
||
|
||
def create_transaction_to_script(node, wallet, txid, script, *, amount_sats): | ||
"""Return signed transaction spending the first output of the | ||
input txid. Note that the node must be able to sign for the | ||
output that is being spent, and the node must not be running | ||
multiple wallets. | ||
""" | ||
random_address = script_to_p2sh(CScript()) | ||
output = wallet.get_utxo(txid=txid) | ||
rawtx = node.createrawtransaction( | ||
inputs=[{"txid": output["txid"], "vout": output["vout"]}], | ||
outputs={random_address: Decimal(amount_sats) / COIN}, | ||
) | ||
tx = CTransaction() | ||
tx.deserialize(BytesIO(bytes.fromhex(rawtx))) | ||
# Replace with our script | ||
tx.vout[0].scriptPubKey = script | ||
# Sign | ||
wallet.sign_tx(tx) | ||
return tx | ||
|
||
|
||
class CatTest(BitcoinTestFramework): | ||
def set_test_params(self): | ||
self.num_nodes = 1 | ||
self.extra_args = [ | ||
["-par=1"] | ||
] # Use only one script thread to get the exact reject reason for testing | ||
self.setup_clean_chain = True | ||
self.rpc_timeout = 120 | ||
|
||
def get_block(self, txs): | ||
self.tip = self.nodes[0].getbestblockhash() | ||
self.height = self.nodes[0].getblockcount() | ||
self.log.debug(self.height) | ||
block = create_block( | ||
int(self.tip, 16), create_coinbase(self.height + 1)) | ||
block.vtx.extend(txs) | ||
add_witness_commitment(block) | ||
block.hashMerkleRoot = block.calc_merkle_root() | ||
block.solve() | ||
return block.serialize(True).hex(), block.hash | ||
|
||
def add_block(self, txs): | ||
block, h = self.get_block(txs) | ||
reason = self.nodes[0].submitblock(block) | ||
if reason: | ||
self.log.debug("Reject Reason: [%s]", reason) | ||
assert_equal(self.nodes[0].getbestblockhash(), h) | ||
return h | ||
|
||
def run_test(self): | ||
# The goal is to test a number of circumstances and combinations of parameters. Roughly: | ||
# | ||
# - Taproot OP_CAT | ||
# - CAT should fail when stack limit is hit | ||
# - CAT should fail if there is insuffecient number of element on the stack | ||
# - CAT should be able to concatenate two 8 byte payloads and check the resulting 16 byte payload | ||
# - Segwit v0 OP_CAT | ||
# - Spend should fail due to using disabled opcodes | ||
|
||
wallet = MiniWallet(self.nodes[0], mode=MiniWalletMode.RAW_P2PK) | ||
self.nodes[0].add_p2p_connection(P2PInterface()) | ||
|
||
BLOCKS = 200 | ||
self.log.info("Mining %d blocks for mature coinbases", BLOCKS) | ||
# Drop the last 100 as they're unspendable! | ||
coinbase_txids = [ | ||
self.nodes[0].getblock(b)["tx"][0] | ||
for b in self.generate(wallet, BLOCKS)[:-100] | ||
] | ||
def get_coinbase(): return coinbase_txids.pop() | ||
self.log.info("Creating setup transactions") | ||
|
||
outputs = [CTxOut(i * 1000, random_p2sh()) for i in range(1, 11)] | ||
# Add some fee satoshis | ||
amount_sats = sum(out.nValue for out in outputs) + 200 * 500 | ||
|
||
self.log.info( | ||
"Creating funding txn for 10 random outputs as a Taproot script") | ||
private_key = ECKey() | ||
# use simple deterministic private key (k=1) | ||
private_key.set((1).to_bytes(32, "big"), False) | ||
assert private_key.is_valid | ||
public_key, _ = compute_xonly_pubkey(private_key.get_bytes()) | ||
|
||
self.log.info( | ||
"Creating CAT tx with not enough values on the stack") | ||
not_enough_stack_elements_script = CScript([OP_CAT, OP_EQUAL, OP_2]) | ||
taproot_not_enough_stack_elements = taproot_construct( | ||
public_key, [("only-path", not_enough_stack_elements_script, 0xC0)]) | ||
taproot_not_enough_stack_elements_funding_tx = create_transaction_to_script( | ||
self.nodes[0], | ||
wallet, | ||
get_coinbase(), | ||
taproot_not_enough_stack_elements.scriptPubKey, | ||
amount_sats=amount_sats, | ||
) | ||
|
||
self.log.info( | ||
"Creating CAT tx that exceeds the stack element limit size") | ||
# Convert hex value to bytes | ||
hex_bytes = bytes.fromhex(('00' * 8)) | ||
stack_limit_script = CScript( | ||
[ | ||
hex_bytes, | ||
OP_DUP, | ||
OP_CAT, | ||
# 16 bytes on the stack | ||
OP_DUP, | ||
OP_CAT, | ||
# 32 bytes on the stack | ||
OP_DUP, | ||
OP_CAT, | ||
# 64 bytes on the stack | ||
OP_DUP, | ||
OP_CAT, | ||
# 128 bytes on the stack | ||
OP_DUP, | ||
OP_CAT, | ||
# 256 bytes on the stack | ||
OP_DUP, | ||
OP_CAT, | ||
# 512 bytes on the stack | ||
OP_DUP, | ||
OP_CAT, | ||
]) | ||
|
||
taproot_stack_limit = taproot_construct( | ||
public_key, [("only-path", stack_limit_script, 0xC0)]) | ||
taproot_stack_limit_funding_tx = create_transaction_to_script( | ||
self.nodes[0], | ||
wallet, | ||
get_coinbase(), | ||
taproot_stack_limit.scriptPubKey, | ||
amount_sats=amount_sats, | ||
) | ||
self.log.info( | ||
"Creating CAT tx that concatenates to values and verifies") | ||
hex_value_verify = bytes.fromhex('00' * 16) | ||
op_cat_verify_script = CScript([ | ||
hex_bytes, | ||
OP_DUP, | ||
OP_CAT, | ||
hex_value_verify, | ||
OP_EQUAL, | ||
]) | ||
|
||
taproot_op_cat = taproot_construct( | ||
public_key, [("only-path", op_cat_verify_script, 0xC0)]) | ||
taproot_op_cat_funding_tx = create_transaction_to_script( | ||
self.nodes[0], | ||
wallet, | ||
get_coinbase(), | ||
taproot_op_cat.scriptPubKey, | ||
amount_sats=amount_sats, | ||
) | ||
|
||
self.log.info("Creating a CAT segwit funding tx") | ||
segwit_cat_funding_tx = create_transaction_to_script( | ||
self.nodes[0], | ||
wallet, | ||
get_coinbase(), | ||
CScript([0, sha256(op_cat_verify_script)]), | ||
amount_sats=amount_sats, | ||
) | ||
|
||
funding_txs = [ | ||
taproot_not_enough_stack_elements_funding_tx, | ||
taproot_stack_limit_funding_tx, | ||
taproot_op_cat_funding_tx, | ||
segwit_cat_funding_tx, | ||
] | ||
self.log.info("Obtaining TXIDs") | ||
( | ||
taproot_not_enough_stack_elements_outpoint, | ||
taproot_stack_limit_outpoint, | ||
taproot_op_cat_outpoint, | ||
segwit_op_cat_outpoint, | ||
) = [COutPoint(int(tx.rehash(), 16), 0) for tx in funding_txs] | ||
|
||
self.log.info("Funding all outputs") | ||
self.add_block(funding_txs) | ||
|
||
self.log.info("Testing Taproot not enough stack elements OP_CAT spend") | ||
# Test sendrawtransaction | ||
taproot_op_cat_not_enough_stack_elements_spend = CTransaction() | ||
taproot_op_cat_not_enough_stack_elements_spend.nVersion = 2 | ||
taproot_op_cat_not_enough_stack_elements_spend.vin = [ | ||
CTxIn(taproot_not_enough_stack_elements_outpoint)] | ||
taproot_op_cat_not_enough_stack_elements_spend.vout = outputs | ||
taproot_op_cat_not_enough_stack_elements_spend.wit.vtxinwit += [ | ||
CTxInWitness()] | ||
taproot_op_cat_not_enough_stack_elements_spend.wit.vtxinwit[0].scriptWitness.stack = [ | ||
not_enough_stack_elements_script, | ||
bytes([0xC0 + taproot_not_enough_stack_elements.negflag]) + | ||
taproot_not_enough_stack_elements.internal_pubkey, | ||
] | ||
|
||
assert_raises_rpc_error( | ||
-26, | ||
STACK_TOO_SHORT_ERROR, | ||
self.nodes[0].sendrawtransaction, | ||
taproot_op_cat_not_enough_stack_elements_spend.serialize().hex(), | ||
) | ||
self.log.info( | ||
"OP_CAT with wrong size stack rejected by sendrawtransaction as discouraged" | ||
) | ||
|
||
self.log.info("Testing Taproot tx with stack element size limit") | ||
taproot_op_cat_stack_limit_spend = CTransaction() | ||
taproot_op_cat_stack_limit_spend.nVersion = 2 | ||
taproot_op_cat_stack_limit_spend.vin = [ | ||
CTxIn(taproot_stack_limit_outpoint)] | ||
taproot_op_cat_stack_limit_spend.vout = outputs | ||
taproot_op_cat_stack_limit_spend.wit.vtxinwit += [ | ||
CTxInWitness()] | ||
taproot_op_cat_stack_limit_spend.wit.vtxinwit[0].scriptWitness.stack = [ | ||
stack_limit_script, | ||
bytes([0xC0 + taproot_stack_limit.negflag]) + | ||
taproot_stack_limit.internal_pubkey, | ||
] | ||
|
||
assert_raises_rpc_error( | ||
-26, | ||
MAX_PUSH_ERROR, | ||
self.nodes[0].sendrawtransaction, | ||
taproot_op_cat_stack_limit_spend.serialize().hex(), | ||
) | ||
self.log.info( | ||
"OP_CAT with stack size limit rejected by sendrawtransaction as discouraged" | ||
) | ||
|
||
self.log.info("Testing Taproot OP_CAT usage") | ||
taproot_op_cat_transaction = CTransaction() | ||
taproot_op_cat_transaction.nVersion = 2 | ||
taproot_op_cat_transaction.vin = [ | ||
CTxIn(taproot_op_cat_outpoint)] | ||
taproot_op_cat_transaction.vout = outputs | ||
taproot_op_cat_transaction.wit.vtxinwit += [ | ||
CTxInWitness()] | ||
taproot_op_cat_transaction.wit.vtxinwit[0].scriptWitness.stack = [ | ||
op_cat_verify_script, | ||
bytes([0xC0 + taproot_op_cat.negflag]) + | ||
taproot_op_cat.internal_pubkey, | ||
] | ||
|
||
assert_equal( | ||
self.nodes[0].sendrawtransaction( | ||
taproot_op_cat_transaction.serialize().hex()), | ||
taproot_op_cat_transaction.rehash(), | ||
) | ||
self.log.info( | ||
"Taproot OP_CAT verify spend accepted by sendrawtransaction" | ||
) | ||
self.add_block([taproot_op_cat_transaction]) | ||
|
||
self.log.info("Testing Segwitv0 CAT usage") | ||
segwitv0_op_cat_transaction = CTransaction() | ||
segwitv0_op_cat_transaction.nVersion = 2 | ||
segwitv0_op_cat_transaction.vin = [ | ||
CTxIn(segwit_op_cat_outpoint)] | ||
segwitv0_op_cat_transaction.vout = outputs | ||
segwitv0_op_cat_transaction.wit.vtxinwit += [ | ||
CTxInWitness()] | ||
segwitv0_op_cat_transaction.wit.vtxinwit[0].scriptWitness.stack = [ | ||
op_cat_verify_script, | ||
] | ||
|
||
assert_raises_rpc_error( | ||
-26, | ||
DISABLED_OP_CODE, | ||
self.nodes[0].sendrawtransaction, | ||
segwitv0_op_cat_transaction.serialize().hex(), | ||
) | ||
self.log.info( | ||
"allowed by consensus, disallowed by relay policy" | ||
) | ||
|
||
|
||
if __name__ == "__main__": | ||
CatTest().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