2019-06-21 00:15:26 -04:00
#!/usr/bin/env python3
2022-12-24 20:49:50 -03:00
# Copyright (c) 2015-2022 The Bitcoin Core developers
2019-06-21 00:15:26 -04:00
# Distributed under the MIT software license, see the accompanying
# file COPYING or http://www.opensource.org/licenses/mit-license.php.
""" Test p2p permission message.
Test that permissions are correctly calculated and applied
"""
2020-01-22 17:27:14 -03:00
from test_framework . messages import (
2023-01-14 21:02:27 -03:00
SEQUENCE_FINAL ,
2020-01-22 17:27:14 -03:00
)
2020-07-19 03:47:05 -04:00
from test_framework . p2p import P2PDataStore
2019-06-21 00:15:26 -04:00
from test_framework . test_node import ErrorMatch
from test_framework . test_framework import BitcoinTestFramework
from test_framework . util import (
2024-09-03 16:20:49 -04:00
append_config ,
2019-06-21 00:15:26 -04:00
assert_equal ,
p2p_port ,
2024-09-03 16:20:49 -04:00
tor_port ,
2019-06-21 00:15:26 -04:00
)
2023-01-14 21:02:27 -03:00
from test_framework . wallet import MiniWallet
2019-06-21 00:15:26 -04:00
2020-01-22 17:27:14 -03:00
2019-06-21 00:15:26 -04:00
class P2PPermissionsTests ( BitcoinTestFramework ) :
def set_test_params ( self ) :
self . num_nodes = 2
def run_test ( self ) :
2023-01-14 21:02:27 -03:00
self . wallet = MiniWallet ( self . nodes [ 0 ] )
2020-01-22 17:27:14 -03:00
self . check_tx_relay ( )
2019-06-21 00:15:26 -04:00
self . checkpermission (
2019-08-16 09:56:56 -04:00
# default permissions (no specific permissions)
[ " -whitelist=127.0.0.1 " ] ,
2020-06-06 11:07:25 -04:00
# Make sure the default values in the command line documentation match the ones here
2020-12-22 21:54:18 -03:00
[ " relay " , " noban " , " mempool " , " download " ] )
2020-08-20 14:42:54 -04:00
2020-06-07 09:24:18 -04:00
self . checkpermission (
# no permission (even with forcerelay)
[ " -whitelist=@127.0.0.1 " , " -whitelistforcerelay=1 " ] ,
2020-12-22 21:54:18 -03:00
[ ] )
2020-06-07 09:24:18 -04:00
2019-08-16 04:45:13 -04:00
self . checkpermission (
2019-08-16 09:56:56 -04:00
# relay permission removed (no specific permissions)
[ " -whitelist=127.0.0.1 " , " -whitelistrelay=0 " ] ,
2020-12-22 21:54:18 -03:00
[ " noban " , " mempool " , " download " ] )
2019-08-16 04:45:13 -04:00
2019-06-21 00:15:26 -04:00
self . checkpermission (
2019-08-16 09:56:56 -04:00
# forcerelay and relay permission added
# Legacy parameter interaction which set whitelistrelay to true
# if whitelistforcerelay is true
[ " -whitelist=127.0.0.1 " , " -whitelistforcerelay " ] ,
2020-12-22 21:54:18 -03:00
[ " forcerelay " , " relay " , " noban " , " mempool " , " download " ] )
2019-06-21 00:15:26 -04:00
# Let's make sure permissions are merged correctly
# For this, we need to use whitebind instead of bind
# by modifying the configuration file.
ip_port = " 127.0.0.1: {} " . format ( p2p_port ( 1 ) )
2023-01-23 13:35:04 -03:00
self . nodes [ 1 ] . replace_in_config ( [ ( " bind=127.0.0.1 " , " whitebind=bloomfilter,forcerelay@ " + ip_port ) ] )
2024-09-03 16:20:49 -04:00
# Explicitly bind the tor port to prevent collisions with the default tor port
append_config ( self . nodes [ 1 ] . datadir_path , [ f " bind=127.0.0.1: { tor_port ( self . nodes [ 1 ] . index ) } =onion " ] )
2019-06-21 00:15:26 -04:00
self . checkpermission (
2020-01-22 17:36:44 -03:00
[ " -whitelist=noban@127.0.0.1 " ] ,
2019-08-16 09:56:56 -04:00
# Check parameter interaction forcerelay should activate relay
2020-12-22 21:54:18 -03:00
[ " noban " , " bloomfilter " , " forcerelay " , " relay " , " download " ] )
2023-01-23 13:35:04 -03:00
self . nodes [ 1 ] . replace_in_config ( [ ( " whitebind=bloomfilter,forcerelay@ " + ip_port , " bind=127.0.0.1 " ) ] )
2024-09-03 16:20:49 -04:00
self . nodes [ 1 ] . replace_in_config ( [ ( f " bind=127.0.0.1: { tor_port ( self . nodes [ 1 ] . index ) } =onion " , " " ) ] )
2019-06-21 00:15:26 -04:00
self . checkpermission (
2019-08-16 09:56:56 -04:00
# legacy whitelistrelay should be ignored
[ " -whitelist=noban,mempool@127.0.0.1 " , " -whitelistrelay " ] ,
2020-12-22 21:54:18 -03:00
[ " noban " , " mempool " , " download " ] )
2020-08-20 14:42:54 -04:00
2019-06-21 00:15:26 -04:00
self . checkpermission (
2019-08-16 09:56:56 -04:00
# legacy whitelistforcerelay should be ignored
[ " -whitelist=noban,mempool@127.0.0.1 " , " -whitelistforcerelay " ] ,
2020-12-22 21:54:18 -03:00
[ " noban " , " mempool " , " download " ] )
2019-06-21 00:15:26 -04:00
self . checkpermission (
2019-08-16 09:56:56 -04:00
# missing mempool permission to be considered legacy whitelisted
[ " -whitelist=noban@127.0.0.1 " ] ,
2020-12-22 21:54:18 -03:00
[ " noban " , " download " ] )
2019-06-21 00:15:26 -04:00
self . checkpermission (
2019-08-16 09:56:56 -04:00
# all permission added
[ " -whitelist=all@127.0.0.1 " ] ,
2020-12-22 21:54:18 -03:00
[ " forcerelay " , " noban " , " mempool " , " bloomfilter " , " relay " , " download " , " addr " ] )
2019-06-21 00:15:26 -04:00
2023-08-07 16:57:18 -04:00
for flag , permissions in [ ( [ " -whitelist=noban,out@127.0.0.1 " ] , [ " noban " , " download " ] ) , ( [ " -whitelist=noban@127.0.0.1 " ] , [ ] ) ] :
self . restart_node ( 0 , flag )
self . connect_nodes ( 0 , 1 )
peerinfo = self . nodes [ 0 ] . getpeerinfo ( ) [ 0 ]
assert_equal ( peerinfo [ ' permissions ' ] , permissions )
2019-06-21 00:15:26 -04:00
self . stop_node ( 1 )
2023-08-07 16:57:18 -04:00
self . nodes [ 1 ] . assert_start_raises_init_error ( [ " -whitelist=in,out@127.0.0.1 " ] , " Only direction was set, no permissions " , match = ErrorMatch . PARTIAL_REGEX )
2019-06-21 00:15:26 -04:00
self . nodes [ 1 ] . assert_start_raises_init_error ( [ " -whitelist=oopsie@127.0.0.1 " ] , " Invalid P2P permission " , match = ErrorMatch . PARTIAL_REGEX )
self . nodes [ 1 ] . assert_start_raises_init_error ( [ " -whitelist=noban@127.0.0.1:230 " ] , " Invalid netmask specified in " , match = ErrorMatch . PARTIAL_REGEX )
self . nodes [ 1 ] . assert_start_raises_init_error ( [ " -whitebind=noban@127.0.0.1/10 " ] , " Cannot resolve -whitebind address " , match = ErrorMatch . PARTIAL_REGEX )
2022-06-22 14:22:25 -04:00
self . nodes [ 1 ] . assert_start_raises_init_error ( [ " -whitebind=noban@127.0.0.1 " , " -bind=127.0.0.1 " , " -listen=0 " ] , " Cannot set -bind or -whitebind together with -listen=0 " , match = ErrorMatch . PARTIAL_REGEX )
2019-06-21 00:15:26 -04:00
2020-01-22 17:27:14 -03:00
def check_tx_relay ( self ) :
2020-07-09 11:59:54 -04:00
self . log . debug ( " Create a connection from a forcerelay peer that rebroadcasts raw txs " )
2020-08-17 05:10:44 -04:00
# A test framework p2p connection is needed to send the raw transaction directly. If a full node was used, it could only
2020-07-09 11:59:54 -04:00
# rebroadcast via the inv-getdata mechanism. However, even for forcerelay connections, a full node would
2020-01-22 17:27:14 -03:00
# currently not request a txid that is already in the mempool.
self . restart_node ( 1 , extra_args = [ " -whitelist=forcerelay@127.0.0.1 " ] )
p2p_rebroadcast_wallet = self . nodes [ 1 ] . add_p2p_connection ( P2PDataStore ( ) )
self . log . debug ( " Send a tx from the wallet initially " )
2023-01-14 21:02:27 -03:00
tx = self . wallet . create_self_transfer ( sequence = SEQUENCE_FINAL ) [ ' tx ' ]
2020-01-22 17:27:14 -03:00
txid = tx . rehash ( )
self . log . debug ( " Wait until tx is in node[1] ' s mempool " )
p2p_rebroadcast_wallet . send_txs_and_test ( [ tx ] , self . nodes [ 1 ] )
self . log . debug ( " Check that node[1] will send the tx to node[0] even though it is already in the mempool " )
2020-09-17 04:46:07 -03:00
self . connect_nodes ( 1 , 0 )
2023-08-09 09:25:34 -04:00
with self . nodes [ 1 ] . assert_debug_log ( [ " Force relaying tx {} (wtxid= {} ) from peer=0 " . format ( txid , tx . getwtxid ( ) ) ] ) :
2020-01-22 17:27:14 -03:00
p2p_rebroadcast_wallet . send_txs_and_test ( [ tx ] , self . nodes [ 1 ] )
2020-08-17 11:50:47 -04:00
self . wait_until ( lambda : txid in self . nodes [ 0 ] . getrawmempool ( ) )
2020-01-22 17:27:14 -03:00
2020-01-22 18:02:24 -03:00
self . log . debug ( " Check that node[1] will not send an invalid tx to node[0] " )
tx . vout [ 0 ] . nValue + = 1
2023-08-04 09:56:04 -04:00
# add dust to cause policy rejection but no disconnection
tx . vout . append ( tx . vout [ 0 ] )
tx . vout [ - 1 ] . nValue = 0
2020-01-22 18:02:24 -03:00
txid = tx . rehash ( )
2020-08-12 11:04:29 -04:00
# Send the transaction twice. The first time, it'll be rejected by ATMP because it conflicts
2024-07-31 08:19:28 -04:00
# with a mempool transaction. The second time, it'll be in the m_lazy_recent_rejects filter.
2020-01-22 18:02:24 -03:00
p2p_rebroadcast_wallet . send_txs_and_test (
[ tx ] ,
self . nodes [ 1 ] ,
success = False ,
2023-08-04 09:56:04 -04:00
reject_reason = ' {} (wtxid= {} ) from peer=0 was not accepted: dust ' . format ( txid , tx . getwtxid ( ) )
2020-08-12 11:04:29 -04:00
)
p2p_rebroadcast_wallet . send_txs_and_test (
[ tx ] ,
self . nodes [ 1 ] ,
success = False ,
2023-08-09 09:25:34 -04:00
reject_reason = ' Not relaying non-mempool transaction {} (wtxid= {} ) from forcerelay peer=0 ' . format ( txid , tx . getwtxid ( ) )
2020-01-22 18:02:24 -03:00
)
2020-12-22 21:54:18 -03:00
def checkpermission ( self , args , expectedPermissions ) :
2019-06-21 00:15:26 -04:00
self . restart_node ( 1 , args )
2020-09-17 04:46:07 -03:00
self . connect_nodes ( 0 , 1 )
2019-06-21 00:15:26 -04:00
peerinfo = self . nodes [ 1 ] . getpeerinfo ( ) [ 0 ]
assert_equal ( len ( expectedPermissions ) , len ( peerinfo [ ' permissions ' ] ) )
for p in expectedPermissions :
2020-12-22 21:54:18 -03:00
if p not in peerinfo [ ' permissions ' ] :
2019-06-21 00:15:26 -04:00
raise AssertionError ( " Expected permissions %r is not granted. " % p )
2020-01-22 17:36:44 -03:00
2019-06-21 00:15:26 -04:00
if __name__ == ' __main__ ' :
2024-07-16 17:05:14 -04:00
P2PPermissionsTests ( __file__ ) . main ( )