1
0
mirror of https://github.com/samba-team/samba.git synced 2025-01-11 05:18:09 +03:00

python: add a failed test to show Popen deadlock

`Popen.wait()` will deadlock when using stdout=PIPE and/or stderr=PIPE and the
child process generates large output to a pipe such that it blocks waiting for
the OS pipe buffer to accept more data. Use communicate() to avoid that.

This patch is commited to show the issue, a fix patch will come later.

Signed-off-by: Joe Guo <joeg@catalyst.net.nz>
Reviewed-by: Douglas Bagnall <douglas.bagnall@catalyst.net.nz>
Reviewed-by: Andrew Bartlett <abartlet@samba.org>
This commit is contained in:
Joe Guo 2017-09-15 15:31:34 +12:00 committed by Andrew Bartlett
parent ea0cd977a3
commit 8ed3cac9e5
5 changed files with 161 additions and 1 deletions

View File

@ -0,0 +1,105 @@
# Copyright (C) Catalyst IT Ltd. 2017
# This program is free software; you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation; either version 3 of the License, or
# (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program. If not, see <http://www.gnu.org/licenses/>.
"""
Blackbox tests for blackboxtest check output methods.
"""
import time
import signal
from samba.tests import BlackboxTestCase
class TimeoutHelper():
"""
Timeout class using alarm signal.
Raise a Timeout exception if a function timeout.
Usage:
try:
with Timeout(3):
foobar("Request 1")
except TimeoutHelper.Timeout:
print "Timeout"
"""
class Timeout(Exception):
pass
def __init__(self, sec):
self.sec = sec
def __enter__(self):
signal.signal(signal.SIGALRM, self.raise_timeout)
signal.alarm(self.sec)
def __exit__(self, *args):
signal.alarm(0) # disable alarm
def raise_timeout(self, *args):
raise TimeoutHelper.Timeout()
def _make_cmdline(data='$', repeat=5*1024*1024, retcode=0):
"""Build a command to call gen_output.py to generate large output"""
return 'gen_output.py --data {} --repeat {} --retcode {}'.format(data, repeat, retcode)
class CheckOutputTests(BlackboxTestCase):
"""
Blackbox tests for check_xxx methods.
The check_xxx methods in BlackboxTestCase will deadlock
on large output from command which caused by Popen.wait().
This is a test case to show the deadlock issue,
will fix in another commit.
"""
def test_check_run_timeout(self):
"""Call check_run with large output."""
try:
with TimeoutHelper(10):
self.check_run(_make_cmdline())
except TimeoutHelper.Timeout:
self.fail(msg='Timeout!')
def test_check_exit_code_with_large_output_success(self):
try:
with TimeoutHelper(10):
self.check_exit_code(_make_cmdline(retcode=0), 0)
except TimeoutHelper.Timeout:
self.fail(msg='Timeout!')
def test_check_exit_code_with_large_output_failure(self):
try:
with TimeoutHelper(10):
self.check_exit_code(_make_cmdline(retcode=1), 1)
except TimeoutHelper.Timeout:
self.fail(msg='Timeout!')
def test_check_output_with_large_output(self):
data = '@'
repeat = 5 * 1024 * 1024 # 5M
expected = data * repeat
cmdline = _make_cmdline(data=data, repeat=repeat)
try:
with TimeoutHelper(10):
actual = self.check_output(cmdline)
self.assertEqual(actual, expected)
except TimeoutHelper.Timeout:
self.fail(msg='Timeout!')

View File

@ -0,0 +1 @@
^samba.tests.blackbox.check_output

View File

@ -53,6 +53,7 @@ except ImportError:
else: else:
planpythontestsuite("none", "subunit.tests.test_suite") planpythontestsuite("none", "subunit.tests.test_suite")
planpythontestsuite("none", "samba.tests.blackbox.ndrdump") planpythontestsuite("none", "samba.tests.blackbox.ndrdump")
planpythontestsuite("none", "samba.tests.blackbox.check_output")
planpythontestsuite("none", "api", name="ldb.python", extra_path=['lib/ldb/tests/python']) planpythontestsuite("none", "api", name="ldb.python", extra_path=['lib/ldb/tests/python'])
planpythontestsuite("none", "samba.tests.credentials", py3_compatible=True) planpythontestsuite("none", "samba.tests.credentials", py3_compatible=True)
planpythontestsuite("none", "samba.tests.registry", py3_compatible=True) planpythontestsuite("none", "samba.tests.registry", py3_compatible=True)

View File

@ -0,0 +1,53 @@
#!/usr/bin/env python
# Copyright (C) Catalyst IT Ltd. 2017
#
# This program is free software; you can redistribute it and/or modify
# it under the terms of the GNU General Public License as published by
# the Free Software Foundation; either version 3 of the License, or
# (at your option) any later version.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
# GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License
# along with this program. If not, see <http://www.gnu.org/licenses/>.
"""
A data generator to help tests.
Generate large output to stdout by repeating input data.
Usage:
python gen_output.py --data @ --repeat 1024 --retcode 1
The above command will output @ x 1024 (1K) and exit with 1.
"""
import sys
import argparse
parser = argparse.ArgumentParser(description='Generate output data')
parser.add_argument(
'--data', type=str, default='$',
help='Characters used to generate data by repeating them'
)
parser.add_argument(
'--repeat', type=int, default=1024 * 1024,
help='How many times to repeat the data'
)
parser.add_argument(
'--retcode', type=int, default=0,
help='Specify the exit code for this script'
)
args = parser.parse_args()
sys.stdout.write(args.data * args.repeat)
sys.exit(args.retcode)

View File

@ -1,5 +1,5 @@
#!/usr/bin/env python #!/usr/bin/env python
if bld.CONFIG_SET('AD_DC_BUILD_IS_ENABLED'): if bld.CONFIG_SET('AD_DC_BUILD_IS_ENABLED'):
for script in ['samba-tool', 'samba_dnsupdate', 'samba_spnupdate', 'samba_kcc', 'samba_upgradeprovision', 'samba_upgradedns']: for script in ['samba-tool', 'samba_dnsupdate', 'samba_spnupdate', 'samba_kcc', 'samba_upgradeprovision', 'samba_upgradedns', 'gen_output.py']:
bld.SAMBA_SCRIPT(script, pattern=script, installdir='.') bld.SAMBA_SCRIPT(script, pattern=script, installdir='.')