2019-05-09 10:42:56 -04:00
#!/usr/bin/env python3
2020-12-31 09:48:25 +01:00
# Copyright (c) 2019-2020 The Bitcoin Core developers
2019-05-09 10:42:56 -04:00
# Distributed under the MIT software license, see the accompanying
# file COPYING or http://www.opensource.org/licenses/mit-license.php.
2020-06-08 18:40:18 -07:00
""" Test p2p blocksonly mode & block-relay-only connections. """
import time
2019-05-09 10:42:56 -04:00
2021-09-11 10:02:02 +02:00
from test_framework . messages import msg_tx , msg_inv , CInv , MSG_WTX
2020-06-08 18:40:18 -07:00
from test_framework . p2p import P2PInterface , P2PTxInvStore
2019-05-09 10:42:56 -04:00
from test_framework . test_framework import BitcoinTestFramework
from test_framework . util import assert_equal
2021-05-06 02:22:53 +02:00
from test_framework . wallet import MiniWallet
2019-05-09 10:42:56 -04:00
class P2PBlocksOnly ( BitcoinTestFramework ) :
def set_test_params ( self ) :
self . num_nodes = 1
self . extra_args = [ [ " -blocksonly " ] ]
def run_test ( self ) :
2021-05-06 02:22:53 +02:00
self . miniwallet = MiniWallet ( self . nodes [ 0 ] )
# Add enough mature utxos to the wallet, so that all txs spend confirmed coins
2021-09-12 11:03:56 +02:00
self . miniwallet . rescan_utxos ( )
2021-05-06 02:22:53 +02:00
2020-06-10 16:01:40 -07:00
self . blocksonly_mode_tests ( )
2020-06-08 18:40:18 -07:00
self . blocks_relay_conn_tests ( )
2020-06-10 15:46:39 -07:00
2020-06-10 16:01:40 -07:00
def blocksonly_mode_tests ( self ) :
self . log . info ( " Tests with node running in -blocksonly mode " )
2019-05-09 10:42:56 -04:00
assert_equal ( self . nodes [ 0 ] . getnetworkinfo ( ) [ ' localrelay ' ] , False )
2020-06-10 16:01:40 -07:00
self . nodes [ 0 ] . add_p2p_connection ( P2PInterface ( ) )
2021-05-06 02:22:53 +02:00
tx , txid , wtxid , tx_hex = self . check_p2p_tx_violation ( )
2019-08-28 17:23:45 -04:00
2021-09-11 10:02:02 +02:00
self . log . info ( ' Check that tx invs also violate the protocol ' )
self . nodes [ 0 ] . add_p2p_connection ( P2PInterface ( ) )
with self . nodes [ 0 ] . assert_debug_log ( [ ' transaction (0000000000000000000000000000000000000000000000000000000000001234) inv sent in violation of protocol, disconnecting peer ' ] ) :
self . nodes [ 0 ] . p2ps [ 0 ] . send_message ( msg_inv ( [ CInv ( t = MSG_WTX , h = 0x1234 ) ] ) )
self . nodes [ 0 ] . p2ps [ 0 ] . wait_for_disconnect ( )
del self . nodes [ 0 ] . p2ps [ 0 ]
2019-05-09 10:42:56 -04:00
self . log . info ( ' Check that txs from rpc are not rejected and relayed to other peers ' )
2020-06-10 16:01:40 -07:00
tx_relay_peer = self . nodes [ 0 ] . add_p2p_connection ( P2PInterface ( ) )
2019-05-09 10:42:56 -04:00
assert_equal ( self . nodes [ 0 ] . getpeerinfo ( ) [ 0 ] [ ' relaytxes ' ] , True )
2020-06-10 15:46:39 -07:00
assert_equal ( self . nodes [ 0 ] . testmempoolaccept ( [ tx_hex ] ) [ 0 ] [ ' allowed ' ] , True )
2021-09-11 10:02:02 +02:00
with self . nodes [ 0 ] . assert_debug_log ( [ ' received getdata for: wtx {} peer ' . format ( wtxid ) ] ) :
2020-06-10 15:46:39 -07:00
self . nodes [ 0 ] . sendrawtransaction ( tx_hex )
2020-09-03 18:05:26 -07:00
tx_relay_peer . wait_for_tx ( txid )
2019-05-09 10:42:56 -04:00
assert_equal ( self . nodes [ 0 ] . getmempoolinfo ( ) [ ' size ' ] , 1 )
2020-09-16 15:01:23 +02:00
self . log . info ( " Restarting node 0 with relay permission and blocksonly " )
2020-12-22 16:54:18 -08:00
self . restart_node ( 0 , [ " -persistmempool=0 " , " -whitelist=relay@127.0.0.1 " , " -blocksonly " ] )
2020-07-09 17:59:54 +02:00
assert_equal ( self . nodes [ 0 ] . getrawmempool ( ) , [ ] )
2020-04-04 20:17:06 -07:00
first_peer = self . nodes [ 0 ] . add_p2p_connection ( P2PInterface ( ) )
second_peer = self . nodes [ 0 ] . add_p2p_connection ( P2PInterface ( ) )
peer_1_info = self . nodes [ 0 ] . getpeerinfo ( ) [ 0 ]
2020-09-16 15:01:23 +02:00
assert_equal ( peer_1_info [ ' permissions ' ] , [ ' relay ' ] )
2020-04-04 20:17:06 -07:00
peer_2_info = self . nodes [ 0 ] . getpeerinfo ( ) [ 1 ]
2020-09-16 15:01:23 +02:00
assert_equal ( peer_2_info [ ' permissions ' ] , [ ' relay ' ] )
2020-06-10 15:46:39 -07:00
assert_equal ( self . nodes [ 0 ] . testmempoolaccept ( [ tx_hex ] ) [ 0 ] [ ' allowed ' ] , True )
2020-04-04 20:17:06 -07:00
2020-09-16 15:01:23 +02:00
self . log . info ( ' Check that the tx from first_peer with relay-permission is relayed to others (ie.second_peer) ' )
2020-04-04 20:17:06 -07:00
with self . nodes [ 0 ] . assert_debug_log ( [ " received getdata " ] ) :
2020-09-16 15:01:23 +02:00
# Note that normally, first_peer would never send us transactions since we're a blocksonly node.
# By activating blocksonly, we explicitly tell our peers that they should not send us transactions,
# and Bitcoin Core respects that choice and will not send transactions.
# But if, for some reason, first_peer decides to relay transactions to us anyway, we should relay them to
# second_peer since we gave relay permission to first_peer.
# See https://github.com/bitcoin/bitcoin/issues/19943 for details.
2020-06-10 15:46:39 -07:00
first_peer . send_message ( msg_tx ( tx ) )
2020-09-16 15:01:23 +02:00
self . log . info ( ' Check that the peer with relay-permission is still connected after sending the transaction ' )
2020-04-04 20:17:06 -07:00
assert_equal ( first_peer . is_connected , True )
second_peer . wait_for_tx ( txid )
assert_equal ( self . nodes [ 0 ] . getmempoolinfo ( ) [ ' size ' ] , 1 )
2020-09-16 15:01:23 +02:00
self . log . info ( " Relay-permission peer ' s transaction is accepted and relayed " )
2020-07-09 17:59:54 +02:00
2020-06-08 18:40:18 -07:00
self . nodes [ 0 ] . disconnect_p2ps ( )
2021-08-19 17:10:24 +02:00
self . generate ( self . nodes [ 0 ] , 1 )
2020-06-08 18:40:18 -07:00
def blocks_relay_conn_tests ( self ) :
self . log . info ( ' Tests with node in normal mode with block-relay-only connections ' )
self . restart_node ( 0 , [ " -noblocksonly " ] ) # disables blocks only mode
assert_equal ( self . nodes [ 0 ] . getnetworkinfo ( ) [ ' localrelay ' ] , True )
# Ensure we disconnect if a block-relay-only connection sends us a transaction
self . nodes [ 0 ] . add_outbound_p2p_connection ( P2PInterface ( ) , p2p_idx = 0 , connection_type = " block-relay-only " )
assert_equal ( self . nodes [ 0 ] . getpeerinfo ( ) [ 0 ] [ ' relaytxes ' ] , False )
2021-09-12 11:03:56 +02:00
_ , txid , _ , tx_hex = self . check_p2p_tx_violation ( )
2020-06-08 18:40:18 -07:00
self . log . info ( " Check that txs from RPC are not sent to blockrelay connection " )
conn = self . nodes [ 0 ] . add_outbound_p2p_connection ( P2PTxInvStore ( ) , p2p_idx = 1 , connection_type = " block-relay-only " )
2020-06-10 16:01:40 -07:00
2020-06-08 18:40:18 -07:00
self . nodes [ 0 ] . sendrawtransaction ( tx_hex )
# Bump time forward to ensure nNextInvSend timer pops
self . nodes [ 0 ] . setmocktime ( int ( time . time ( ) ) + 60 )
2021-04-27 13:32:23 +02:00
conn . sync_send_with_ping ( )
2020-06-08 18:40:18 -07:00
assert ( int ( txid , 16 ) not in conn . get_invs ( ) )
2021-09-12 11:03:56 +02:00
def check_p2p_tx_violation ( self ) :
2020-06-08 18:40:18 -07:00
self . log . info ( ' Check that txs from P2P are rejected and result in disconnect ' )
2021-09-12 11:03:56 +02:00
spendtx = self . miniwallet . create_self_transfer ( from_node = self . nodes [ 0 ] )
2020-06-10 16:01:40 -07:00
with self . nodes [ 0 ] . assert_debug_log ( [ ' transaction sent in violation of protocol peer=0 ' ] ) :
2021-05-06 02:22:53 +02:00
self . nodes [ 0 ] . p2ps [ 0 ] . send_message ( msg_tx ( spendtx [ ' tx ' ] ) )
2020-06-10 16:01:40 -07:00
self . nodes [ 0 ] . p2ps [ 0 ] . wait_for_disconnect ( )
assert_equal ( self . nodes [ 0 ] . getmempoolinfo ( ) [ ' size ' ] , 0 )
# Remove the disconnected peer
del self . nodes [ 0 ] . p2ps [ 0 ]
2021-05-06 02:22:53 +02:00
return spendtx [ ' tx ' ] , spendtx [ ' txid ' ] , spendtx [ ' wtxid ' ] , spendtx [ ' hex ' ]
2020-06-10 16:01:40 -07:00
2019-05-09 10:42:56 -04:00
if __name__ == ' __main__ ' :
P2PBlocksOnly ( ) . main ( )