diff --git a/flaky-3.7.0.tar.gz b/flaky-3.7.0.tar.gz deleted file mode 100644 index e691111..0000000 --- a/flaky-3.7.0.tar.gz +++ /dev/null @@ -1,3 +0,0 @@ -version https://git-lfs.github.com/spec/v1 -oid sha256:3ad100780721a1911f57a165809b7ea265a7863305acb66708220820caf8aa0d -size 29591 diff --git a/flaky-3.8.1.tar.gz b/flaky-3.8.1.tar.gz new file mode 100644 index 0000000..aef03f4 --- /dev/null +++ b/flaky-3.8.1.tar.gz @@ -0,0 +1,3 @@ +version https://git-lfs.github.com/spec/v1 +oid sha256:47204a81ec905f3d5acfbd61daeabcada8f9d4031616d9bcb0618461729699f5 +size 25248 diff --git a/python-flaky.changes b/python-flaky.changes index 0fe6e3f..b550f8c 100644 --- a/python-flaky.changes +++ b/python-flaky.changes @@ -1,3 +1,25 @@ +------------------------------------------------------------------- +Tue May 7 05:05:31 UTC 2024 - Steve Kowalik + +- Update to 3.8.1: + * Support pytest >= 8.1.1, and Python >= 3.12. +- Drop patches, included upstream: + * remove_genty.patch + * remove_mock.patch + * remove_nose.patch +- Switch to pyproject macros. + +------------------------------------------------------------------- +Fri Oct 20 10:57:54 UTC 2023 - Matej Cepl + +- Add remove_genty.patch to remove dependency on the external + genty package (gh#box/flaky!197). + +------------------------------------------------------------------- +Fri Oct 20 10:26:48 UTC 2023 - Matej Cepl + +- Clean up the SPEC file + ------------------------------------------------------------------- Fri Apr 21 12:25:08 UTC 2023 - Dirk Müller diff --git a/python-flaky.spec b/python-flaky.spec index 10869a9..14bdec0 100644 --- a/python-flaky.spec +++ b/python-flaky.spec @@ -1,7 +1,7 @@ # -# spec file +# spec file for package python-flaky # -# Copyright (c) 2023 SUSE LLC +# Copyright (c) 2024 SUSE LLC # # All modifications and additions to the file contributed by third parties # remain the property of their copyright owners, unless otherwise agreed @@ -16,7 +16,7 @@ # -%{?!python_module:%define python_module() python-%{**} python3-%{**}} +%{?sle15_python_module_pythons} %global flavor @BUILD_FLAVOR@%{nil} %if "%{flavor}" == "test" %define psuffix -test @@ -24,27 +24,21 @@ %else %bcond_with test %endif -%{?sle15_python_module_pythons} Name: python-flaky%{?psuffix} -Version: 3.7.0 +Version: 3.8.1 Release: 0 Summary: Plugin for nose or py.test that automatically reruns flaky tests License: Apache-2.0 URL: https://github.com/box/flaky Source: https://files.pythonhosted.org/packages/source/f/flaky/flaky-%{version}.tar.gz -# PATCH-FEATURE-UPSTREAM remove_nose.patch gh#box/flaky#171 mcepl@suse.com -# remove dependency on nose -Patch0: remove_nose.patch -# PATCH-FEATURE-UPSTREAM remove_mock.patch gh#box/flaky#171 mcepl@suse.com -# this patch makes things totally awesome -Patch1: remove_mock.patch +BuildRequires: %{python_module pip} BuildRequires: %{python_module setuptools} +BuildRequires: %{python_module wheel} BuildRequires: fdupes BuildRequires: python-rpm-macros BuildArch: noarch %if %{with test} BuildRequires: %{python_module flaky >= %{version}} -BuildRequires: %{python_module genty} BuildRequires: %{python_module pytest} %if 0%{?suse_version} <= 1500 BuildRequires: python-mock @@ -66,12 +60,12 @@ For more information about flaky, see `this presentation 1 -- -- self._flaky_plugin.prepareTestCase(self._mock_test_case) -- if is_failure: -- actual_plugin_handles_failure = self._flaky_plugin.handleFailure( -- self._mock_test_case, -- self._mock_error, -- ) -- else: -- actual_plugin_handles_failure = self._flaky_plugin.handleError( -- self._mock_test_case, -- self._mock_error, -- ) -- -- self.assertEqual( -- expected_plugin_handles_failure or None, -- actual_plugin_handles_failure, -- 'Expected plugin{} to handle the test run, but it did{}.'.format( -- ' to' if expected_plugin_handles_failure else '', -- '' if actual_plugin_handles_failure else ' not' -- ), -- ) -- self._assert_flaky_attributes_contains( -- { -- FlakyNames.CURRENT_RUNS: current_runs + 1, -- FlakyNames.CURRENT_ERRORS: tuple(current_errors), -- }, -- ) -- expected_test_case_calls = [mock.call.address(), mock.call.address()] -- expected_result_calls = [] -- if expected_plugin_handles_failure: -- expected_test_case_calls.append(('__hash__',)) -- expected_stream_calls = [mock.call.writelines([ -- self._mock_test_method_name, -- ' failed ({} runs remaining out of {}).'.format( -- max_runs - current_runs - 1, max_runs -- ), -- 'Exception: Error in test_method', -- '\n', -- ])] -- else: -- if did_plugin_retry_test: -- if is_failure: -- expected_result_calls.append( -- mock.call.addFailure( -- self._mock_test_case, -- self._mock_error, -- ), -- ) -- else: -- expected_result_calls.append(mock.call.addError( -- self._mock_test_case, -- self._mock_error, -- )) -- expected_stream_calls = [mock.call.writelines([ -- self._mock_test_method_name, -- ' failed; it passed {} out of the required {} times.'.format( -- current_passes, -- min_passes -- ), -- 'Exception: Error in test_method', -- '\n' -- ])] -- self.assertEqual( -- self._mock_nose_result.mock_calls, -- expected_result_calls, -- ) -- self.assertEqual( -- self._mock_test_case.mock_calls, -- expected_test_case_calls, -- 'Unexpected TestCase calls: {} vs {}'.format( -- self._mock_test_case.mock_calls, -- expected_test_case_calls -- ) -- ) -- self.assertEqual(self._mock_stream.mock_calls, expected_stream_calls) -- -- def _test_flaky_plugin_handles_success( -- self, -- current_passes=0, -- current_runs=0, -- is_test_method=True, -- max_runs=2, -- min_passes=1 -- ): -- self._assert_flaky_plugin_configured() -- self._expect_test_flaky(is_test_method, max_runs, min_passes) -- self._set_flaky_attribute( -- FlakyNames.CURRENT_PASSES, -- current_passes, -- ) -- self._set_flaky_attribute( -- FlakyNames.CURRENT_RUNS, -- current_runs, -- ) -- -- retries_remaining = current_runs + 1 < max_runs -- too_few_passes = current_passes + 1 < min_passes -- expected_plugin_handles_success = too_few_passes and retries_remaining -- -- self._flaky_plugin.prepareTestCase(self._mock_test_case) -- actual_plugin_handles_success = self._flaky_plugin.addSuccess( -- self._mock_test_case, -- ) -- -- self.assertEqual( -- expected_plugin_handles_success or None, -- actual_plugin_handles_success, -- 'Expected plugin{} to handle the test run, but it did{}.'.format( -- ' not' if expected_plugin_handles_success else '', -- '' if actual_plugin_handles_success else ' not' -- ), -- ) -- self._assert_flaky_attributes_contains( -- { -- FlakyNames.CURRENT_RUNS: current_runs + 1, -- FlakyNames.CURRENT_PASSES: current_passes + 1, -- }, -- ) -- expected_test_case_calls = [mock.call.address(), mock.call.address()] -- expected_stream_calls = [mock.call.writelines([ -- self._mock_test_method_name, -- " passed {} out of the required {} times. ".format( -- current_passes + 1, -- min_passes, -- ), -- ])] -- if expected_plugin_handles_success: -- _rerun_text = 'Running test again until it passes {0} times.\n' -- expected_test_case_calls.append(('__hash__',)) -- expected_stream_calls.append( -- mock.call.write(_rerun_text.format(min_passes)), -- ) -- else: -- expected_stream_calls.append(mock.call.write('Success!\n')) -- self.assertEqual( -- self._mock_test_case.mock_calls, -- expected_test_case_calls, -- 'Unexpected TestCase calls = {} vs {}'.format( -- self._mock_test_case.mock_calls, -- expected_test_case_calls, -- ), -- ) -- self.assertEqual(self._mock_stream.mock_calls, expected_stream_calls) -- -- def _test_flaky_plugin_report(self, expected_stream_value): -- self._assert_flaky_plugin_configured() -- mock_stream = Mock() -- self._mock_stream.getvalue.return_value = expected_stream_value -- -- self._flaky_plugin.report(mock_stream) -- -- self.assertEqual( -- mock_stream.mock_calls, -- [ -- mock.call.write('===Flaky Test Report===\n\n'), -- mock.call.write(expected_stream_value), -- mock.call.write('\n===End Flaky Test Report===\n'), -- ], -- ) -- -- @genty_dataset( -- multiprocess_plugin_absent=(None, 'StringIO'), -- processes_argument_absent=(0, 'StringIO'), -- processes_equals_one=(1, 'MultiprocessingStringIO'), -- processes_equals_two=(2, 'MultiprocessingStringIO'), -- ) -- def test_flaky_plugin_get_stream(self, mp_workers, expected_class_name): -- options = Mock() -- conf = Mock() -- self._flaky_plugin.enabled = True -- options.multiprocess_workers = mp_workers -- if mp_workers is None: -- del options.multiprocess_workers -- self._flaky_plugin.configure(options, conf) -- # pylint:disable=protected-access -- self.assertEqual( -- self._flaky_plugin._stream.__class__.__name__, -- expected_class_name, -- ) ---- a/test/test_nose/test_nose_example.py -+++ /dev/null -@@ -1,98 +0,0 @@ --# coding: utf-8 -- --from __future__ import unicode_literals -- --from unittest import TestCase, skip -- --from genty import genty, genty_dataset --from nose.tools import raises -- --from flaky import flaky -- -- --# This is an end-to-end example of the flaky package in action. Consider it --# a live tutorial, showing the various features in action. -- -- --class ExampleTests(TestCase): -- _threshold = -1 -- -- def test_non_flaky_thing(self): -- """Flaky will not interact with this test""" -- -- @raises(AssertionError) -- def test_non_flaky_failing_thing(self): -- """Flaky will also not interact with this test""" -- self.assertEqual(0, 1) -- -- @flaky(3, 2) -- def test_flaky_thing_that_fails_then_succeeds(self): -- """ -- Flaky will run this test 3 times. It will fail once and then succeed twice. -- """ -- self._threshold += 1 -- if self._threshold < 1: -- raise Exception("Threshold is not high enough: {} vs {}.".format( -- self._threshold, 1), -- ) -- -- @flaky(3, 2) -- def test_flaky_thing_that_succeeds_then_fails_then_succeeds(self): -- """ -- Flaky will run this test 3 times. It will succeed once, fail once, and then succeed one more time. -- """ -- self._threshold += 1 -- if self._threshold == 1: -- self.assertEqual(0, 1) -- -- @flaky(2, 2) -- def test_flaky_thing_that_always_passes(self): -- """Flaky will run this test twice. Both will succeed.""" -- -- @skip("This really fails! Remove this decorator to see the test failure.") -- @flaky() -- def test_flaky_thing_that_always_fails(self): -- """Flaky will run this test twice. Both will fail.""" -- self.assertEqual(0, 1) -- -- --@flaky --class ExampleFlakyTests(TestCase): -- _threshold = -1 -- -- def test_flaky_thing_that_fails_then_succeeds(self): -- """ -- Flaky will run this test twice. It will fail once and then succeed. -- """ -- self._threshold += 1 -- if self._threshold < 1: -- raise Exception("Threshold is not high enough: {} vs {}.".format( -- self._threshold, 1), -- ) -- -- --def test_function(): -- """ -- Nose will import this function and wrap it in a :class:`FunctionTestCase`. -- It's included in the example to make sure flaky handles it correctly. -- """ -- -- --@flaky --def test_flaky_function(param=[]): -- # pylint:disable=dangerous-default-value -- param_length = len(param) -- param.append(None) -- assert param_length == 1 -- -- --@genty --class ExampleFlakyTestsWithUnicodeTestNames(ExampleFlakyTests): -- @genty_dataset('ascii name', 'ńőń ȁŝćȉȉ ŝƭȕƒƒ') -- def test_non_flaky_thing(self, message): -- self._threshold += 1 -- if self._threshold < 1: -- raise Exception( -- "Threshold is not high enough: {} vs {} for '{}'.".format( -- self._threshold, 1, message), -- ) ---- a/test/test_nose/test_nose_options_example.py -+++ /dev/null -@@ -1,54 +0,0 @@ --# coding: utf-8 -- --from __future__ import unicode_literals -- --from unittest import TestCase -- --from flaky import flaky -- --# This is a series of tests that do not use the flaky decorator; the flaky --# behavior is intended to be enabled with the --force-flaky option on the --# command line. -- -- --class ExampleTests(TestCase): -- _threshold = -2 -- -- def test_something_flaky(self): -- """ -- Flaky will run this test twice. -- It will fail once and then succeed once. -- This ensures that we mark tests as flaky even if they don't have a -- decorator when we use the command-line options. -- """ -- self._threshold += 1 -- if self._threshold < 0: -- raise Exception("Threshold is not high enough.") -- -- @flaky(3, 1) -- def test_flaky_thing_that_fails_then_succeeds(self): -- """ -- Flaky will run this test 3 times. -- It will fail twice and then succeed once. -- This ensures that the flaky decorator overrides any command-line -- options we specify. -- """ -- self._threshold += 1 -- if self._threshold < 1: -- raise Exception("Threshold is not high enough.") -- -- --@flaky(3, 1) --class ExampleFlakyTests(TestCase): -- _threshold = -1 -- -- def test_flaky_thing_that_fails_then_succeeds(self): -- """ -- Flaky will run this test 3 times. -- It will fail twice and then succeed once. -- This ensures that the flaky decorator on a test suite overrides any -- command-line options we specify. -- """ -- self._threshold += 1 -- if self._threshold < 1: -- raise Exception("Threshold is not high enough.") ---- a/setup.py -+++ b/setup.py -@@ -68,9 +68,6 @@ def main(): - cmdclass={'test': Tox}, - zip_safe=False, - entry_points={ -- 'nose.plugins.0.10': [ -- 'flaky = flaky.flaky_nose_plugin:FlakyPlugin' -- ], - 'pytest11': [ - 'flaky = flaky.flaky_pytest_plugin' - ] ---- a/flaky/flaky_nose_plugin.py -+++ /dev/null -@@ -1,285 +0,0 @@ --# coding: utf-8 -- --from __future__ import unicode_literals -- --import logging --from optparse import OptionGroup --import os -- --from nose.failure import Failure --from nose.plugins import Plugin --from nose.result import TextTestResult -- --from flaky._flaky_plugin import _FlakyPlugin -- -- --class FlakyPlugin(_FlakyPlugin, Plugin): -- """ -- Plugin for nosetests that allows retrying flaky tests. -- """ -- name = 'flaky' -- -- def __init__(self): -- super(FlakyPlugin, self).__init__() -- self._logger = logging.getLogger('nose.plugins.flaky') -- self._flaky_result = None -- self._nose_result = None -- self._flaky_report = True -- self._force_flaky = False -- self._max_runs = None -- self._min_passes = None -- self._test_status = {} -- self._tests_that_reran = set() -- self._tests_that_have_been_reported = set() -- -- def options(self, parser, env=os.environ): -- """ -- Base class override. -- Add options to the nose argument parser. -- """ -- # pylint:disable=dangerous-default-value -- super(FlakyPlugin, self).options(parser, env=env) -- self.add_report_option(parser.add_option) -- group = OptionGroup( -- parser, "Force flaky", "Force all tests to be flaky.") -- self.add_force_flaky_options(group.add_option) -- parser.add_option_group(group) -- -- def _get_stream(self, multiprocess=False): -- """ -- Get the stream used to store the flaky report. -- If this nose run is going to use the multiprocess plugin, then use -- a multiprocess-list backed StringIO proxy; otherwise, use the default -- stream. -- -- :param multiprocess: -- Whether or not this test run is configured for multiprocessing. -- :type multiprocess: -- `bool` -- :return: -- The stream to use for storing the flaky report. -- :rtype: -- :class:`StringIO` or :class:`MultiprocessingStringIO` -- """ -- if multiprocess: -- from flaky.multiprocess_string_io import MultiprocessingStringIO -- return MultiprocessingStringIO() -- return self._stream -- -- def configure(self, options, conf): -- """Base class override.""" -- super(FlakyPlugin, self).configure(options, conf) -- if not self.enabled: -- return -- is_multiprocess = int(getattr(options, 'multiprocess_workers', 0)) > 0 -- self._stream = self._get_stream(is_multiprocess) -- self._flaky_result = TextTestResult(self._stream, [], 0) -- self._flaky_report = options.flaky_report -- self._flaky_success_report = options.flaky_success_report -- self._force_flaky = options.force_flaky -- self._max_runs = options.max_runs -- self._min_passes = options.min_passes -- -- def startTest(self, test): -- """ -- Base class override. Called before a test is run. -- -- Add the test to the test status tracker, so it can potentially -- be rerun during afterTest. -- -- :param test: -- The test that is going to be run. -- :type test: -- :class:`nose.case.Test` -- """ -- # pylint:disable=invalid-name -- self._test_status[test] = None -- -- def afterTest(self, test): -- """ -- Base class override. Called after a test is run. -- -- If the test was marked for rerun, rerun the test. -- -- :param test: -- The test that has been run. -- :type test: -- :class:`nose.case.Test` -- """ -- # pylint:disable=invalid-name -- if self._test_status[test]: -- self._tests_that_reran.add(id(test)) -- test.run(self._flaky_result) -- self._test_status.pop(test, None) -- -- def _mark_test_for_rerun(self, test): -- """ -- Base class override. Rerun a flaky test. -- -- In this case, don't actually rerun the test, but mark it for -- rerun during afterTest. -- -- :param test: -- The test that is going to be rerun. -- :type test: -- :class:`nose.case.Test` -- """ -- self._test_status[test] = True -- -- def handleError(self, test, err): -- """ -- Baseclass override. Called when a test raises an exception. -- -- If the test isn't going to be rerun again, then report the error -- to the nose test result. -- -- :param test: -- The test that has raised an error -- :type test: -- :class:`nose.case.Test` -- :param err: -- Information about the test failure (from sys.exc_info()) -- :type err: -- `tuple` of `class`, :class:`Exception`, `traceback` -- :return: -- True, if the test will be rerun; False, if nose should handle it. -- :rtype: -- `bool` -- """ -- # pylint:disable=invalid-name -- want_error = self._handle_test_error_or_failure(test, err) -- if not want_error and id(test) in self._tests_that_reran: -- self._nose_result.addError(test, err) -- return want_error or None -- -- def handleFailure(self, test, err): -- """ -- Baseclass override. Called when a test fails. -- -- If the test isn't going to be rerun again, then report the failure -- to the nose test result. -- -- :param test: -- The test that has raised an error -- :type test: -- :class:`nose.case.Test` -- :param err: -- Information about the test failure (from sys.exc_info()) -- :type err: -- `tuple` of `class`, :class:`Exception`, `traceback` -- :return: -- True, if the test will be rerun; False, if nose should handle it. -- :rtype: -- `bool` -- """ -- # pylint:disable=invalid-name -- want_failure = self._handle_test_error_or_failure(test, err) -- if not want_failure and id(test) in self._tests_that_reran: -- self._nose_result.addFailure(test, err) -- return want_failure or None -- -- def addSuccess(self, test): -- """ -- Baseclass override. Called when a test succeeds. -- -- Count remaining retries and compare with number of required successes -- that have not yet been achieved; retry if necessary. -- -- Returning True from this method keeps the test runner from reporting -- the test as a success; this way we can retry and only report as a -- success if we have achieved the required number of successes. -- -- :param test: -- The test that has succeeded -- :type test: -- :class:`nose.case.Test` -- :return: -- True, if the test will be rerun; False, if nose should handle it. -- :rtype: -- `bool` -- """ -- # pylint:disable=invalid-name -- will_handle = self._handle_test_success(test) -- test_id = id(test) -- # If this isn't a rerun, the builtin reporter is going to report it as a success -- if will_handle and test_id not in self._tests_that_reran: -- self._tests_that_have_been_reported.add(test_id) -- # If this test hasn't already been reported as successful, then do it now -- if not will_handle and test_id in self._tests_that_reran and test_id not in self._tests_that_have_been_reported: -- self._nose_result.addSuccess(test) -- return will_handle or None -- -- def report(self, stream): -- """ -- Baseclass override. Write details about flaky tests to the test report. -- -- :param stream: -- The test stream to which the report can be written. -- :type stream: -- `file` -- """ -- if self._flaky_report: -- self._add_flaky_report(stream) -- -- def prepareTestResult(self, result): -- """ -- Baseclass override. Called right before the first test is run. -- -- Stores the test result so that errors and failures can be reported -- to the nose test result. -- -- :param result: -- The nose test result that needs to be informed of test failures. -- :type result: -- :class:`nose.result.TextTestResult` -- """ -- # pylint:disable=invalid-name -- self._nose_result = result -- -- def prepareTestCase(self, test): -- """ -- Baseclass override. Called right before a test case is run. -- -- If the test class is marked flaky and the test callable is not, copy -- the flaky attributes from the test class to the test callable. -- -- :param test: -- The test that is being prepared to run -- :type test: -- :class:`nose.case.Test` -- """ -- # pylint:disable=invalid-name -- if not isinstance(test.test, Failure): -- test_class = test.test -- self._copy_flaky_attributes(test, test_class) -- if self._force_flaky and not self._has_flaky_attributes(test): -- self._make_test_flaky( -- test, self._max_runs, self._min_passes) -- -- @staticmethod -- def _get_test_callable_name(test): -- """ -- Base class override. -- """ -- _, _, class_and_callable_name = test.address() -- first_dot_index = class_and_callable_name.find('.') -- test_callable_name = class_and_callable_name[first_dot_index + 1:] -- return test_callable_name -- -- @classmethod -- def _get_test_callable(cls, test): -- """ -- Base class override. -- -- :param test: -- The test that has raised an error or succeeded -- :type test: -- :class:`nose.case.Test` -- """ -- callable_name = cls._get_test_callable_name(test) -- test_callable = getattr( -- test.test, -- callable_name, -- getattr(test.test, 'test', test.test), -- ) -- return test_callable