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:
parent
ea0cd977a3
commit
8ed3cac9e5
105
python/samba/tests/blackbox/check_output.py
Normal file
105
python/samba/tests/blackbox/check_output.py
Normal 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!')
|
1
selftest/knownfail.d/python-tests
Normal file
1
selftest/knownfail.d/python-tests
Normal file
@ -0,0 +1 @@
|
|||||||
|
^samba.tests.blackbox.check_output
|
@ -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)
|
||||||
|
53
source4/scripting/bin/gen_output.py
Executable file
53
source4/scripting/bin/gen_output.py
Executable 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)
|
@ -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='.')
|
||||||
|
Loading…
Reference in New Issue
Block a user