2016-03-19 20:58:06 +01:00
#!/usr/bin/env python3
2018-07-26 18:36:45 -04:00
# Copyright (c) 2014-2018 The Bitcoin Core developers
2014-10-23 09:48:19 +08:00
# Distributed under the MIT software license, see the accompanying
2014-04-07 17:29:36 +02:00
# file COPYING or http://www.opensource.org/licenses/mit-license.php.
2017-01-17 18:34:40 -05:00
""" Test running bitcoind with the -rpcbind and -rpcallowip options. """
2014-04-07 17:29:36 +02:00
2017-03-22 10:29:39 -04:00
import sys
2017-05-18 16:36:39 -04:00
from test_framework . test_framework import BitcoinTestFramework , SkipTest
2015-05-02 12:53:35 +02:00
from test_framework . util import *
from test_framework . netutil import *
2014-04-07 17:29:36 +02:00
2016-07-25 01:31:05 +03:00
class RPCBindTest ( BitcoinTestFramework ) :
2017-06-09 18:21:21 -04:00
def set_test_params ( self ) :
2016-07-25 01:31:05 +03:00
self . setup_clean_chain = True
2018-03-06 16:48:15 -05:00
self . bind_to_localhost_only = False
2016-07-25 01:31:05 +03:00
self . num_nodes = 1
2014-06-27 10:05:46 +02:00
2016-07-25 01:31:05 +03:00
def setup_network ( self ) :
2017-08-24 11:37:25 -04:00
self . add_nodes ( self . num_nodes , None )
2014-04-07 17:29:36 +02:00
2018-02-22 15:43:26 +01:00
def add_options ( self , parser ) :
parser . add_option ( " --ipv4 " , action = ' store_true ' , dest = " run_ipv4 " , help = " Run ipv4 tests only " , default = False )
parser . add_option ( " --ipv6 " , action = ' store_true ' , dest = " run_ipv6 " , help = " Run ipv6 tests only " , default = False )
parser . add_option ( " --nonloopback " , action = ' store_true ' , dest = " run_nonloopback " , help = " Run non-loopback tests only " , default = False )
2016-07-25 01:31:05 +03:00
def run_bind_test ( self , allow_ips , connect_to , addresses , expected ) :
'''
Start a node with requested rpcallowip and rpcbind parameters ,
then try to connect , and check if the set of bound addresses
matches the expected set .
'''
2017-06-09 16:35:17 -04:00
self . log . info ( " Bind test for %s " % str ( addresses ) )
2016-07-25 01:31:05 +03:00
expected = [ ( addr_to_hex ( addr ) , port ) for ( addr , port ) in expected ]
base_args = [ ' -disablewallet ' , ' -nolisten ' ]
if allow_ips :
base_args + = [ ' -rpcallowip= ' + x for x in allow_ips ]
binds = [ ' -rpcbind= ' + addr for addr in addresses ]
2017-06-09 16:35:17 -04:00
self . nodes [ 0 ] . rpchost = connect_to
self . start_node ( 0 , base_args + binds )
2017-06-02 14:30:36 -04:00
pid = self . nodes [ 0 ] . process . pid
2017-02-22 11:29:43 +01:00
assert_equal ( set ( get_bind_addrs ( pid ) ) , set ( expected ) )
2017-03-23 23:56:31 -04:00
self . stop_nodes ( )
2016-07-25 01:31:05 +03:00
def run_allowip_test ( self , allow_ips , rpchost , rpcport ) :
'''
2016-09-25 15:01:31 +02:00
Start a node with rpcallow IP , and request getnetworkinfo
2016-07-25 01:31:05 +03:00
at a non - localhost IP .
'''
2017-06-09 16:35:17 -04:00
self . log . info ( " Allow IP test for %s : %d " % ( rpchost , rpcport ) )
2016-07-25 01:31:05 +03:00
base_args = [ ' -disablewallet ' , ' -nolisten ' ] + [ ' -rpcallowip= ' + x for x in allow_ips ]
2017-06-09 16:35:17 -04:00
self . nodes [ 0 ] . rpchost = None
self . start_nodes ( [ base_args ] )
2017-02-22 11:29:43 +01:00
# connect to node through non-loopback interface
2018-01-02 08:57:27 -05:00
node = get_rpc_proxy ( rpc_url ( self . nodes [ 0 ] . datadir , 0 , " %s : %d " % ( rpchost , rpcport ) ) , 0 , coveragedir = self . options . coveragedir )
2017-02-22 11:29:43 +01:00
node . getnetworkinfo ( )
2017-03-23 23:56:31 -04:00
self . stop_nodes ( )
2016-07-25 01:31:05 +03:00
def run_test ( self ) :
# due to OS-specific network stats queries, this test works only on Linux
2018-02-22 15:43:26 +01:00
if sum ( [ self . options . run_ipv4 , self . options . run_ipv6 , self . options . run_nonloopback ] ) > 1 :
raise AssertionError ( " Only one of --ipv4, --ipv6 and --nonloopback can be set " )
self . log . info ( " Check for linux " )
2017-03-22 10:29:39 -04:00
if not sys . platform . startswith ( ' linux ' ) :
2018-02-22 15:43:26 +01:00
raise SkipTest ( " This test can only be run on linux. " )
self . log . info ( " Check for ipv6 " )
have_ipv6 = test_ipv6_local ( )
if not have_ipv6 and not self . options . run_ipv4 :
raise SkipTest ( " This test requires ipv6 support. " )
self . log . info ( " Check for non-loopback interface " )
self . non_loopback_ip = None
2016-07-25 01:31:05 +03:00
for name , ip in all_interfaces ( ) :
if ip != ' 127.0.0.1 ' :
2018-02-22 15:43:26 +01:00
self . non_loopback_ip = ip
2016-07-25 01:31:05 +03:00
break
2018-02-22 15:43:26 +01:00
if self . non_loopback_ip is None and self . options . run_nonloopback :
raise SkipTest ( " This test requires a non-loopback ip address. " )
self . defaultport = rpc_port ( 0 )
if not self . options . run_nonloopback :
self . _run_loopback_tests ( )
if not self . options . run_ipv4 and not self . options . run_ipv6 :
self . _run_nonloopback_tests ( )
def _run_loopback_tests ( self ) :
if self . options . run_ipv4 :
# check only IPv4 localhost (explicit)
self . run_bind_test ( [ ' 127.0.0.1 ' ] , ' 127.0.0.1 ' , [ ' 127.0.0.1 ' ] ,
[ ( ' 127.0.0.1 ' , self . defaultport ) ] )
# check only IPv4 localhost (explicit) with alternative port
self . run_bind_test ( [ ' 127.0.0.1 ' ] , ' 127.0.0.1:32171 ' , [ ' 127.0.0.1:32171 ' ] ,
[ ( ' 127.0.0.1 ' , 32171 ) ] )
# check only IPv4 localhost (explicit) with multiple alternative ports on same host
self . run_bind_test ( [ ' 127.0.0.1 ' ] , ' 127.0.0.1:32171 ' , [ ' 127.0.0.1:32171 ' , ' 127.0.0.1:32172 ' ] ,
[ ( ' 127.0.0.1 ' , 32171 ) , ( ' 127.0.0.1 ' , 32172 ) ] )
else :
# check default without rpcallowip (IPv4 and IPv6 localhost)
self . run_bind_test ( None , ' 127.0.0.1 ' , [ ] ,
[ ( ' 127.0.0.1 ' , self . defaultport ) , ( ' ::1 ' , self . defaultport ) ] )
# check default with rpcallowip (IPv6 any)
self . run_bind_test ( [ ' 127.0.0.1 ' ] , ' 127.0.0.1 ' , [ ] ,
[ ( ' ::0 ' , self . defaultport ) ] )
# check only IPv6 localhost (explicit)
self . run_bind_test ( [ ' [::1] ' ] , ' [::1] ' , [ ' [::1] ' ] ,
[ ( ' ::1 ' , self . defaultport ) ] )
# check both IPv4 and IPv6 localhost (explicit)
self . run_bind_test ( [ ' 127.0.0.1 ' ] , ' 127.0.0.1 ' , [ ' 127.0.0.1 ' , ' [::1] ' ] ,
[ ( ' 127.0.0.1 ' , self . defaultport ) , ( ' ::1 ' , self . defaultport ) ] )
def _run_nonloopback_tests ( self ) :
self . log . info ( " Using interface %s for testing " % self . non_loopback_ip )
2016-07-25 01:31:05 +03:00
# check only non-loopback interface
2018-02-22 15:43:26 +01:00
self . run_bind_test ( [ self . non_loopback_ip ] , self . non_loopback_ip , [ self . non_loopback_ip ] ,
[ ( self . non_loopback_ip , self . defaultport ) ] )
2016-07-25 01:31:05 +03:00
# Check that with invalid rpcallowip, we are denied
2018-02-22 15:43:26 +01:00
self . run_allowip_test ( [ self . non_loopback_ip ] , self . non_loopback_ip , self . defaultport )
assert_raises_rpc_error ( - 342 , " non-JSON HTTP response with ' 403 Forbidden ' from server " , self . run_allowip_test , [ ' 1.1.1.1 ' ] , self . non_loopback_ip , self . defaultport )
2014-04-07 17:29:36 +02:00
if __name__ == ' __main__ ' :
2016-08-19 22:41:09 +02:00
RPCBindTest ( ) . main ( )