0
0
mirror of https://github.com/darkk/redsocks.git synced 2025-08-25 19:25:30 +00:00
redsocks/tests/test_smoke.py
Leonid Evdokimov 9ef1950dd1 Fix typo
2016-08-19 16:20:22 +03:00

193 lines
7.8 KiB
Python

from functools import partial
from subprocess import check_call, CalledProcessError, Popen, PIPE
import time
import conftest
import pytest
@pytest.mark.skipif(not pytest.config.getoption('--vmdebug'), reason='need --vmdebug option to run')
def test_vmdebug(net):
check_call('sleep 365d'.split())
GOOD_AUTH = 'connect_none connect_basic connect_digest socks5_none socks5_auth httperr_connect_digest'.split()
BAD_AUTH = 'connect_nopass connect_baduser connect_badpass socks5_nopass socks5_baduser socks5_badpass httperr_proxy_refuse'.split()
UGLY_AUTH = 'httperr_connect_nopass httperr_connect_baduser httperr_connect_badpass'.split()
assert set(conftest.TANKS) == set(GOOD_AUTH + BAD_AUTH + UGLY_AUTH + ['httperr_proxy_timeout'])
@pytest.mark.parametrize('tank', GOOD_AUTH)
def test_smoke(net, tank):
vm = net.vm['tank%d' % conftest.TANKS[tank]]
page = vm.do('curl --max-time 0.5 http://10.0.1.80/')
assert 'Welcome to nginx!' in page
@pytest.mark.parametrize('tank', BAD_AUTH)
def test_badauth(net, tank):
vm = net.vm['tank%d' % conftest.TANKS[tank]]
with pytest.raises(CalledProcessError) as excinfo:
vm.do('curl --max-time 0.5 http://10.0.1.80/')
assert excinfo.value.returncode == 52 # Empty reply from server
@pytest.mark.parametrize('tank', UGLY_AUTH)
def test_uglyauth(net, tank):
vm = net.vm['tank%d' % conftest.TANKS[tank]]
page = vm.do('curl -sSv --max-time 0.5 http://10.0.1.80/')
assert '<!-- ERR_CACHE_ACCESS_DENIED -->' in page
@pytest.mark.parametrize('tank', set(conftest.TANKS) - set(UGLY_AUTH + ['httperr_connect_digest']))
def test_econnrefused(net, tank):
vm = net.vm['tank%d' % conftest.TANKS[tank]]
with pytest.raises(CalledProcessError) as excinfo:
vm.do('curl --max-time 0.5 http://10.0.1.80:81/')
if tank == 'httperr_proxy_timeout':
assert excinfo.value.returncode == 28 # Operation timed out
else:
assert excinfo.value.returncode == 52 # Empty reply from server
def test_econnrefused_httperr(net):
tank = 'httperr_connect_digest'
vm = net.vm['tank%d' % conftest.TANKS[tank]]
page = vm.do('curl --max-time 0.5 http://10.0.1.80:81/')
assert '<!-- ERR_CONNECT_FAIL -->' in page
RTT = 200 # ms
@pytest.fixture(scope="function")
def slow_net(request, net):
def close():
net.vm['gw'].netcall('tc qdisc del dev ethx root')
net.vm['gw'].netcall('tc qdisc del dev ethw root')
request.addfinalizer(close)
net.vm['gw'].netcall('tc qdisc add dev ethw root netem delay %dms' % (RTT / 2))
net.vm['gw'].netcall('tc qdisc add dev ethx root netem delay %dms' % (RTT / 2))
return net
LATENCY = {
'connect_none': 3 * RTT,
'connect_basic': 3 * RTT,
'connect_digest': 3 * RTT,
'socks5_none': 4 * RTT,
'socks5_auth': 5 * RTT,
'regw_direct': 2 * RTT,
}
def heatup(vm):
vm.do('curl -o /dev/null http://10.0.1.80/') # heatup L2 and auth caches
def http_ping(vm):
s = vm.do('curl -sS -w %{{time_connect}}/%{{time_total}}/%{{http_code}}/%{{size_download}} -o /dev/null http://10.0.1.80/')
connect, total, code, size = s.split('/')
connect, total, code, size = float(connect) * 1000, float(total) * 1000, int(code), int(size)
return connect, total, code, size
@pytest.mark.parametrize('tank', set(conftest.TANKS) & set(LATENCY))
def test_latency_tank(slow_net, tank):
vm = slow_net.vm['tank%d' % conftest.TANKS[tank]]
heatup(vm)
connect, total, code, size = http_ping(vm)
assert code == 200 and size == 612
assert connect < 0.005 and LATENCY[tank]-RTT*.2 < total and total < LATENCY[tank]+RTT*.2
def test_latency_regw(slow_net):
vm, tank = slow_net.vm['regw'], 'regw_direct'
heatup(vm)
connect, total, code, size = http_ping(vm)
assert code == 200 and size == 612
assert RTT*.8 < connect and connect < RTT*1.2 and LATENCY[tank]-RTT*.2 < total and total < LATENCY[tank]+RTT*.2
def test_nonce_reuse(slow_net):
""" nonce reuse works and has no latency penalty """
tank = 'connect_digest'
vm = slow_net.vm['tank%d' % conftest.TANKS[tank]]
heatup(vm)
begin = time.time()
s = conftest.pmap([partial(http_ping, vm) for _ in range(5)])
total_sum = time.time() - begin
for connect, total, code, size in s:
assert code == 200 and size == 612
assert connect < 0.005 and LATENCY[tank]-RTT*.2 < total and total < LATENCY[tank]+RTT*.2
assert total_sum < total * 1.5
@pytest.mark.parametrize('tank, delay', [(t, d)
for t in set(conftest.TANKS) & set(LATENCY)
for d in [_*0.001 for _ in range(1, LATENCY[t]+RTT, RTT/2)]
] + [
('httperr_proxy_refuse', (1 + 0*RTT/2) * 0.001),
('httperr_proxy_refuse', (1 + 1*RTT/2) * 0.001),
('httperr_proxy_refuse', (1 + 2*RTT/2) * 0.001),
('httperr_proxy_refuse', (1 + 3*RTT/2) * 0.001),
('httperr_proxy_timeout', (1 + 3*RTT/2) * 0.001),
])
def test_impatient_client(slow_net, tank, delay):
vm, regw = slow_net.vm['tank%d' % conftest.TANKS[tank]], slow_net.vm['regw']
before, start = regw.lsfd(), time.time()
try:
page = vm.do('curl --max-time %s http://10.0.1.80/1M' % delay)
#assert 'Welcome to nginx!' in page
except CalledProcessError, e:
if tank == 'httperr_proxy_refuse':
assert e.returncode in (28, 52) # Operation timeout / Empty reply
else:
assert e.returncode == 28 # Operation timeout
curl_time = time.time() - start
assert curl_time < delay + RTT*0.001 # sanity check
time.sleep( (LATENCY.get(tank, delay*1000) + 4*RTT)*0.001 )
if tank == 'httperr_proxy_timeout':
time.sleep(135) # default connect() timeout is long
assert before == regw.lsfd() # no leaks
@pytest.fixture(scope='function', params=range(80, 60, -1))
def lowfd_net(request, net):
def close():
net.vm['regw'].call('sudo ./prlimit-nofile {pid} 1024')
request.addfinalizer(close)
net.vm['regw'].call('sudo ./prlimit-nofile {pid} %d' % request.param)
return net
def test_accept_overflow(lowfd_net):
tank = 'connect_none'
vm, regw = lowfd_net.vm['tank%d' % conftest.TANKS[tank]], lowfd_net.vm['regw']
lsfd = regw.lsfd()
year = str(time.gmtime().tm_year)
discard_cmd = vm.fmt('sudo docker exec --interactive {sha} nc 10.0.1.13 discard')
daytime_cmd = vm.fmt('sudo docker exec --interactive {sha} nc 10.0.1.13 daytime')
dtstart = time.time()
time.sleep(0.5)
proc = [Popen(discard_cmd, stdin=PIPE, stdout=PIPE) for i in xrange(7)]
time.sleep(0.5)
dt = Popen(daytime_cmd, stdin=PIPE, stdout=PIPE)
time.sleep(0.5)
logs = regw.logs(since=dtstart)
if any(['Too many open files' in _[1] for _ in logs]):
# anything may happen, except leaks
proc[0].communicate('/dev/null\x0d\x0a')
dt.communicate('')
time.sleep(1)
for p in proc[1:]:
p.communicate('/dev/null\x0d\x0a')
else:
assert any(['reached redsocks_conn_max limit' in _[1] for _ in logs])
# RLIMIT_NOFILE was not hit, this `daytime` call actually returns data
daytime, _ = dt.communicate('')
assert year in daytime and dt.returncode == 0
time.sleep(0.5)
dtdeath = time.time()
proc.append(Popen(discard_cmd, stdin=PIPE, stdout=PIPE))
time.sleep(0.5)
logs = regw.logs(since=dtdeath)
assert any(['reached redsocks_conn_max limit' in _[1] for _ in logs])
time.sleep(0.5)
dt = Popen(daytime_cmd, stdin=PIPE, stdout=PIPE)
time.sleep(0.5)
assert all([p.poll() is None for p in proc]) and dt.poll() is None # processes are running
out, _ = proc[0].communicate('/dev/null\x0d\x0a')
daytime, _ = dt.communicate('')
assert year in daytime and dt.returncode == 0
time.sleep(1)
for p in proc[1:]:
out, _ = p.communicate('/dev/null\x0d\x0a')
assert out == ''
time.sleep(1)
assert lsfd == regw.lsfd() # no leaks