From 8ed3cac9e558c60ab0b7e32133cea5e156ef9549 Mon Sep 17 00:00:00 2001 From: Joe Guo Date: Fri, 15 Sep 2017 15:31:34 +1200 Subject: [PATCH] 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 Reviewed-by: Douglas Bagnall Reviewed-by: Andrew Bartlett --- python/samba/tests/blackbox/check_output.py | 105 ++++++++++++++++++++ selftest/knownfail.d/python-tests | 1 + selftest/tests.py | 1 + source4/scripting/bin/gen_output.py | 53 ++++++++++ source4/scripting/bin/wscript_build | 2 +- 5 files changed, 161 insertions(+), 1 deletion(-) create mode 100644 python/samba/tests/blackbox/check_output.py create mode 100644 selftest/knownfail.d/python-tests create mode 100755 source4/scripting/bin/gen_output.py diff --git a/python/samba/tests/blackbox/check_output.py b/python/samba/tests/blackbox/check_output.py new file mode 100644 index 00000000000..d7e41a838b4 --- /dev/null +++ b/python/samba/tests/blackbox/check_output.py @@ -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 . + +""" +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!') diff --git a/selftest/knownfail.d/python-tests b/selftest/knownfail.d/python-tests new file mode 100644 index 00000000000..754fed5e41f --- /dev/null +++ b/selftest/knownfail.d/python-tests @@ -0,0 +1 @@ +^samba.tests.blackbox.check_output diff --git a/selftest/tests.py b/selftest/tests.py index 3e3ef84a5ec..639bc63649d 100644 --- a/selftest/tests.py +++ b/selftest/tests.py @@ -53,6 +53,7 @@ except ImportError: else: planpythontestsuite("none", "subunit.tests.test_suite") 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", "samba.tests.credentials", py3_compatible=True) planpythontestsuite("none", "samba.tests.registry", py3_compatible=True) diff --git a/source4/scripting/bin/gen_output.py b/source4/scripting/bin/gen_output.py new file mode 100755 index 00000000000..8dc1d6aff24 --- /dev/null +++ b/source4/scripting/bin/gen_output.py @@ -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 . + +""" +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) diff --git a/source4/scripting/bin/wscript_build b/source4/scripting/bin/wscript_build index 1f1ead91c9c..43c21231312 100644 --- a/source4/scripting/bin/wscript_build +++ b/source4/scripting/bin/wscript_build @@ -1,5 +1,5 @@ #!/usr/bin/env python 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='.')