From 9e8cb806b8982455c9b49be86ccc445502e8dc31 Mon Sep 17 00:00:00 2001 From: Zac Hatfield-Dodds Date: Sat, 17 Feb 2024 20:00:50 -0800 Subject: [PATCH 1/5] Update to new settings syntax --- .../2017-04-05-how-not-to-die-hard-with-hypothesis.md | 7 +++---- 1 file changed, 3 insertions(+), 4 deletions(-) diff --git a/website/content/2017-04-05-how-not-to-die-hard-with-hypothesis.md b/website/content/2017-04-05-how-not-to-die-hard-with-hypothesis.md index 57e919ef04..d410879c1a 100644 --- a/website/content/2017-04-05-how-not-to-die-hard-with-hypothesis.md +++ b/website/content/2017-04-05-how-not-to-die-hard-with-hypothesis.md @@ -69,6 +69,8 @@ from hypothesis import note, settings from hypothesis.stateful import RuleBasedStateMachine, invariant, rule +# The default is not always enough for Hypothesis to find a failing example. +@settings(max_examples=2000) class DieHardProblem(RuleBasedStateMachine): small = 0 big = 0 @@ -112,10 +114,7 @@ class DieHardProblem(RuleBasedStateMachine): assert self.big != 4 -# The default of 200 is sometimes not enough for Hypothesis to find -# a falsifying example. -with settings(max_examples=2000): - DieHardTest = DieHardProblem.TestCase +DieHardTest = DieHardProblem.TestCase ``` Calling `pytest` on this file quickly digs up a solution: From 0b362800c703c58e21e0a721ffbc5a02273643f0 Mon Sep 17 00:00:00 2001 From: Zac Hatfield-Dodds Date: Sat, 17 Feb 2024 20:00:50 -0800 Subject: [PATCH 2/5] Make shrinker timeout patchable --- .../src/hypothesis/internal/conjecture/engine.py | 14 ++++++++------ 1 file changed, 8 insertions(+), 6 deletions(-) diff --git a/hypothesis-python/src/hypothesis/internal/conjecture/engine.py b/hypothesis-python/src/hypothesis/internal/conjecture/engine.py index 04be0974a2..2a011a8b11 100644 --- a/hypothesis-python/src/hypothesis/internal/conjecture/engine.py +++ b/hypothesis-python/src/hypothesis/internal/conjecture/engine.py @@ -47,6 +47,13 @@ MIN_TEST_CALLS = 10 BUFFER_SIZE = 8 * 1024 +# If the shrinking phase takes more than five minutes, abort it early and print +# a warning. Many CI systems will kill a build after around ten minutes with +# no output, and appearing to hang isn't great for interactive use either - +# showing partially-shrunk examples is better than quitting with no examples! +# (but make it monkeypatchable, for the rare users who need to keep on shrinking) +MAX_SHRINKING_SECONDS = 300 + @attr.s class HealthCheckState: @@ -934,12 +941,7 @@ def shrink_interesting_examples(self): return self.debug("Shrinking interesting examples") - - # If the shrinking phase takes more than five minutes, abort it early and print - # a warning. Many CI systems will kill a build after around ten minutes with - # no output, and appearing to hang isn't great for interactive use either - - # showing partially-shrunk examples is better than quitting with no examples! - self.finish_shrinking_deadline = time.perf_counter() + 300 + self.finish_shrinking_deadline = time.perf_counter() + MAX_SHRINKING_SECONDS for prev_data in sorted( self.interesting_examples.values(), key=lambda d: sort_key(d.buffer) From 4c187990447aefb7008d7a59a0bd3cd28ad2770d Mon Sep 17 00:00:00 2001 From: Zac Hatfield-Dodds Date: Sat, 17 Feb 2024 20:00:50 -0800 Subject: [PATCH 3/5] Add a note to researchers prompted by the SBFT'24 competition, which included the Ghostwriter as a baseline. --- hypothesis-python/docs/ghostwriter.rst | 59 ++++++++++++++++++++++++++ 1 file changed, 59 insertions(+) diff --git a/hypothesis-python/docs/ghostwriter.rst b/hypothesis-python/docs/ghostwriter.rst index 1b0596ab11..1ed4a12cf1 100644 --- a/hypothesis-python/docs/ghostwriter.rst +++ b/hypothesis-python/docs/ghostwriter.rst @@ -4,3 +4,62 @@ Ghostwriting tests for you .. automodule:: hypothesis.extra.ghostwriter :members: + +A note for test-generation researchers +-------------------------------------- + +Ghostwritten tests are intended as a *starting point for human authorship*, +to demonstrate best practice, help novices past blank-page paralysis, and save time +for experts. They *may* be ready-to-run, or include placeholders and ``# TODO:`` +comments to fill in strategies for unknown types. In either case, improving tests +for their own code gives users a well-scoped and immediately rewarding context in +which to explore property-based testing. + +By contrast, most test-generation tools aim to produce ready-to-run test suites... +and implicitly assume that the current behavior is the desired behavior. +However, the code might contain bugs, and we want our tests to fail if it does! +Worse, tools require that the code to be tested is finished and executable, +making it impossible to generate tests as part of the development process. + +`Fraser 2013`_ found that evolving a high-coverage test suite (e.g. Randoop_, EvoSuite_, Pynguin_) +"leads to clear improvements in commonly applied quality metrics such as code coverage +[but] no measurable improvement in the number of bugs actually found by developers" +and that "generating a set of test cases, even high coverage test cases, +does not necessarily improve our ability to test software". +Invariant detection (famously Daikon_; in PBT see e.g. `Alonso 2022`_, +QuickSpec_, Speculate_) relies on code execution. Program slicing (e.g. FUDGE_, +FuzzGen_, WINNIE_) requires downstream consumers of the code to test. + +Ghostwriter inspects the function name, argument names and types, and docstrings. +It can be used on buggy or incomplete code, runs in a few seconds, and produces +a single semantically-meaningful test per function or group of functions. +Rather than detecting regressions, these tests check semantic properties such as +`encode/decode or save/load round-trips `__, +for `commutative, associative, and distributive operations +`__, +`equivalence between methods `__, +`array shapes `__, +and idempotence. Where no property is detected, we simply check for +'no error on valid input' and allow the user to supply their own invariants. + +Evaluations such as the SBFT24_ competition_ measure performance on a task which +the Ghostwriter is not intended to perform. I'd love to see qualitative user +studies, such as `PBT in Practice`_ for test generation, which could check +whether the Ghostwriter is onto something or tilting at windmills. +If you're interested in similar questions, `drop me an email`_! + +.. _Daikon: https://plse.cs.washington.edu/daikon/pubs/ +.. _Alonso 2022: https://doi.org/10.1145/3540250.3559080 +.. _QuickSpec: http://www.cse.chalmers.se/~nicsma/papers/quickspec2.pdf +.. _Speculate: https://matela.com.br/paper/speculate.pdf +.. _FUDGE: https://research.google/pubs/pub48314/ +.. _FuzzGen: https://www.usenix.org/conference/usenixsecurity20/presentation/ispoglou +.. _WINNIE: https://www.ndss-symposium.org/wp-content/uploads/2021-334-paper.pdf +.. _Fraser 2013: https://doi.org/10.1145/2483760.2483774 +.. _Randoop: https://homes.cs.washington.edu/~mernst/pubs/feedback-testgen-icse2007.pdf +.. _EvoSuite: https://www.evosuite.org/wp-content/papercite-data/pdf/esecfse11.pdf +.. _Pynguin: https://arxiv.org/abs/2007.14049 +.. _SBFT24: https://arxiv.org/abs/2401.15189 +.. _competition: https://github.com/ThunderKey/python-tool-competition-2024 +.. _PBT in Practice: https://harrisongoldste.in/papers/icse24-pbt-in-practice.pdf +.. _drop me an email: mailto:zac@zhd.dev?subject=Hypothesis%20Ghostwriter%20research From 4acbb919aaf9c2586e924d28c0e5c525ece2c80e Mon Sep 17 00:00:00 2001 From: Zac Hatfield-Dodds Date: Sat, 17 Feb 2024 20:00:50 -0800 Subject: [PATCH 4/5] Use canonical module names --- .../src/hypothesis/extra/ghostwriter.py | 7 ++++--- .../recorded/addition_op_multimagic.txt | 6 +++--- .../ghostwriter/recorded/division_operator.txt | 4 ++-- .../division_operator_with_annotations.txt | 4 ++-- ...ivision_roundtrip_arithmeticerror_handler.txt | 4 ++-- .../division_roundtrip_error_handler.txt | 4 ++-- ...undtrip_error_handler_without_annotations.txt | 4 ++-- .../division_roundtrip_typeerror_handler.txt | 4 ++-- .../recorded/multiplication_operator.txt | 16 ++++++++-------- .../multiplication_operator_unittest.txt | 16 ++++++++-------- 10 files changed, 35 insertions(+), 34 deletions(-) diff --git a/hypothesis-python/src/hypothesis/extra/ghostwriter.py b/hypothesis-python/src/hypothesis/extra/ghostwriter.py index 57d1b5330b..cfc52fbb9c 100644 --- a/hypothesis-python/src/hypothesis/extra/ghostwriter.py +++ b/hypothesis-python/src/hypothesis/extra/ghostwriter.py @@ -732,9 +732,10 @@ def _get_module_helper(obj): dots = [i for i, c in enumerate(module_name) if c == "."] + [None] for idx in dots: - if getattr(sys.modules.get(module_name[:idx]), obj.__name__, None) is obj: - KNOWN_FUNCTION_LOCATIONS[obj] = module_name[:idx] - return module_name[:idx] + for candidate in (module_name[:idx].lstrip("_"), module_name[:idx]): + if getattr(sys.modules.get(candidate), obj.__name__, None) is obj: + KNOWN_FUNCTION_LOCATIONS[obj] = candidate + return candidate return module_name diff --git a/hypothesis-python/tests/ghostwriter/recorded/addition_op_multimagic.txt b/hypothesis-python/tests/ghostwriter/recorded/addition_op_multimagic.txt index 194ff573e0..830aeb0e86 100644 --- a/hypothesis-python/tests/ghostwriter/recorded/addition_op_multimagic.txt +++ b/hypothesis-python/tests/ghostwriter/recorded/addition_op_multimagic.txt @@ -1,16 +1,16 @@ # This test code was written by the `hypothesis.extra.ghostwriter` module # and is provided under the Creative Commons Zero public domain dedication. -import _operator import numpy +import operator import test_expected_output from hypothesis import given, strategies as st @given(a=st.floats(), b=st.floats()) def test_equivalent_add_add_add(a: float, b: float) -> None: - result_0_add = _operator.add(a, b) - result_1_add = numpy.add(a, b) + result_0_add = numpy.add(a, b) + result_1_add = operator.add(a, b) result_2_add = test_expected_output.add(a=a, b=b) assert result_0_add == result_1_add, (result_0_add, result_1_add) assert result_0_add == result_2_add, (result_0_add, result_2_add) diff --git a/hypothesis-python/tests/ghostwriter/recorded/division_operator.txt b/hypothesis-python/tests/ghostwriter/recorded/division_operator.txt index 71bbf7a040..2a09e739e8 100644 --- a/hypothesis-python/tests/ghostwriter/recorded/division_operator.txt +++ b/hypothesis-python/tests/ghostwriter/recorded/division_operator.txt @@ -1,7 +1,7 @@ # This test code was written by the `hypothesis.extra.ghostwriter` module # and is provided under the Creative Commons Zero public domain dedication. -import _operator +import operator from hypothesis import given, strategies as st # TODO: replace st.nothing() with an appropriate strategy @@ -11,4 +11,4 @@ truediv_operands = st.nothing() @given(a=truediv_operands) def test_identity_binary_operation_truediv(a): - assert a == _operator.truediv(a, "identity element here") + assert a == operator.truediv(a, "identity element here") diff --git a/hypothesis-python/tests/ghostwriter/recorded/division_operator_with_annotations.txt b/hypothesis-python/tests/ghostwriter/recorded/division_operator_with_annotations.txt index 21eb93d172..3bb86ef548 100644 --- a/hypothesis-python/tests/ghostwriter/recorded/division_operator_with_annotations.txt +++ b/hypothesis-python/tests/ghostwriter/recorded/division_operator_with_annotations.txt @@ -1,7 +1,7 @@ # This test code was written by the `hypothesis.extra.ghostwriter` module # and is provided under the Creative Commons Zero public domain dedication. -import _operator +import operator from hypothesis import given, strategies as st # TODO: replace st.nothing() with an appropriate strategy @@ -11,4 +11,4 @@ truediv_operands = st.nothing() @given(a=truediv_operands) def test_identity_binary_operation_truediv(a) -> None: - assert a == _operator.truediv(a, "identity element here") + assert a == operator.truediv(a, "identity element here") diff --git a/hypothesis-python/tests/ghostwriter/recorded/division_roundtrip_arithmeticerror_handler.txt b/hypothesis-python/tests/ghostwriter/recorded/division_roundtrip_arithmeticerror_handler.txt index 31f9f7d421..8745936fa0 100644 --- a/hypothesis-python/tests/ghostwriter/recorded/division_roundtrip_arithmeticerror_handler.txt +++ b/hypothesis-python/tests/ghostwriter/recorded/division_roundtrip_arithmeticerror_handler.txt @@ -1,7 +1,7 @@ # This test code was written by the `hypothesis.extra.ghostwriter` module # and is provided under the Creative Commons Zero public domain dedication. -import _operator +import operator import test_expected_output from hypothesis import given, reject, strategies as st @@ -10,7 +10,7 @@ from hypothesis import given, reject, strategies as st def test_roundtrip_divide_mul(a: int, b: int) -> None: try: value0 = test_expected_output.divide(a=a, b=b) - value1 = _operator.mul(value0, b) + value1 = operator.mul(value0, b) except ArithmeticError: reject() assert a == value1, (a, value1) diff --git a/hypothesis-python/tests/ghostwriter/recorded/division_roundtrip_error_handler.txt b/hypothesis-python/tests/ghostwriter/recorded/division_roundtrip_error_handler.txt index 7f2b360762..4aea99d133 100644 --- a/hypothesis-python/tests/ghostwriter/recorded/division_roundtrip_error_handler.txt +++ b/hypothesis-python/tests/ghostwriter/recorded/division_roundtrip_error_handler.txt @@ -1,7 +1,7 @@ # This test code was written by the `hypothesis.extra.ghostwriter` module # and is provided under the Creative Commons Zero public domain dedication. -import _operator +import operator import test_expected_output from hypothesis import given, reject, strategies as st @@ -12,5 +12,5 @@ def test_roundtrip_divide_mul(a: int, b: int) -> None: value0 = test_expected_output.divide(a=a, b=b) except ZeroDivisionError: reject() - value1 = _operator.mul(value0, b) + value1 = operator.mul(value0, b) assert a == value1, (a, value1) diff --git a/hypothesis-python/tests/ghostwriter/recorded/division_roundtrip_error_handler_without_annotations.txt b/hypothesis-python/tests/ghostwriter/recorded/division_roundtrip_error_handler_without_annotations.txt index 719d9067aa..cb0d5bc614 100644 --- a/hypothesis-python/tests/ghostwriter/recorded/division_roundtrip_error_handler_without_annotations.txt +++ b/hypothesis-python/tests/ghostwriter/recorded/division_roundtrip_error_handler_without_annotations.txt @@ -1,7 +1,7 @@ # This test code was written by the `hypothesis.extra.ghostwriter` module # and is provided under the Creative Commons Zero public domain dedication. -import _operator +import operator import test_expected_output from hypothesis import given, reject, strategies as st @@ -12,5 +12,5 @@ def test_roundtrip_divide_mul(a, b): value0 = test_expected_output.divide(a=a, b=b) except ZeroDivisionError: reject() - value1 = _operator.mul(value0, b) + value1 = operator.mul(value0, b) assert a == value1, (a, value1) diff --git a/hypothesis-python/tests/ghostwriter/recorded/division_roundtrip_typeerror_handler.txt b/hypothesis-python/tests/ghostwriter/recorded/division_roundtrip_typeerror_handler.txt index 4fc16c7964..52e923d66c 100644 --- a/hypothesis-python/tests/ghostwriter/recorded/division_roundtrip_typeerror_handler.txt +++ b/hypothesis-python/tests/ghostwriter/recorded/division_roundtrip_typeerror_handler.txt @@ -1,7 +1,7 @@ # This test code was written by the `hypothesis.extra.ghostwriter` module # and is provided under the Creative Commons Zero public domain dedication. -import _operator +import operator import test_expected_output from hypothesis import given, reject, strategies as st @@ -13,7 +13,7 @@ def test_roundtrip_divide_mul(a: int, b: int) -> None: value0 = test_expected_output.divide(a=a, b=b) except ZeroDivisionError: reject() - value1 = _operator.mul(value0, b) + value1 = operator.mul(value0, b) except TypeError: reject() assert a == value1, (a, value1) diff --git a/hypothesis-python/tests/ghostwriter/recorded/multiplication_operator.txt b/hypothesis-python/tests/ghostwriter/recorded/multiplication_operator.txt index 4281f161e3..15d4924c0f 100644 --- a/hypothesis-python/tests/ghostwriter/recorded/multiplication_operator.txt +++ b/hypothesis-python/tests/ghostwriter/recorded/multiplication_operator.txt @@ -1,7 +1,7 @@ # This test code was written by the `hypothesis.extra.ghostwriter` module # and is provided under the Creative Commons Zero public domain dedication. -import _operator +import operator from hypothesis import given, strategies as st # TODO: replace st.nothing() with an appropriate strategy @@ -11,25 +11,25 @@ mul_operands = st.nothing() @given(a=mul_operands, b=mul_operands, c=mul_operands) def test_associative_binary_operation_mul(a, b, c): - left = _operator.mul(a, _operator.mul(b, c)) - right = _operator.mul(_operator.mul(a, b), c) + left = operator.mul(a, operator.mul(b, c)) + right = operator.mul(operator.mul(a, b), c) assert left == right, (left, right) @given(a=mul_operands, b=mul_operands) def test_commutative_binary_operation_mul(a, b): - left = _operator.mul(a, b) - right = _operator.mul(b, a) + left = operator.mul(a, b) + right = operator.mul(b, a) assert left == right, (left, right) @given(a=mul_operands) def test_identity_binary_operation_mul(a): - assert a == _operator.mul(a, 1) + assert a == operator.mul(a, 1) @given(a=mul_operands, b=mul_operands, c=mul_operands) def test_add_distributes_over_binary_operation_mul(a, b, c): - left = _operator.add(_operator.mul(a, b), _operator.mul(a, c)) - right = _operator.mul(a, _operator.add(b, c)) + left = operator.add(operator.mul(a, b), operator.mul(a, c)) + right = operator.mul(a, operator.add(b, c)) assert left == right, (left, right) diff --git a/hypothesis-python/tests/ghostwriter/recorded/multiplication_operator_unittest.txt b/hypothesis-python/tests/ghostwriter/recorded/multiplication_operator_unittest.txt index a9517c73d8..42d93b662f 100644 --- a/hypothesis-python/tests/ghostwriter/recorded/multiplication_operator_unittest.txt +++ b/hypothesis-python/tests/ghostwriter/recorded/multiplication_operator_unittest.txt @@ -1,7 +1,7 @@ # This test code was written by the `hypothesis.extra.ghostwriter` module # and is provided under the Creative Commons Zero public domain dedication. -import _operator +import operator import unittest from hypothesis import given, strategies as st @@ -13,22 +13,22 @@ class TestBinaryOperationmul(unittest.TestCase): @given(a=mul_operands, b=mul_operands, c=mul_operands) def test_associative_binary_operation_mul(self, a, b, c): - left = _operator.mul(a, _operator.mul(b, c)) - right = _operator.mul(_operator.mul(a, b), c) + left = operator.mul(a, operator.mul(b, c)) + right = operator.mul(operator.mul(a, b), c) self.assertEqual(left, right) @given(a=mul_operands, b=mul_operands) def test_commutative_binary_operation_mul(self, a, b): - left = _operator.mul(a, b) - right = _operator.mul(b, a) + left = operator.mul(a, b) + right = operator.mul(b, a) self.assertEqual(left, right) @given(a=mul_operands) def test_identity_binary_operation_mul(self, a): - self.assertEqual(a, _operator.mul(a, 1)) + self.assertEqual(a, operator.mul(a, 1)) @given(a=mul_operands, b=mul_operands, c=mul_operands) def test_add_distributes_over_binary_operation_mul(self, a, b, c): - left = _operator.add(_operator.mul(a, b), _operator.mul(a, c)) - right = _operator.mul(a, _operator.add(b, c)) + left = operator.add(operator.mul(a, b), operator.mul(a, c)) + right = operator.mul(a, operator.add(b, c)) self.assertEqual(left, right) From 35546c4b5fbd47fc78ffe75af8ba3250a64bf506 Mon Sep 17 00:00:00 2001 From: Zac Hatfield-Dodds Date: Sat, 17 Feb 2024 20:00:50 -0800 Subject: [PATCH 5/5] Improved binop ghostwriting --- hypothesis-python/RELEASE.rst | 3 + .../src/hypothesis/extra/ghostwriter.py | 103 +++++++++++------- .../recorded/addition_op_magic.txt | 4 +- .../recorded/addition_op_multimagic.txt | 16 ++- .../recorded/division_binop_error_handler.txt | 4 +- .../recorded/division_operator.txt | 4 +- .../division_operator_with_annotations.txt | 4 +- .../ghostwriter/recorded/matmul_magic.txt | 34 ++++++ .../recorded/multiplication_magic.txt | 41 +++++++ .../recorded/multiplication_operator.txt | 14 ++- .../multiplication_operator_unittest.txt | 14 ++- .../tests/ghostwriter/test_expected_output.py | 2 + 12 files changed, 186 insertions(+), 57 deletions(-) create mode 100644 hypothesis-python/RELEASE.rst create mode 100644 hypothesis-python/tests/ghostwriter/recorded/matmul_magic.txt create mode 100644 hypothesis-python/tests/ghostwriter/recorded/multiplication_magic.txt diff --git a/hypothesis-python/RELEASE.rst b/hypothesis-python/RELEASE.rst new file mode 100644 index 0000000000..0c898d8d82 --- /dev/null +++ b/hypothesis-python/RELEASE.rst @@ -0,0 +1,3 @@ +RELEASE_TYPE: patch + +This patch improves :doc:`the Ghostwriter ` for binary operators. diff --git a/hypothesis-python/src/hypothesis/extra/ghostwriter.py b/hypothesis-python/src/hypothesis/extra/ghostwriter.py index cfc52fbb9c..8917d5bd87 100644 --- a/hypothesis-python/src/hypothesis/extra/ghostwriter.py +++ b/hypothesis-python/src/hypothesis/extra/ghostwriter.py @@ -764,7 +764,7 @@ def _get_qualname(obj, *, include_module=False): def _write_call( - func: Callable, *pass_variables: str, except_: Except, assign: str = "" + func: Callable, *pass_variables: str, except_: Except = Exception, assign: str = "" ) -> str: """Write a call to `func` with explicit and implicit arguments. @@ -1269,11 +1269,29 @@ def make_(how, *args, **kwargs): hints = get_type_hints(func) hints.pop("return", None) params = _get_params(func) - if len(hints) == len(params) == 2: - a, b = hints.values() + if (len(hints) == len(params) == 2) or ( + _get_module(func) == "operator" + and "item" not in func.__name__ + and tuple(params) in [("a", "b"), ("x", "y")] + ): + a, b = hints.values() or [Any, Any] arg1, arg2 = params if a == b and len(arg1) == len(arg2) <= 3: - make_(_make_binop_body, func, annotate=annotate) + # https://en.wikipedia.org/wiki/Distributive_property#Other_examples + known = { + "mul": "add", + "matmul": "add", + "or_": "and_", + "and_": "or_", + }.get(func.__name__, "") + distributes_over = getattr(sys.modules[_get_module(func)], known, None) + make_( + _make_binop_body, + func, + commutative=func.__name__ != "matmul", + distributes_over=distributes_over, + annotate=annotate, + ) del by_name[name] # Look for Numpy ufuncs or gufuncs, and write array-oriented tests for them. @@ -1478,10 +1496,17 @@ def roundtrip( return _make_test(*_make_roundtrip_body(funcs, except_, style, annotate)) -def _make_equiv_body(funcs, except_, style, annotate): +def _get_varnames(funcs): var_names = [f"result_{f.__name__}" for f in funcs] if len(set(var_names)) < len(var_names): - var_names = [f"result_{i}_{ f.__name__}" for i, f in enumerate(funcs)] + var_names = [f"result_{f.__name__}_{_get_module(f)}" for f in funcs] + if len(set(var_names)) < len(var_names): + var_names = [f"result_{i}_{f.__name__}" for i, f in enumerate(funcs)] + return var_names + + +def _make_equiv_body(funcs, except_, style, annotate): + var_names = _get_varnames(funcs) test_lines = [ _write_call(f, assign=vname, except_=except_) for vname, f in zip(var_names, funcs) @@ -1521,10 +1546,7 @@ def _make_equiv_body(funcs, except_, style, annotate): def _make_equiv_errors_body(funcs, except_, style, annotate): - var_names = [f"result_{f.__name__}" for f in funcs] - if len(set(var_names)) < len(var_names): - var_names = [f"result_{i}_{ f.__name__}" for i, f in enumerate(funcs)] - + var_names = _get_varnames(funcs) first, *rest = funcs first_call = _write_call(first, assign=var_names[0], except_=except_) extra_imports, suppress = _exception_string(except_) @@ -1724,18 +1746,11 @@ def maker( maker( "associative", "abc", + _write_call(func, "a", _write_call(func, "b", "c"), assign="left"), _write_call( func, - "a", - _write_call(func, "b", "c", except_=Exception), - except_=Exception, - assign="left", - ), - _write_call( - func, - _write_call(func, "a", "b", except_=Exception), + _write_call(func, "a", "b"), "c", - except_=Exception, assign="right", ), ) @@ -1743,8 +1758,8 @@ def maker( maker( "commutative", "ab", - _write_call(func, "a", "b", except_=Exception, assign="left"), - _write_call(func, "b", "a", except_=Exception, assign="right"), + _write_call(func, "a", "b", assign="left"), + _write_call(func, "b", "a", assign="right"), ) if identity is not None: # Guess that the identity element is the minimal example from our operands @@ -1766,34 +1781,42 @@ def maker( compile(repr(identity), "", "exec") except SyntaxError: identity = repr(identity) # type: ignore - maker( - "identity", - "a", + identity_parts = [ + f"{identity = }", _assert_eq( style, "a", - _write_call(func, "a", repr(identity), except_=Exception), + _write_call(func, "a", "identity"), ), - ) + _assert_eq( + style, + "a", + _write_call(func, "identity", "a"), + ), + ] + maker("identity", "a", "\n".join(identity_parts)) if distributes_over: - maker( - distributes_over.__name__ + "_distributes_over", - "abc", + do = distributes_over + dist_parts = [ + _write_call(func, "a", _write_call(do, "b", "c"), assign="left"), _write_call( - distributes_over, - _write_call(func, "a", "b", except_=Exception), - _write_call(func, "a", "c", except_=Exception), - except_=Exception, - assign="left", + do, + _write_call(func, "a", "b"), + _write_call(func, "a", "c"), + assign="ldist", ), + _assert_eq(style, "ldist", "left"), + "\n", + _write_call(func, _write_call(do, "a", "b"), "c", assign="right"), _write_call( - func, - "a", - _write_call(distributes_over, "b", "c", except_=Exception), - except_=Exception, - assign="right", + do, + _write_call(func, "a", "c"), + _write_call(func, "b", "c"), + assign="rdist", ), - ) + _assert_eq(style, "rdist", "right"), + ] + maker(do.__name__ + "_distributes_over", "abc", "\n".join(dist_parts)) _, operands_repr = _valid_syntax_repr(operands) operands_repr = _st_strategy_names(operands_repr) diff --git a/hypothesis-python/tests/ghostwriter/recorded/addition_op_magic.txt b/hypothesis-python/tests/ghostwriter/recorded/addition_op_magic.txt index 23827cb909..0f164bc2f3 100644 --- a/hypothesis-python/tests/ghostwriter/recorded/addition_op_magic.txt +++ b/hypothesis-python/tests/ghostwriter/recorded/addition_op_magic.txt @@ -23,4 +23,6 @@ def test_commutative_binary_operation_add(a: float, b: float) -> None: @given(a=add_operands) def test_identity_binary_operation_add(a: float) -> None: - assert a == test_expected_output.add(a=a, b=0.0) + identity = 0.0 + assert a == test_expected_output.add(a=a, b=identity) + assert a == test_expected_output.add(a=identity, b=a) diff --git a/hypothesis-python/tests/ghostwriter/recorded/addition_op_multimagic.txt b/hypothesis-python/tests/ghostwriter/recorded/addition_op_multimagic.txt index 830aeb0e86..fc71ea7a34 100644 --- a/hypothesis-python/tests/ghostwriter/recorded/addition_op_multimagic.txt +++ b/hypothesis-python/tests/ghostwriter/recorded/addition_op_multimagic.txt @@ -9,8 +9,14 @@ from hypothesis import given, strategies as st @given(a=st.floats(), b=st.floats()) def test_equivalent_add_add_add(a: float, b: float) -> None: - result_0_add = numpy.add(a, b) - result_1_add = operator.add(a, b) - result_2_add = test_expected_output.add(a=a, b=b) - assert result_0_add == result_1_add, (result_0_add, result_1_add) - assert result_0_add == result_2_add, (result_0_add, result_2_add) + result_add_numpy = numpy.add(a, b) + result_add_operator = operator.add(a, b) + result_add_test_expected_output = test_expected_output.add(a=a, b=b) + assert result_add_numpy == result_add_operator, ( + result_add_numpy, + result_add_operator, + ) + assert result_add_numpy == result_add_test_expected_output, ( + result_add_numpy, + result_add_test_expected_output, + ) diff --git a/hypothesis-python/tests/ghostwriter/recorded/division_binop_error_handler.txt b/hypothesis-python/tests/ghostwriter/recorded/division_binop_error_handler.txt index 4248fe5754..ff209f01e1 100644 --- a/hypothesis-python/tests/ghostwriter/recorded/division_binop_error_handler.txt +++ b/hypothesis-python/tests/ghostwriter/recorded/division_binop_error_handler.txt @@ -23,4 +23,6 @@ def test_commutative_binary_operation_divide(a: int, b: int) -> None: @given(a=divide_operands) def test_identity_binary_operation_divide(a: int) -> None: - assert a == test_expected_output.divide(a=a, b=1) + identity = 1 + assert a == test_expected_output.divide(a=a, b=identity) + assert a == test_expected_output.divide(a=identity, b=a) diff --git a/hypothesis-python/tests/ghostwriter/recorded/division_operator.txt b/hypothesis-python/tests/ghostwriter/recorded/division_operator.txt index 2a09e739e8..8daeac4f88 100644 --- a/hypothesis-python/tests/ghostwriter/recorded/division_operator.txt +++ b/hypothesis-python/tests/ghostwriter/recorded/division_operator.txt @@ -11,4 +11,6 @@ truediv_operands = st.nothing() @given(a=truediv_operands) def test_identity_binary_operation_truediv(a): - assert a == operator.truediv(a, "identity element here") + identity = "identity element here" + assert a == operator.truediv(a, identity) + assert a == operator.truediv(identity, a) diff --git a/hypothesis-python/tests/ghostwriter/recorded/division_operator_with_annotations.txt b/hypothesis-python/tests/ghostwriter/recorded/division_operator_with_annotations.txt index 3bb86ef548..bd26cbffde 100644 --- a/hypothesis-python/tests/ghostwriter/recorded/division_operator_with_annotations.txt +++ b/hypothesis-python/tests/ghostwriter/recorded/division_operator_with_annotations.txt @@ -11,4 +11,6 @@ truediv_operands = st.nothing() @given(a=truediv_operands) def test_identity_binary_operation_truediv(a) -> None: - assert a == operator.truediv(a, "identity element here") + identity = "identity element here" + assert a == operator.truediv(a, identity) + assert a == operator.truediv(identity, a) diff --git a/hypothesis-python/tests/ghostwriter/recorded/matmul_magic.txt b/hypothesis-python/tests/ghostwriter/recorded/matmul_magic.txt new file mode 100644 index 0000000000..f73976aca5 --- /dev/null +++ b/hypothesis-python/tests/ghostwriter/recorded/matmul_magic.txt @@ -0,0 +1,34 @@ +# This test code was written by the `hypothesis.extra.ghostwriter` module +# and is provided under the Creative Commons Zero public domain dedication. + +import operator +from hypothesis import given, strategies as st + +# TODO: replace st.nothing() with an appropriate strategy + +matmul_operands = st.nothing() + + +@given(a=matmul_operands, b=matmul_operands, c=matmul_operands) +def test_associative_binary_operation_matmul(a, b, c): + left = operator.matmul(a, operator.matmul(b, c)) + right = operator.matmul(operator.matmul(a, b), c) + assert left == right, (left, right) + + +@given(a=matmul_operands) +def test_identity_binary_operation_matmul(a): + identity = "identity element here" + assert a == operator.matmul(a, identity) + assert a == operator.matmul(identity, a) + + +@given(a=matmul_operands, b=matmul_operands, c=matmul_operands) +def test_add_distributes_over_binary_operation_matmul(a, b, c): + left = operator.matmul(a, operator.add(b, c)) + ldist = operator.add(operator.matmul(a, b), operator.matmul(a, c)) + assert ldist == left, (ldist, left) + + right = operator.matmul(operator.add(a, b), c) + rdist = operator.add(operator.matmul(a, c), operator.matmul(b, c)) + assert rdist == right, (rdist, right) diff --git a/hypothesis-python/tests/ghostwriter/recorded/multiplication_magic.txt b/hypothesis-python/tests/ghostwriter/recorded/multiplication_magic.txt new file mode 100644 index 0000000000..674da337d6 --- /dev/null +++ b/hypothesis-python/tests/ghostwriter/recorded/multiplication_magic.txt @@ -0,0 +1,41 @@ +# This test code was written by the `hypothesis.extra.ghostwriter` module +# and is provided under the Creative Commons Zero public domain dedication. + +import operator +from hypothesis import given, strategies as st + +# TODO: replace st.nothing() with an appropriate strategy + +mul_operands = st.nothing() + + +@given(a=mul_operands, b=mul_operands, c=mul_operands) +def test_associative_binary_operation_mul(a, b, c): + left = operator.mul(a, operator.mul(b, c)) + right = operator.mul(operator.mul(a, b), c) + assert left == right, (left, right) + + +@given(a=mul_operands, b=mul_operands) +def test_commutative_binary_operation_mul(a, b): + left = operator.mul(a, b) + right = operator.mul(b, a) + assert left == right, (left, right) + + +@given(a=mul_operands) +def test_identity_binary_operation_mul(a): + identity = "identity element here" + assert a == operator.mul(a, identity) + assert a == operator.mul(identity, a) + + +@given(a=mul_operands, b=mul_operands, c=mul_operands) +def test_add_distributes_over_binary_operation_mul(a, b, c): + left = operator.mul(a, operator.add(b, c)) + ldist = operator.add(operator.mul(a, b), operator.mul(a, c)) + assert ldist == left, (ldist, left) + + right = operator.mul(operator.add(a, b), c) + rdist = operator.add(operator.mul(a, c), operator.mul(b, c)) + assert rdist == right, (rdist, right) diff --git a/hypothesis-python/tests/ghostwriter/recorded/multiplication_operator.txt b/hypothesis-python/tests/ghostwriter/recorded/multiplication_operator.txt index 15d4924c0f..2ed8968b9f 100644 --- a/hypothesis-python/tests/ghostwriter/recorded/multiplication_operator.txt +++ b/hypothesis-python/tests/ghostwriter/recorded/multiplication_operator.txt @@ -25,11 +25,17 @@ def test_commutative_binary_operation_mul(a, b): @given(a=mul_operands) def test_identity_binary_operation_mul(a): - assert a == operator.mul(a, 1) + identity = 1 + assert a == operator.mul(a, identity) + assert a == operator.mul(identity, a) @given(a=mul_operands, b=mul_operands, c=mul_operands) def test_add_distributes_over_binary_operation_mul(a, b, c): - left = operator.add(operator.mul(a, b), operator.mul(a, c)) - right = operator.mul(a, operator.add(b, c)) - assert left == right, (left, right) + left = operator.mul(a, operator.add(b, c)) + ldist = operator.add(operator.mul(a, b), operator.mul(a, c)) + assert ldist == left, (ldist, left) + + right = operator.mul(operator.add(a, b), c) + rdist = operator.add(operator.mul(a, c), operator.mul(b, c)) + assert rdist == right, (rdist, right) diff --git a/hypothesis-python/tests/ghostwriter/recorded/multiplication_operator_unittest.txt b/hypothesis-python/tests/ghostwriter/recorded/multiplication_operator_unittest.txt index 42d93b662f..912ba51fe2 100644 --- a/hypothesis-python/tests/ghostwriter/recorded/multiplication_operator_unittest.txt +++ b/hypothesis-python/tests/ghostwriter/recorded/multiplication_operator_unittest.txt @@ -25,10 +25,16 @@ class TestBinaryOperationmul(unittest.TestCase): @given(a=mul_operands) def test_identity_binary_operation_mul(self, a): - self.assertEqual(a, operator.mul(a, 1)) + identity = 1 + self.assertEqual(a, operator.mul(a, identity)) + self.assertEqual(a, operator.mul(identity, a)) @given(a=mul_operands, b=mul_operands, c=mul_operands) def test_add_distributes_over_binary_operation_mul(self, a, b, c): - left = operator.add(operator.mul(a, b), operator.mul(a, c)) - right = operator.mul(a, operator.add(b, c)) - self.assertEqual(left, right) + left = operator.mul(a, operator.add(b, c)) + ldist = operator.add(operator.mul(a, b), operator.mul(a, c)) + self.assertEqual(ldist, left) + + right = operator.mul(operator.add(a, b), c) + rdist = operator.add(operator.mul(a, c), operator.mul(b, c)) + self.assertEqual(rdist, right) diff --git a/hypothesis-python/tests/ghostwriter/test_expected_output.py b/hypothesis-python/tests/ghostwriter/test_expected_output.py index 87d959da85..e5431f9642 100644 --- a/hypothesis-python/tests/ghostwriter/test_expected_output.py +++ b/hypothesis-python/tests/ghostwriter/test_expected_output.py @@ -186,6 +186,8 @@ def sequence_from_collections(items: CollectionsSequence[int]) -> int: ghostwriter.equivalent(sorted, sorted, sorted, annotate=True), ), ("addition_op_magic", ghostwriter.magic(add)), + ("multiplication_magic", ghostwriter.magic(operator.mul)), + ("matmul_magic", ghostwriter.magic(operator.matmul)), ("addition_op_multimagic", ghostwriter.magic(add, operator.add, numpy.add)), ("division_fuzz_error_handler", ghostwriter.fuzz(divide)), (