{
  "RepoName": "https://github.com/srittau/python-asserts.git",
  "CommitSHA": "38849833a92ef24b342e0ca59a8fa5752a07bab4",
  "Type": "parentheses mismatch",
  "ErrorMessage": "======================================================================\nFAIL: test_assert_dict_equal__custom_value_message (test_asserts.AssertTest)\n----------------------------------------------------------------------\nTraceback (most recent call last):\n  File \"/home/user/Projects/user/repobench/repoben/python-asserts/srittau_python-asserts/test_asserts.py\", line 473, in test_assert_dict_equal__custom_value_message\n    assert_dict_equal(\n  File \"/home/user/Projects/user/repobench/repoben/python-asserts/srittau_python-asserts/test_asserts.py\", line 82, in __exit__\n    raise AssertionError(\"no AssertionError raised\")\nAssertionError: no AssertionError raised\n\n======================================================================\nFAIL: test_assert_dict_equal__values_do_not_match (test_asserts.AssertTest)\n----------------------------------------------------------------------\nTraceback (most recent call last):\n  File \"/home/user/Projects/user/repobench/repoben/python-asserts/srittau_python-asserts/test_asserts.py\", line 434, in test_assert_dict_equal__values_do_not_match\n    assert_dict_equal({\"foo\": 15}, {\"foo\": 10})\n  File \"/home/user/Projects/user/repobench/repoben/python-asserts/srittau_python-asserts/test_asserts.py\", line 82, in __exit__\n    raise AssertionError(\"no AssertionError raised\")\nAssertionError: no AssertionError raised\n\n======================================================================\nFAIL: test_assert_dict_superset__custom_value_message (test_asserts.AssertTest)\n----------------------------------------------------------------------\nTraceback (most recent call last):\n  File \"/home/user/Projects/user/repobench/repoben/python-asserts/srittau_python-asserts/test_asserts.py\", line 530, in test_assert_dict_superset__custom_value_message\n    assert_dict_superset(\n  File \"/home/user/Projects/user/repobench/repoben/python-asserts/srittau_python-asserts/test_asserts.py\", line 82, in __exit__\n    raise AssertionError(\"no AssertionError raised\")\nAssertionError: no AssertionError raised\n\n======================================================================\nFAIL: test_assert_dict_superset__values_do_not_match (test_asserts.AssertTest)\n----------------------------------------------------------------------\nTraceback (most recent call last):\n  File \"/home/user/Projects/user/repobench/repoben/python-asserts/srittau_python-asserts/test_asserts.py\", line 503, in test_assert_dict_superset__values_do_not_match\n    assert_dict_superset({\"foo\": 15}, {\"foo\": 10, \"bar\": 15})\n  File \"/home/user/Projects/user/repobench/repoben/python-asserts/srittau_python-asserts/test_asserts.py\", line 82, in __exit__\n    raise AssertionError(\"no AssertionError raised\")\nAssertionError: no AssertionError raised\n\n======================================================================\nFAIL: test_assert_equal__equal_objects (test_asserts.AssertTest)\n----------------------------------------------------------------------\nTraceback (most recent call last):\n  File \"/home/user/Projects/user/repobench/repoben/python-asserts/srittau_python-asserts/test_asserts.py\", line 197, in test_assert_equal__equal_objects\n    assert_equal(MyClass(), MyClass())\n  File \"/home/user/Projects/user/repobench/repoben/python-asserts/srittau_python-asserts/asserts/__init__.py\", line 179, in assert_equal\n    fail(msg_fmt.format(msg=msg, first=first, second=second))\n  File \"/home/user/Projects/user/repobench/repoben/python-asserts/srittau_python-asserts/asserts/__init__.py\", line 44, in fail\n    raise AssertionError(msg or \"assertion failure\")\nAssertionError: <test_asserts.AssertTest.test_assert_equal__equal_objects.<locals>.MyClass object at 0x7fc0cd981eb0> != <test_asserts.AssertTest.test_assert_equal__equal_objects.<locals>.MyClass object at 0x7fc0cda94550>\n\n======================================================================\nFAIL: test_assert_equal__not_equal__custom_message (test_asserts.AssertTest)\n----------------------------------------------------------------------\nTraceback (most recent call last):\n  File \"/home/user/Projects/user/repobench/repoben/python-asserts/srittau_python-asserts/test_asserts.py\", line 205, in test_assert_equal__not_equal__custom_message\n    assert_equal(\"string\", 55, \"{msg};{first!r};{second!r}\")\n  File \"/home/user/Projects/user/repobench/repoben/python-asserts/srittau_python-asserts/test_asserts.py\", line 82, in __exit__\n    raise AssertionError(\"no AssertionError raised\")\nAssertionError: no AssertionError raised\n\n======================================================================\nFAIL: test_assert_equal__not_equal__default_message (test_asserts.AssertTest)\n----------------------------------------------------------------------\nTraceback (most recent call last):\n  File \"/home/user/Projects/user/repobench/repoben/python-asserts/srittau_python-asserts/test_asserts.py\", line 201, in test_assert_equal__not_equal__default_message\n    assert_equal(\"string\", 55)\n  File \"/home/user/Projects/user/repobench/repoben/python-asserts/srittau_python-asserts/test_asserts.py\", line 82, in __exit__\n    raise AssertionError(\"no AssertionError raised\")\nAssertionError: no AssertionError raised",
  "Issue": {
    "title": "Equality Assertion Failure for Falsey Values",
    "description": "There appears to be an issue with the equality assertion logic when handling falsey values. Specifically, the code incorrectly determines that falsey values (such as an empty string or zero) equal other values when they are not, which can lead to incorrect test failures. This issue has been identified within the equality assertion functionality and needs to be resolved to ensure accurate assertion behavior for all types of values, including falsey ones. The failure message format should correctly distinguish between falsey values and other values during equality checks. This fix should ensure robust and reliable test assertions across various scenarios.",
    "explanation": "### Summary of the Issue\nThe issue at hand pertains to the incorrect behavior of the equality assertion logic when handling \"falsey\" values in Python. Falsey values are those that are considered to be false when evaluated in a boolean context, such as an empty string (`\"\"`), zero (`0`), `None`, `False`, etc. The problem arose from the system incorrectly equating a falsey value with other values, leading to incorrect assertions and potential test failures. Detailed assertion messages also needed clear differentiation between falsey values and other values during equality checks.\n\n### Content of the Commit\nHere is a summary of the content from the provided commit:\n\n- **Commit Message:** Improves equality assertions for falsey values to ensure they are correctly distinguished from other values during equality checks.\n- **Changed File:** The file modified is `asserts/__init__.py`, which contains the assertion logic.\n- **Modification:** A specific line in the equality assertion function changes the logic to correctly handle falsey values.\n\n### Detailed Explanation\n\n**Cause of the Issue:**\nThe root cause of the problem lies in the logic used to compare values for equality in the equality assertion function. The original comparison logic inadvertently treated falsey values like `0` and `\"\"` as equal to other values, which is incorrect in most assertion contexts. This caused test failures when actually the values being compared were not truly equal.\n\n**Commit Fix:**\nThe commit addressed this problem by refining the logic used for equality comparison. The change ensures that falsey values are not incorrectly deemed equal to other values simply because they both evaluate to `False` in a boolean context. Instead, the refined logic involves a more precise comparison that respects the distinct nature of falsey values and differentiates them from other non-equal values.\n\n**Solution Explained:**\nThe new logic introduced in the commit essentially separates the comparison operation from the boolean evaluation. Initially, the flawed logic checked the equality by inadvertently relying on the equivalence of their boolean representations (`(not first) == second`). This is problematic because different falsey values (like `0` and `\"\"`) both evaluate to `False` but are fundamentally different.\n\nThe corrected logic uses an improved direct comparison for equality that bypasses this pitfall. This means falsey values are only considered equal if they are indeed the same value (e.g., `0` is not equal to `False` or `\"\"`).\n\n### Summary of the Solution\nThe solution involved refining the logic within the equality assertion function to correctly handle falsey values. The modification ensures that falsey values are directly compared in a more precise manner, avoiding incorrect equivalences that led to test failures. The fix implemented in the commit carefully distinguishes between falsey values and other values during equality checks, thus preventing misleading assertion results in testing scripts. This approach leads to robust and reliable test assertions across various scenarios, improving the operational correctness of the testing framework."
  },
  "Explain": "### Summary of the Issue\nThe issue at hand pertains to the incorrect behavior of the equality assertion logic when handling \"falsey\" values in Python. Falsey values are those that are considered to be false when evaluated in a boolean context, such as an empty string (`\"\"`), zero (`0`), `None`, `False`, etc. The problem arose from the system incorrectly equating a falsey value with other values, leading to incorrect assertions and potential test failures. Detailed assertion messages also needed clear differentiation between falsey values and other values during equality checks.\n\n### Content of the Commit\nHere is a summary of the content from the provided commit:\n\n- **Commit Message:** Improves equality assertions for falsey values to ensure they are correctly distinguished from other values during equality checks.\n- **Changed File:** The file modified is `asserts/__init__.py`, which contains the assertion logic.\n- **Modification:** A specific line in the equality assertion function changes the logic to correctly handle falsey values.\n\n### Detailed Explanation\n\n**Cause of the Issue:**\nThe root cause of the problem lies in the logic used to compare values for equality in the equality assertion function. The original comparison logic inadvertently treated falsey values like `0` and `\"\"` as equal to other values, which is incorrect in most assertion contexts. This caused test failures when actually the values being compared were not truly equal.\n\n**Commit Fix:**\nThe commit addressed this problem by refining the logic used for equality comparison. The change ensures that falsey values are not incorrectly deemed equal to other values simply because they both evaluate to `False` in a boolean context. Instead, the refined logic involves a more precise comparison that respects the distinct nature of falsey values and differentiates them from other non-equal values.\n\n**Solution Explained:**\nThe new logic introduced in the commit essentially separates the comparison operation from the boolean evaluation. Initially, the flawed logic checked the equality by inadvertently relying on the equivalence of their boolean representations (`(not first) == second`). This is problematic because different falsey values (like `0` and `\"\"`) both evaluate to `False` but are fundamentally different.\n\nThe corrected logic uses an improved direct comparison for equality that bypasses this pitfall. This means falsey values are only considered equal if they are indeed the same value (e.g., `0` is not equal to `False` or `\"\"`).\n\n### Summary of the Solution\nThe solution involved refining the logic within the equality assertion function to correctly handle falsey values. The modification ensures that falsey values are directly compared in a more precise manner, avoiding incorrect equivalences that led to test failures. The fix implemented in the commit carefully distinguishes between falsey values and other values during equality checks, thus preventing misleading assertion results in testing scripts. This approach leads to robust and reliable test assertions across various scenarios, improving the operational correctness of the testing framework.",
  "Time": "2024-06-10",
  "Difficulty": "Sample",
  "OriginCode": [
    {
      "path": "srittau_python-asserts/pyproject.toml",
      "content": "[tool.poetry]\nname = \"asserts\"\nversion = \"0.13.2.dev0\"\ndescription = \"Stand-alone Assertions\"\nreadme = \"README.md\"\nauthors = [\"Sebastian Rittau <srittau@rittau.biz>\"]\nlicense = \"MIT\"\nhomepage = \"https://github.com/srittau/python-asserts\"\nrepository = \"https://github.com/srittau/python-asserts\"\nclassifiers = [\n    \"Development Status :: 3 - Alpha\",\n    \"Intended Audience :: Developers\",\n    \"Topic :: Software Development :: Quality Assurance\",\n    \"Topic :: Software Development :: Testing\",\n]\npackages = [{ include = \"asserts\" }]\ninclude = [\"*/py.typed\", \"*.pyi\"]\n\n[tool.poetry.urls]\n\"GitHub\" = \"https://github.com/srittau/python-asserts\"\n\"Bug Tracker\" = \"https://github.com/srittau/python-asserts/issues\"\n\"Changes\" = \"https://github.com/srittau/python-asserts/blob/main/CHANGELOG.md\"\n\n[tool.poetry.dependencies]\npython = \">=3.8.1\"\ntyping-extensions = \"^4.10.0\"\n\n[tool.poetry.group.dev.dependencies]\nmypy = \">=1.9,<1.11\"\npoethepoet = \">=0.25,<0.27\"\nruff = \">=0.3.2,<0.5.0\"\n\n[tool.ruff]\nline-length = 79\ntarget-version = \"py38\"\n\n[tool.poe.tasks]\ntest = \"python3 -Wall -m unittest test_asserts\"\ndoctest = \"python3 -m doctest asserts/__init__.py\"\nlint = \"ruff check asserts test_asserts.py\"\ntypecheck = \"mypy asserts test_asserts.py\"\n\n[build-system]\nrequires = [\"poetry-core>=1.0.0\"]\nbuild-backend = \"poetry.core.masonry.api\"\n"
    },
    {
      "path": "srittau_python-asserts/test_asserts.py",
      "content": "# -*- coding: utf-8 -*-\n\nimport re\nimport sys\nfrom collections import OrderedDict\nfrom datetime import datetime, timedelta, timezone\nfrom json import JSONDecodeError\nfrom unittest import TestCase\nfrom warnings import catch_warnings, simplefilter, warn\n\nfrom asserts import (\n    Absent,\n    Exists,\n    Present,\n    assert_almost_equal,\n    assert_between,\n    assert_boolean_false,\n    assert_boolean_true,\n    assert_count_equal,\n    assert_datetime_about_now,\n    assert_datetime_about_now_utc,\n    assert_dict_equal,\n    assert_dict_superset,\n    assert_equal,\n    assert_false,\n    assert_greater,\n    assert_greater_equal,\n    assert_has_attr,\n    assert_in,\n    assert_is,\n    assert_is_instance,\n    assert_is_none,\n    assert_is_not,\n    assert_is_not_none,\n    assert_json_subset,\n    assert_less,\n    assert_less_equal,\n    assert_not_almost_equal,\n    assert_not_equal,\n    assert_not_in,\n    assert_not_is_instance,\n    assert_not_regex,\n    assert_raises,\n    assert_raises_errno,\n    assert_raises_regex,\n    assert_regex,\n    assert_succeeds,\n    assert_true,\n    assert_warns,\n    assert_warns_regex,\n    fail,\n)\n\n\nclass Box:\n    def __init__(self, initial_value):\n        self.value = initial_value\n\n\nclass _DummyObject(object):\n    def __init__(self, value=\"x\"):\n        self.value = value\n\n    def __repr__(self):\n        return \"<Dummy>\"\n\n\ndef _assert_raises_assertion(expected_message):\n    \"\"\"Fail if the context does not raise an AssertionError or the exception\n    message does not match.\n\n    This is used to test assertions, without using those assertions.\n\n    \"\"\"\n\n    class Context(object):\n        def __enter__(self):\n            pass\n\n        def __exit__(self, exc_type, exc_val, exc_tb):\n            if exc_type is None:\n                raise AssertionError(\"no AssertionError raised\")\n            if exc_type != AssertionError:\n                return False\n            if str(exc_val) != expected_message:\n                raise AssertionError(\n                    \"expected exception message {!r}, got {!r}\".format(\n                        expected_message, str(exc_val)\n                    )\n                )\n            return True\n\n    return Context()\n\n\nclass AssertTest(TestCase):\n    _type_string = \"type\" if sys.version_info[0] < 3 else \"class\"\n\n    # fail()\n\n    def test_fail__default_message(self):\n        with _assert_raises_assertion(\"assertion failure\"):\n            fail()\n\n    def test_fail__with_message(self):\n        with _assert_raises_assertion(\"test message\"):\n            fail(\"test message\")\n\n    # assert_true()\n\n    def test_assert_true__truthy_value(self):\n        assert_true(\"Hello World!\")\n\n    def test_assert_true__falsy_value__default_message(self):\n        with _assert_raises_assertion(\"'' is not truthy\"):\n            assert_true(\"\")\n\n    def test_assert_true__falsy_value__custom_message(self):\n        with _assert_raises_assertion(\"0 is not truthy;0\"):\n            assert_true(0, \"{msg};{expr}\")\n\n    # assert_false()\n\n    def test_assert_false__falsy_value(self):\n        assert_false(\"\")\n\n    def test_assert_false__truthy_value__default_message(self):\n        with _assert_raises_assertion(\"25 is not falsy\"):\n            assert_false(25)\n\n    def test_assert_false__truthy_value__custom_message(self):\n        with _assert_raises_assertion(\"'foo' is not falsy;foo\"):\n            assert_false(\"foo\", \"{msg};{expr}\")\n\n    # assert_boolean_true()\n\n    def test_assert_boolean_true__true(self):\n        assert_boolean_true(True)\n\n    def test_assert_boolean_true__false__custom_message(self):\n        with _assert_raises_assertion(\"'Foo' is not True;Foo\"):\n            assert_boolean_true(\"Foo\", \"{msg};{expr}\")\n\n    def test_assert_boolean_true__truthy__default_message(self):\n        with _assert_raises_assertion(\"1 is not True\"):\n            assert_boolean_true(1)\n\n    # assert_boolean_false()\n\n    def test_assert_boolean_false__false(self):\n        assert_boolean_false(False)\n\n    def test_assert_boolean_false__true__default_message(self):\n        with _assert_raises_assertion(\"'foo' is not False\"):\n            assert_boolean_false(\"foo\")\n\n    def test_assert_boolean_false__falsy__custom_message(self):\n        with _assert_raises_assertion(\"0 is not False;0\"):\n            assert_boolean_false(0, \"{msg};{expr}\")\n\n    # assert_is_none()\n\n    def test_assert_is_none__none(self):\n        assert_is_none(None)\n\n    def test_assert_is_none__string__default_message(self):\n        with _assert_raises_assertion(\"'' is not None\"):\n            assert_is_none(\"\")\n\n    def test_assert_is_none__int__custom_message(self):\n        with _assert_raises_assertion(\"55 is not None;55\"):\n            assert_is_none(55, \"{msg};{expr}\")\n\n    # assert_is_not_none()\n\n    def test_assert_is_not_none__string(self):\n        assert_is_not_none(\"\")\n\n    def test_assert_is_not_none__none__default_message(self):\n        with _assert_raises_assertion(\"expression is None\"):\n            assert_is_not_none(None)\n\n    def test_assert_is_not_none__none__custom_message(self):\n        with _assert_raises_assertion(\"expression is None;None\"):\n            assert_is_not_none(None, \"{msg};{expr!r}\")\n\n    # assert_equal()\n\n    def test_assert_equal__equal_strings(self):\n        assert_equal(\"foo\", \"foo\")\n\n    def test_assert_equal__equal_objects(self):\n        class MyClass(object):\n            def __eq__(self, other):\n                return True\n\n        assert_equal(MyClass(), MyClass())\n\n    def test_assert_equal__not_equal__default_message(self):\n        with _assert_raises_assertion(\"'string' != 55\"):\n            assert_equal(\"string\", 55)\n\n    def test_assert_equal__not_equal__custom_message(self):\n        with _assert_raises_assertion(\"'string' != 55;'string';55\"):\n            assert_equal(\"string\", 55, \"{msg};{first!r};{second!r}\")\n\n    def test_assert_equal__dict(self):\n        with _assert_raises_assertion(\"key 'foo' missing from right dict\"):\n            assert_equal({\"foo\": 5}, {})\n\n    # assert_not_equal()\n\n    def test_assert_not_equal__not_equal(self):\n        assert_not_equal(\"abc\", \"def\")\n\n    def test_assert_not_equal__equal__default_message(self):\n        with _assert_raises_assertion(\"'abc' == 'abc'\"):\n            assert_not_equal(\"abc\", \"abc\")\n\n    def test_assert_not_equal__equal__custom_message(self):\n        with _assert_raises_assertion(\"1.0 == 1;1.0;1\"):\n            assert_not_equal(1.0, 1, \"{msg};{first};{second}\")\n\n    # assert_almost_equal()\n\n    def test_assert_almost_equal__same(self):\n        assert_almost_equal(5, 5)\n\n    def test_assert_almost_equal__similar__defaults(self):\n        assert_almost_equal(5, 5.00000001)\n\n    def test_assert_almost_equal__similar__places(self):\n        assert_almost_equal(5, 5.0001, places=3)\n\n    def test_assert_almost_equal__similar__delta(self):\n        assert_almost_equal(5, 5.001, delta=0.1)\n\n    def test_assert_almost_equal__similar__delta_reverse(self):\n        assert_almost_equal(5, 5.001, delta=0.1)\n\n    def test_assert_almost_equal__not_similar__default_message(self):\n        with _assert_raises_assertion(\"5 != 5.0001 within 7 places\"):\n            assert_almost_equal(5, 5.0001)\n\n    def test_assert_almost_equal__not_similar__places__default_message(self):\n        with _assert_raises_assertion(\"5 != 6 within 3 places\"):\n            assert_almost_equal(5, 6, places=3)\n\n    def test_assert_almost_equal__not_similar__delta__default_message(self):\n        with _assert_raises_assertion(\"5 != 6 with delta=0.1\"):\n            assert_almost_equal(5, 6, delta=0.1)\n\n    def test_assert_almost_equal__not_similar__delta_reverse(self):\n        with _assert_raises_assertion(\"6 != 5 with delta=0.3\"):\n            assert_almost_equal(6, 5, delta=0.3)\n\n    def test_assert_almost_equal__not_similar__custom_message(self):\n        with _assert_raises_assertion(\"5 != -5 within 7 places;5;-5;7;None\"):\n            assert_almost_equal(\n                5, -5, msg_fmt=\"{msg};{first};{second};{places};{delta!r}\"\n            )\n\n    def test_assert_almost_equal__not_similar__places__custom_message(self):\n        with _assert_raises_assertion(\"5 != -5 within 3 places;5;-5;3;None\"):\n            assert_almost_equal(\n                5,\n                -5,\n                places=3,\n                msg_fmt=\"{msg};{first};{second};{places};{delta!r}\",\n            )\n\n    def test_assert_almost_equal__not_similar__delta__custom_message(self):\n        with _assert_raises_assertion(\"5 != 6 with delta=0.1;5;6;None;0.1\"):\n            assert_almost_equal(\n                5,\n                6,\n                delta=0.1,\n                msg_fmt=\"{msg};{first};{second};{places!r};{delta}\",\n            )\n\n    def test_assert_almost_equal__wrong_types(self):\n        try:\n            assert_almost_equal(\"5\", \"5\")  # type: ignore[arg-type]\n        except TypeError:\n            pass\n        else:\n            raise AssertionError(\"TypeError not raised\")\n\n    def test_assert_almost_equal__places_and_delta(self):\n        try:\n            assert_almost_equal(5, 5, places=3, delta=0.0003)\n        except TypeError:\n            pass\n        else:\n            raise AssertionError(\"TypeError not raised\")\n\n    def test_assert_almost_equal__delta_eq_0(self):\n        try:\n            assert_almost_equal(5, 5, delta=0)\n        except ValueError:\n            pass\n        else:\n            raise AssertionError(\"ValueError not raised\")\n\n    def test_assert_almost_equal__delta_lt_0(self):\n        try:\n            assert_almost_equal(5, 5, delta=-1)\n        except ValueError:\n            pass\n        else:\n            raise AssertionError(\"ValueError not raised\")\n\n    # assert_not_almost_equal()\n\n    def test_assert_not_almost_equal__same(self):\n        with _assert_raises_assertion(\"5 == 5 within 7 places\"):\n            assert_not_almost_equal(5, 5)\n\n    def test_assert_not_almost_equal__similar__defaults(self):\n        with _assert_raises_assertion(\"5 == 5.00000001 within 7 places\"):\n            assert_not_almost_equal(5, 5.00000001)\n\n    def test_assert_not_almost_equal__similar__places(self):\n        with _assert_raises_assertion(\"5 == 5.0001 within 3 places\"):\n            assert_not_almost_equal(5, 5.0001, places=3)\n\n    def test_assert_not_almost_equal__similar__delta(self):\n        with _assert_raises_assertion(\"5 == 5.1 with delta=0.1\"):\n            assert_not_almost_equal(5, 5.1, delta=0.1)\n\n    def test_assert_not_almost_equal__similar__delta_reverse(self):\n        with _assert_raises_assertion(\"5 != 6 with delta=0.3\"):\n            assert_almost_equal(5, 6, delta=0.3)\n\n    def test_assert_not_almost_equal__not_similar(self):\n        assert_not_almost_equal(5, 5.0001)\n\n    def test_assert_not_almost_equal__not_similar__delta(self):\n        assert_not_almost_equal(5, 5.1, delta=0.05)\n\n    def test_assert_not_almost_equal__not_similar__delta_reverse(self):\n        assert_not_almost_equal(5.1, 5, delta=0.05)\n\n    def test_assert_not_almost_equal__similar__custom_message(self):\n        with _assert_raises_assertion(\n            \"5 == 5.00000001 within 7 places;5;5.00000001;7;None\"\n        ):\n            assert_not_almost_equal(\n                5,\n                5.00000001,\n                msg_fmt=\"{msg};{first};{second};{places};{delta!r}\",\n            )\n\n    def test_assert_not_almost_equal__similar__places__custom_message(self):\n        with _assert_raises_assertion(\n            \"5 == 5.0001 within 3 places;5;5.0001;3;None\"\n        ):\n            assert_not_almost_equal(\n                5,\n                5.0001,\n                places=3,\n                msg_fmt=\"{msg};{first};{second};{places};{delta!r}\",\n            )\n\n    def test_assert_not_almost_equal__similar__delta__custom_message(self):\n        with _assert_raises_assertion(\"5 == 6 with delta=1.1;5;6;None;1.1\"):\n            assert_not_almost_equal(\n                5,\n                6,\n                delta=1.1,\n                msg_fmt=\"{msg};{first};{second};{places!r};{delta}\",\n            )\n\n    def test_assert_not_almost_equal__wrong_types(self):\n        try:\n            assert_not_almost_equal(\"5\", \"5\")  # type: ignore[arg-type]\n        except TypeError:\n            pass\n        else:\n            raise AssertionError(\"TypeError not raised\")\n\n    def test_assert_not_almost_equal__places_and_delta(self):\n        try:\n            assert_not_almost_equal(5, 5, places=3, delta=0.0003)\n        except TypeError:\n            pass\n        else:\n            raise AssertionError(\"TypeError not raised\")\n\n    def test_not_assert_almost_equal__delta_eq_0(self):\n        try:\n            assert_not_almost_equal(5, 5, delta=0)\n        except ValueError:\n            pass\n        else:\n            raise AssertionError(\"ValueError not raised\")\n\n    def test_not_assert_almost_equal__delta_lt_0(self):\n        try:\n            assert_not_almost_equal(5, 5, delta=-1)\n        except ValueError:\n            pass\n        else:\n            raise AssertionError(\"ValueError not raised\")\n\n    # assert_dict_equal()\n\n    def test_assert_dict_equal__empty_dicts(self):\n        assert_dict_equal({}, {})\n\n    def test_assert_dict_equal__dicts_are_equal(self):\n        assert_dict_equal({\"foo\": 5}, {\"foo\": 5})\n\n    def test_assert_dict_equal__one_key_missing_from_right(self):\n        with _assert_raises_assertion(\"key 'foo' missing from right dict\"):\n            assert_dict_equal({\"bar\": 10, \"foo\": 5}, {\"bar\": 10})\n\n    def test_assert_dict_equal__multiple_keys_missing_from_right(self):\n        with _assert_raises_assertion(\n            \"keys 'bar', 'foo' missing from right dict\"\n        ):\n            assert_dict_equal({\"foo\": 5, \"bar\": 10, \"baz\": 15}, {\"baz\": 15})\n\n    def test_assert_dict_equal__one_key_missing_from_left(self):\n        with _assert_raises_assertion(\"extra key 'foo' in right dict\"):\n            assert_dict_equal({\"bar\": 10}, {\"bar\": 10, \"foo\": 5})\n\n    def test_assert_dict_equal__multiple_keys_missing_from_left(self):\n        with _assert_raises_assertion(\"extra keys 'bar', 'foo' in right dict\"):\n            assert_dict_equal({\"baz\": 15}, {\"foo\": 5, \"bar\": 10, \"baz\": 15})\n\n    def test_assert_dict_equal__values_do_not_match(self):\n        with _assert_raises_assertion(\"key 'foo' differs: 15 != 10\"):\n            assert_dict_equal({\"foo\": 15}, {\"foo\": 10})\n\n    def test_assert_dict_equal__not_string_keys(self):\n        with _assert_raises_assertion(\"key 10 missing from right dict\"):\n            assert_dict_equal({10: \"foo\"}, {})\n        with _assert_raises_assertion(\"keys 'foo', 5 missing from right dict\"):\n            assert_dict_equal({5: \"\", \"foo\": \"\"}, {})\n        with _assert_raises_assertion(\"extra key 10 in right dict\"):\n            assert_dict_equal({}, {10: \"foo\"})\n        with _assert_raises_assertion(\"extra keys 'foo', 5 in right dict\"):\n            assert_dict_equal({}, {5: \"\", \"foo\": \"\"})\n\n    def test_assert_dict_equal__message_precedence(self):\n        with _assert_raises_assertion(\"key 'foo' missing from right dict\"):\n            assert_dict_equal(\n                {\"foo\": \"\", \"bar\": \"\", \"baz\": 5},\n                {\"bar\": \"\", \"baz\": 10, \"extra\": \"\"},\n            )\n        with _assert_raises_assertion(\"extra key 'extra' in right dict\"):\n            assert_dict_equal(\n                {\"bar\": \"\", \"baz\": 5}, {\"bar\": \"\", \"baz\": 10, \"extra\": \"\"}\n            )\n\n    def test_assert_dict_equal__custom_key_message(self):\n        with _assert_raises_assertion(\n            \"key 'foo' missing from right dict;\"\n            \"{'foo': ''};{'bar': ''};['foo'];['bar']\"\n        ):\n            assert_dict_equal(\n                {\"foo\": \"\"},\n                {\"bar\": \"\"},\n                key_msg_fmt=\"{msg};{first!r};{second!r};\"\n                \"{missing_keys!r};{extra_keys!r}\",\n            )\n\n    def test_assert_dict_equal__custom_value_message(self):\n        with _assert_raises_assertion(\n            \"key 'foo' differs: 5 != 10;{'foo': 5};{'foo': 10};\" \"'foo';5;10\"\n        ):\n            assert_dict_equal(\n                {\"foo\": 5},\n                {\"foo\": 10},\n                value_msg_fmt=\"{msg};{first!r};{second!r};\"\n                \"{key!r};{first_value};{second_value}\",\n            )\n\n    # assert_dict_superset()\n\n    def test_assert_dict_superset__empty_dicts(self):\n        assert_dict_superset({}, {})\n\n    def test_assert_dict_superset__dicts_are_equal(self):\n        assert_dict_superset({\"foo\": 5}, {\"foo\": 5})\n\n    def test_assert_dict_superset__dicts_is_superset(self):\n        assert_dict_superset({\"foo\": 5}, {\"foo\": 5, \"bar\": 10})\n\n    def test_assert_dict_superset__one_key_missing_from_right(self):\n        with _assert_raises_assertion(\"key 'foo' missing from right dict\"):\n            assert_dict_superset({\"bar\": 10, \"foo\": 5}, {\"bar\": 10})\n\n    def test_assert_dict_superset__multiple_keys_missing_from_right(self):\n        with _assert_raises_assertion(\n            \"keys 'bar', 'foo' missing from right dict\"\n        ):\n            assert_dict_superset({\"foo\": 5, \"bar\": 10, \"baz\": 15}, {\"baz\": 15})\n\n    def test_assert_dict_superset__values_do_not_match(self):\n        with _assert_raises_assertion(\"key 'foo' differs: 15 != 10\"):\n            assert_dict_superset({\"foo\": 15}, {\"foo\": 10, \"bar\": 15})\n\n    def test_assert_dict_superset__not_string_keys(self):\n        with _assert_raises_assertion(\"key 10 missing from right dict\"):\n            assert_dict_superset({10: \"foo\"}, {})\n        with _assert_raises_assertion(\"keys 'foo', 5 missing from right dict\"):\n            assert_dict_superset({5: \"\", \"foo\": \"\"}, {})\n\n    def test_assert_dict_superset__message_precedence(self):\n        with _assert_raises_assertion(\"key 'foo' missing from right dict\"):\n            assert_dict_superset({\"foo\": \"\", \"bar\": 5}, {\"bar\": 1})\n\n    def test_assert_dict_superset__custom_key_message(self):\n        with _assert_raises_assertion(\n            \"key 'foo' missing from right dict;\"\n            \"{'foo': ''};{'bar': ''};['foo']\"\n        ):\n            assert_dict_superset(\n                {\"foo\": \"\"},\n                {\"bar\": \"\"},\n                key_msg_fmt=\"{msg};{first!r};{second!r};\" \"{missing_keys!r}\",\n            )\n\n    def test_assert_dict_superset__custom_value_message(self):\n        with _assert_raises_assertion(\n            \"key 'foo' differs: 5 != 10;{'foo': 5};{'foo': 10};\" \"'foo';5;10\"\n        ):\n            assert_dict_superset(\n                {\"foo\": 5},\n                {\"foo\": 10},\n                value_msg_fmt=\"{msg};{first!r};{second!r};\"\n                \"{key!r};{first_value};{second_value}\",\n            )\n\n    # assert_less()\n\n    def test_assert_less(self):\n        assert_less(4, 5)\n        with _assert_raises_assertion(\"5 is not less than 5\"):\n            assert_less(5, 5)\n        with _assert_raises_assertion(\"'foo' is not less than 'bar'\"):\n            assert_less(\"foo\", \"bar\")\n        with _assert_raises_assertion(\"6 is not less than 5;6;5\"):\n            assert_less(6, 5, \"{msg};{first};{second}\")\n\n    # assert_less_equal()\n\n    def test_assert_less_equal(self):\n        assert_less_equal(4, 5)\n        assert_less_equal(5, 5)\n        with _assert_raises_assertion(\n            \"'foo' is not less than or equal to 'bar'\"\n        ):\n            assert_less_equal(\"foo\", \"bar\")\n        with _assert_raises_assertion(\"6 is not less than or equal to 5;6;5\"):\n            assert_less_equal(6, 5, \"{msg};{first};{second}\")\n\n    # assert_greater()\n\n    def test_assert_greater(self):\n        assert_greater(5, 4)\n        with _assert_raises_assertion(\"5 is not greater than 5\"):\n            assert_greater(5, 5)\n        with _assert_raises_assertion(\"'bar' is not greater than 'foo'\"):\n            assert_greater(\"bar\", \"foo\")\n        with _assert_raises_assertion(\"5 is not greater than 6;5;6\"):\n            assert_greater(5, 6, \"{msg};{first};{second}\")\n\n    # assert_greater_equal()\n\n    def test_assert_greater_equal(self):\n        assert_greater_equal(5, 4)\n        assert_greater_equal(5, 5)\n        with _assert_raises_assertion(\n            \"'bar' is not greater than or equal to 'foo'\"\n        ):\n            assert_greater_equal(\"bar\", \"foo\")\n        with _assert_raises_assertion(\n            \"5 is not greater than or equal to 6;5;6\"\n        ):\n            assert_greater_equal(5, 6, \"{msg};{first};{second}\")\n\n    # assert_regex()\n\n    def test_assert_regex__matches_string(self):\n        assert_regex(\"This is a test text\", \"is.*test\")\n\n    def test_assert_regex__matches_regex(self):\n        regex = re.compile(\"is.*test\")\n        assert_regex(\"This is a test text\", regex)\n\n    def test_assert_regex__does_not_match_string__default_message(self):\n        with _assert_raises_assertion(\n            \"'This is a test text' does not match 'not found'\"\n        ):\n            assert_regex(\"This is a test text\", \"not found\")\n\n    def test_assert_regex__does_not_match_regex__default_message(self):\n        regex = re.compile(r\"not found\")\n        with _assert_raises_assertion(\n            \"'This is a test text' does not match 'not found'\"\n        ):\n            assert_regex(\"This is a test text\", regex)\n\n    def test_assert_regex__does_not_match_string__custom_message(self):\n        with _assert_raises_assertion(\n            \"'Wrong text' does not match 'not found';\"\n            \"'Wrong text';'not found'\"\n        ):\n            assert_regex(\n                \"Wrong text\", r\"not found\", \"{msg};{text!r};{pattern!r}\"\n            )\n\n    def test_assert_regex__does_not_match_regex__custom_message(self):\n        regex = re.compile(r\"not found\")\n        with _assert_raises_assertion(\n            \"'Wrong text' does not match 'not found';'Wrong text';\"\n            \"'not found'\"\n        ):\n            assert_regex(\"Wrong text\", regex, \"{msg};{text!r};{pattern!r}\")\n\n    # assert_not_regex()\n\n    def test_assert_not_regex__does_not_match_string(self):\n        assert_not_regex(\"This is a test text\", \"no match\")\n\n    def test_assert_not_regex__does_not_match_regex(self):\n        regex = re.compile(\"no match\")\n        assert_not_regex(\"This is a test text\", regex)\n\n    def test_assert_not_regex__matches_string__default_message(self):\n        with _assert_raises_assertion(\n            \"'This is a test text' matches 'is.*test'\"\n        ):\n            assert_not_regex(\"This is a test text\", \"is.*test\")\n\n    def test_assert_not_regex__matches_regex__default_message(self):\n        regex = re.compile(\"is.*test\")\n        with _assert_raises_assertion(\n            \"'This is a test text' matches 'is.*test'\"\n        ):\n            assert_not_regex(\"This is a test text\", regex)\n\n    def test_assert_not_regex__matches_string__custom_message(self):\n        with _assert_raises_assertion(\n            \"'This is a test text' matches 'is.*test';\"\n            \"'This is a test text';'is.*test'\"\n        ):\n            assert_not_regex(\n                \"This is a test text\",\n                \"is.*test\",\n                \"{msg};{text!r};{pattern!r}\",\n            )\n\n    def test_assert_not_regex__matches_regex__custom_message(self):\n        regex = re.compile(\"is.*test\")\n        with _assert_raises_assertion(\n            \"'This is a test text' matches 'is.*test';'This is a test text';\"\n            \"'is.*test'\"\n        ):\n            assert_not_regex(\n                \"This is a test text\", regex, \"{msg};{text!r};{pattern!r}\"\n            )\n\n    # assert_is()\n\n    def test_assert_is__same(self):\n        x = _DummyObject()\n        assert_is(x, x)\n\n    def test_assert_is__not_same__default_message(self):\n        with _assert_raises_assertion(\"'x' is not 'y'\"):\n            assert_is(\"x\", \"y\")\n\n    def test_assert_is__equal_but_not_same__custom_message(self):\n        x = \"x\"\n        y = _DummyObject(\"y\")\n        with _assert_raises_assertion(\"'x' is not <Dummy>;'x';y\"):\n            assert_is(x, y, \"{msg};{first!r};{second.value}\")\n\n    # assert_is_not()\n\n    def test_assert_is_not__not_same(self):\n        x = _DummyObject()\n        y = _DummyObject()\n        assert_is_not(x, y)\n\n    def test_assert_is_not__same__default_message(self):\n        x = _DummyObject(\"x\")\n        with _assert_raises_assertion(\"both arguments refer to <Dummy>\"):\n            assert_is_not(x, x)\n\n    def test_assert_is_not__same__custom_message(self):\n        x = _DummyObject(\"x\")\n        with _assert_raises_assertion(\"both arguments refer to <Dummy>;x;x\"):\n            assert_is_not(x, x, \"{msg};{first.value};{second.value}\")\n\n    # assert_in()\n\n    def test_assert_in__contains(self):\n        assert_in(\"foo\", [\"foo\", \"bar\", \"baz\"])\n\n    def test_assert_in__does_not_contain__default_message(self):\n        with _assert_raises_assertion(\"'foo' not in []\"):\n            assert_in(\"foo\", [])\n\n    def test_assert_in__does_not_contain__custom_message(self):\n        with _assert_raises_assertion(\"'foo' not in [];'foo';[]\"):\n            assert_in(\"foo\", [], \"{msg};{first!r};{second!r}\")\n\n    # assert_not_in()\n\n    def test_assert_not_in__does_not_contain(self):\n        assert_not_in(\"foo\", [])\n\n    def test_assert_not_in__does_contain__default_message(self):\n        with _assert_raises_assertion(\"'foo' is in ['foo', 'bar', 'baz']\"):\n            assert_not_in(\"foo\", [\"foo\", \"bar\", \"baz\"])\n\n    def test_assert_not_in__does_contain__custom_message(self):\n        with _assert_raises_assertion(\"'foo' is in ['foo', 'bar'];'foo';bar\"):\n            assert_not_in(\"foo\", [\"foo\", \"bar\"], \"{msg};{first!r};{second[1]}\")\n\n    # assert_count_equal()\n\n    def test_assert_count_equal__equal(self):\n        with assert_succeeds(AssertionError):\n            assert_count_equal([\"a\"], [\"a\"])\n\n    def test_assert_count_equal__equal_differing_types(self):\n        with assert_succeeds(AssertionError):\n            assert_count_equal([\"a\"], {\"a\"})\n\n    def test_assert_count_equal__ignore_order(self):\n        with assert_succeeds(AssertionError):\n            assert_count_equal([\"a\", \"b\"], [\"b\", \"a\"])\n\n    def test_assert_count_equal__missing_from_sequence1(self):\n        with _assert_raises_assertion(\"missing from sequence 1: 'a'\"):\n            assert_count_equal([], {\"a\"})\n\n    def test_assert_count_equal__multiple_missing_from_sequence1(self):\n        with _assert_raises_assertion(\"missing from sequence 1: 'b', 'c'\"):\n            assert_count_equal([\"a\"], [\"a\", \"b\", \"c\"])\n\n    def test_assert_count_equal__respect_duplicates(self):\n        with _assert_raises_assertion(\"missing from sequence 1: 'a'\"):\n            assert_count_equal({\"a\"}, [\"a\", \"a\"])\n\n    def test_assert_count_equal__missing_from_sequence2(self):\n        with _assert_raises_assertion(\"missing from sequence 2: 'a', 'c'\"):\n            assert_count_equal([\"a\", \"b\", \"c\"], [\"b\"])\n\n    def test_assert_count_equal__missing_from_both(self):\n        msg = \"missing from sequence 1: 'd'; missing from sequence 2: 'b', 'c'\"\n        with _assert_raises_assertion(msg):\n            assert_count_equal([\"a\", \"b\", \"c\"], [\"a\", \"d\"])\n\n    def test_assert_count_equal__custom_message(self):\n        with _assert_raises_assertion(\"missing from sequence 1: 'a';[];['a']\"):\n            assert_count_equal([], [\"a\"], \"{msg};{first};{second}\")\n\n    # assert_between()\n\n    def test_assert_between__within_range(self):\n        assert_between(0, 10, 0)\n        assert_between(0, 10, 10)\n        assert_between(0, 10, 5)\n\n    def test_assert_between__too_low__default_message(self):\n        with _assert_raises_assertion(\"-1 is not between 0 and 10\"):\n            assert_between(0, 10, -1)\n\n    def test_assert_between__too_high__custom_message(self):\n        with _assert_raises_assertion(\"11 is not between 0 and 10;0;10;11\"):\n            assert_between(0, 10, 11, \"{msg};{lower};{upper};{expr}\")\n\n    # assert_is_instance()\n\n    def _is_instance_message(self, expr, expected_type, real_type):\n        expected_message = (\n            \"{!r} is an instance of <class {}>, expected {}\".format(\n                expr, real_type, expected_type\n            )\n        )\n        if sys.version_info[0] < 3:\n            return expected_message.replace(\"class\", \"type\")\n        else:\n            return expected_message\n\n    def test_assert_is_instance__single_type(self):\n        assert_is_instance(4, int)\n        assert_is_instance(OSError(), Exception)\n\n    def test_assert_is_instance__multiple_types(self):\n        assert_is_instance(4, (str, int))\n\n    def test_assert_is_instance__default_message(self):\n        expected_message = self._is_instance_message(\n            \"my string\", \"<class 'int'>\", \"'str'\"\n        )\n        with _assert_raises_assertion(expected_message):\n            assert_is_instance(\"my string\", int)\n\n    def test_assert_is_instance__custom_message_single_type(self):\n        expected_message = self._is_instance_message(\n            \"my string\", \"<class 'int'>\", \"'str'\"\n        )\n        expected = \"{};my string;(<class 'int'>,)\".format(expected_message)\n        expected = expected.replace(\"class\", self._type_string)\n        with _assert_raises_assertion(expected):\n            assert_is_instance(\"my string\", int, \"{msg};{obj};{types}\")\n\n    def test_assert_is_instance__custom_message_multiple_types(self):\n        expected_message = self._is_instance_message(\n            \"my string\", \"(<class 'int'>, <class 'float'>)\", \"'str'\"\n        )\n        expected = \"{};my string;(<class 'int'>, <class 'float'>)\".format(\n            expected_message\n        )\n        expected = expected.replace(\"class\", self._type_string)\n        with _assert_raises_assertion(expected):\n            assert_is_instance(\n                \"my string\", (int, float), \"{msg};{obj};{types}\"\n            )\n\n    # assert_not_is_instance()\n\n    def _not_is_instance_message(self, obj):\n        expected_message = \"{!r} is an instance of {}\".format(\n            obj, obj.__class__\n        )\n        if sys.version_info[0] < 3:\n            expected_message = expected_message.replace(\"class\", \"type\")\n            expected_message = expected_message.replace(\n                \"type 'OSError'\", \"type 'exceptions.OSError'\"\n            )\n        return expected_message\n\n    def test_assert_not_is_instance__single_type(self):\n        assert_not_is_instance(4, str)\n\n    def test_assert_not_is_instance__multiple_types(self):\n        assert_not_is_instance(4, (str, bytes))\n\n    def test_assert_not_is_instance__default_message(self):\n        obj = OSError()\n        expected_message = self._not_is_instance_message(obj)\n        with _assert_raises_assertion(expected_message):\n            assert_not_is_instance(obj, Exception)\n\n    def test_assert_not_is_instance__custom_message__single_type(self):\n        msg = self._not_is_instance_message(\"Foo\")\n        expected = \"{};Foo;(<class 'str'>,)\".format(msg)\n        expected = expected.replace(\"class\", self._type_string)\n        with _assert_raises_assertion(expected):\n            assert_not_is_instance(\"Foo\", str, \"{msg};{obj};{types!r}\")\n\n    def test_assert_not_is_instance__custom_message__multiple_types(self):\n        msg = self._not_is_instance_message(\"Foo\")\n        expected = \"{};Foo;(<class 'str'>, <class 'int'>)\".format(msg)\n        expected = expected.replace(\"class\", self._type_string)\n        with _assert_raises_assertion(expected):\n            assert_not_is_instance(\"Foo\", (str, int), \"{msg};{obj};{types!r}\")\n\n    # assert_has_attr()\n\n    def test_assert_has_attr__has_attribute(self):\n        d = _DummyObject()\n        assert_has_attr(d, \"value\")\n\n    def test_assert_has_attr__does_not_have_attribute__default_message(self):\n        d = _DummyObject()\n        with _assert_raises_assertion(\"<Dummy> does not have attribute 'foo'\"):\n            assert_has_attr(d, \"foo\")\n\n    def test_assert_has_attr__does_not_have_attribute__custom_message(self):\n        d = _DummyObject()\n        expected = \"<Dummy> does not have attribute 'foo';<Dummy>;foo\"\n        with _assert_raises_assertion(expected):\n            assert_has_attr(d, \"foo\", msg_fmt=\"{msg};{obj!r};{attribute}\")\n\n    # assert_datetime_about_now()\n\n    def test_assert_datetime_about_now__close(self):\n        assert_datetime_about_now(datetime.now())\n\n    def test_assert_datetime_about_now__none__default_message(self):\n        expected_message = r\"^None is not a valid date/time$\"\n        with assert_raises_regex(AssertionError, expected_message):\n            assert_datetime_about_now(None)\n\n    def test_assert_datetime_about_now__none__custom_message(self):\n        dt = datetime.now().date().isoformat()\n        expected = \"None is not a valid date/time;None;{}\".format(dt)\n        with _assert_raises_assertion(expected):\n            assert_datetime_about_now(\n                None, msg_fmt=\"{msg};{actual!r};{now:%Y-%m-%d}\"\n            )\n\n    def test_assert_datetime_about_now__too_low(self):\n        then = datetime.now() - timedelta(minutes=1)\n        with assert_raises(AssertionError):\n            assert_datetime_about_now(then)\n\n    def test_assert_datetime_about_now__too_high(self):\n        then = datetime.now() + timedelta(minutes=1)\n        with assert_raises(AssertionError):\n            assert_datetime_about_now(then)\n\n    def test_assert_datetime_about_now__default_message(self):\n        then = datetime(1990, 4, 13, 12, 30, 15)\n        expected_message = (\n            r\"^datetime.datetime\\(1990, 4, 13, 12, 30, 15\\) \"\n            \"is not close to current date/time$\"\n        )\n        with assert_raises_regex(AssertionError, expected_message):\n            assert_datetime_about_now(then)\n\n    def test_assert_datetime_about_now__custom_message(self):\n        then = datetime(1990, 4, 13, 12, 30, 15)\n        now = datetime.now().date().isoformat()\n        expected = (\n            \"datetime.datetime(1990, 4, 13, 12, 30, 15) \"\n            \"is not close to current date/time;12:30;{}\".format(now)\n        )\n        with _assert_raises_assertion(expected):\n            assert_datetime_about_now(\n                then, msg_fmt=\"{msg};{actual:%H:%M};{now:%Y-%m-%d}\"\n            )\n\n    # assert_datetime_about_now_utc()\n\n    def test_assert_datetime_about_now_utc__close(self):\n        assert_datetime_about_now_utc(\n            datetime.now(timezone.utc).replace(tzinfo=None)\n        )\n\n    def test_assert_datetime_about_now_utc__none__default_message(self):\n        expected_message = r\"^None is not a valid date/time$\"\n        with assert_raises_regex(AssertionError, expected_message):\n            assert_datetime_about_now_utc(None)\n\n    def test_assert_datetime_about_now_utc__none__custom_message(self):\n        dt = datetime.now(timezone.utc).date().isoformat()\n        expected = \"None is not a valid date/time;None;{}\".format(dt)\n        with _assert_raises_assertion(expected):\n            assert_datetime_about_now_utc(\n                None, msg_fmt=\"{msg};{actual!r};{now:%Y-%m-%d}\"\n            )\n\n    def test_assert_datetime_about_now_utc__too_low(self):\n        then = datetime.now(timezone.utc).replace(tzinfo=None) - timedelta(\n            minutes=1\n        )\n        with assert_raises(AssertionError):\n            assert_datetime_about_now_utc(then)\n\n    def test_assert_datetime_about_now_utc__too_high(self):\n        then = datetime.now(timezone.utc).replace(tzinfo=None) + timedelta(\n            minutes=1\n        )\n        with assert_raises(AssertionError):\n            assert_datetime_about_now_utc(then)\n\n    def test_assert_datetime_about_now_utc__default_message(self):\n        then = datetime(1990, 4, 13, 12, 30, 15)\n        expected_message = (\n            r\"datetime.datetime\\(1990, 4, 13, 12, 30, 15\\) \"\n            r\"is not close to current UTC date/time$\"\n        )\n        with assert_raises_regex(AssertionError, expected_message):\n            assert_datetime_about_now_utc(then)\n\n    def test_assert_datetime_about_now_utc__custom_message(self):\n        then = datetime(1990, 4, 13, 12, 30, 15)\n        now = datetime.now(timezone.utc).date().isoformat()\n        expected = (\n            \"datetime.datetime(1990, 4, 13, 12, 30, 15) \"\n            \"is not close to current UTC date/time;12:30;{}\".format(now)\n        )\n        with _assert_raises_assertion(expected):\n            assert_datetime_about_now_utc(\n                then, msg_fmt=\"{msg};{actual:%H:%M};{now:%Y-%m-%d}\"\n            )\n\n    # assert_raises()\n\n    def test_assert_raises__raises_right_exception(self):\n        with assert_raises(KeyError):\n            raise KeyError()\n\n    def test_assert_raises__exc_val(self):\n        exc = KeyError()\n        with assert_raises(KeyError) as context:\n            raise exc\n        assert_is(exc, context.exc_val)\n\n    def test_assert_raises__exc_val_within_context(self):\n        with assert_raises(RuntimeError):\n            with assert_raises(KeyError) as context:\n                context.exc_val\n\n    def test_assert_raises__raises_subclass(self):\n        class MyError(IndexError):\n            pass\n\n        with assert_raises(IndexError):\n            raise MyError()\n\n    def test_assert_raises__exception_not_raised__default_message(self):\n        with _assert_raises_assertion(\"KeyError not raised\"):\n            with assert_raises(KeyError):\n                pass\n\n    def test_assert_raises__exception_not_raised__custom_message(self):\n        expected = \"KeyError not raised;KeyError;KeyError\"\n        with _assert_raises_assertion(expected):\n            with assert_raises(\n                KeyError, msg_fmt=\"{msg};{exc_type.__name__};{exc_name}\"\n            ):\n                pass\n\n    def test_assert_raises__wrong_exception_raised(self):\n        try:\n            with assert_raises(IndexError):\n                raise KeyError()\n        except KeyError:\n            pass\n        except Exception as exc:\n            fail(str(exc) + \" was raised\")\n        else:\n            fail(\"no exception raised\")\n\n    def test_assert_raises__add_test_called(self):\n        called = Box(False)\n\n        def extra_test(exc):\n            assert_is_instance(exc, KeyError)\n            called.value = True\n\n        with assert_raises(KeyError) as context:\n            context.add_test(extra_test)\n            raise KeyError()\n        assert_true(called.value, \"extra_test() was not called\")\n\n    def test_assert_raises__add_test_not_called(self):\n        called = Box(False)\n\n        def extra_test(_):\n            called.value = True\n\n        with assert_raises(AssertionError):\n            with assert_raises(KeyError) as context:\n                context.add_test(extra_test)\n        assert_false(called.value, \"extra_test() was unexpectedly called\")\n\n    # assert_raises_regex()\n\n    def test_assert_raises_regex__raises_right_exception(self):\n        with assert_raises_regex(KeyError, r\"test.*\"):\n            raise KeyError(\"test message\")\n\n    def test_assert_raises_regex__raises_right_exception__compiled(self):\n        with assert_raises_regex(KeyError, re.compile(r\"test.*\")):\n            raise KeyError(\"test message\")\n\n    def test_assert_raises_regex__exception_not_raised__default_message(self):\n        with _assert_raises_assertion(\"KeyError not raised\"):\n            with assert_raises_regex(KeyError, r\"test\"):\n                pass\n\n    def test_assert_raises_regex__exception_not_raised__custom_message(self):\n        expected = \"KeyError not raised;KeyError;KeyError;'';test\"\n        with _assert_raises_assertion(expected):\n            msg_fmt = \"{msg};{exc_type.__name__};{exc_name};{text!r};{pattern}\"\n            with assert_raises_regex(KeyError, r\"test\", msg_fmt=msg_fmt):\n                pass\n\n    def test_assert_raises_regex__no_message__default_message(self):\n        with _assert_raises_assertion(\"KeyError without message\"):\n            with assert_raises_regex(KeyError, r\"test\"):\n                raise KeyError()\n\n    def test_assert_raises_regex__no_message__custom_message(self):\n        expected = \"KeyError without message;KeyError;KeyError;None;test\"\n        with _assert_raises_assertion(expected):\n            msg_fmt = \"{msg};{exc_type.__name__};{exc_name};{text!r};{pattern}\"\n            with assert_raises_regex(KeyError, r\"test\", msg_fmt=msg_fmt):\n                raise KeyError()\n\n    def test_assert_raises_regex__wrong_exception_raised(self):\n        try:\n            with assert_raises_regex(IndexError, \"test message\"):\n                raise KeyError(\"test message\")\n        except KeyError:\n            pass\n        except Exception as exc:\n            fail(str(exc) + \" was raised\")\n        else:\n            fail(\"no exception raised\")\n\n    def test_assert_raises_regex__wrong_error__default_message(self):\n        with _assert_raises_assertion(\"'wrong message' does not match 'test'\"):\n            with assert_raises_regex(KeyError, r\"test\"):\n                raise KeyError(\"wrong message\")\n\n    def test_assert_raises_regex__wrong_error__pattern_default_message(self):\n        with _assert_raises_assertion(\"'wrong message' does not match 'test'\"):\n            with assert_raises_regex(KeyError, re.compile(r\"test\")):\n                raise KeyError(\"wrong message\")\n\n    def test_assert_raises_regex__wrong_error__custom_message(self):\n        expected = (\n            \"'wrong message' does not match 'test';KeyError;KeyError;\"\n            \"'wrong message';test\"\n        )\n        with _assert_raises_assertion(expected):\n            msg_fmt = \"{msg};{exc_type.__name__};{exc_name};{text!r};{pattern}\"\n            with assert_raises_regex(KeyError, r\"test\", msg_fmt=msg_fmt):\n                raise KeyError(\"wrong message\")\n\n    # assert_raises_errno()\n\n    def test_assert_raises_errno__right_errno(self):\n        with assert_raises_errno(OSError, 20):\n            raise OSError(20, \"Test error\")\n\n    def test_assert_raises_errno__no_exception_raised__default_message(self):\n        with _assert_raises_assertion(\"OSError not raised\"):\n            with assert_raises_errno(OSError, 20):\n                pass\n\n    def test_assert_raises_errno__no_exception_raised__custom_message(self):\n        expected = \"OSError not raised;OSError;OSError;20;None\"\n        with _assert_raises_assertion(expected):\n            msg_fmt = (\n                \"{msg};{exc_type.__name__};{exc_name};{expected_errno};\"\n                \"{actual_errno}\"\n            )\n            with assert_raises_errno(OSError, 20, msg_fmt=msg_fmt):\n                pass\n\n    def test_assert_raises_errno__wrong_class_raised(self):\n        class RightClass(OSError):\n            pass\n\n        class WrongClass(OSError):\n            pass\n\n        try:\n            with assert_raises_errno(RightClass, 20):\n                raise WrongClass(20, \"Test error\")\n        except WrongClass:\n            pass\n        else:\n            raise AssertionError(\"WrongClass was not raised\")\n\n    def test_assert_raises_errno__wrong_errno__default_message(self):\n        with _assert_raises_assertion(\"wrong errno: 20 != 1\"):\n            with assert_raises_errno(OSError, 20):\n                raise OSError(1, \"Test error\")\n\n    def test_assert_raises_errno__wrong_errno__custom_message(self):\n        expected = \"wrong errno: 20 != 1;OSError;OSError;20;1\"\n        with _assert_raises_assertion(expected):\n            msg_fmt = (\n                \"{msg};{exc_type.__name__};{exc_name};{expected_errno};\"\n                \"{actual_errno}\"\n            )\n            with assert_raises_errno(OSError, 20, msg_fmt=msg_fmt):\n                raise OSError(1, \"Test error\")\n\n    # assert_succeeds()\n\n    def test_assert_succeeds__no_exception_raised(self):\n        with assert_succeeds(KeyError):\n            pass\n\n    def test_assert_succeeds__expected_exception__default_message(self):\n        with _assert_raises_assertion(\"KeyError was unexpectedly raised\"):\n            with assert_succeeds(KeyError):\n                raise KeyError()\n\n    def test_assert_succeeds__expected_exception__custom_message(self):\n        expected = (\n            \"KeyError was unexpectedly raised;KeyError;KeyError;test error\"\n        )\n        with _assert_raises_assertion(expected):\n            msg_fmt = (\n                \"{msg};{exc_type.__name__};{exc_name};{exception.args[0]}\"\n            )\n            with assert_succeeds(KeyError, msg_fmt=msg_fmt):\n                raise KeyError(\"test error\")\n\n    def test_assert_succeeds__unexpected_exception(self):\n        try:\n            with assert_succeeds(ValueError):\n                raise KeyError()\n        except KeyError:\n            pass\n        else:\n            raise AssertionError(\"KeyError was not raised\")\n\n    # assert_warns()\n\n    def test_assert_warns__default_message(self):\n        with assert_raises_regex(AssertionError, r\"^ImportWarning not issued\"):\n            with assert_warns(ImportWarning):\n                pass\n\n    def test_assert_warns__custom_message(self):\n        exception = \"ImportWarning not issued;ImportWarning;ImportWarning\"\n        with _assert_raises_assertion(exception):\n            msg_fmt = \"{msg};{exc_type.__name__};{exc_name}\"\n            with assert_warns(ImportWarning, msg_fmt=msg_fmt):\n                pass\n\n    def test_assert_warns__warned(self):\n        with assert_succeeds(AssertionError):\n            with assert_warns(FutureWarning):\n                warn(\"foo\", FutureWarning)\n\n    def test_assert_warns__not_warned(self):\n        with assert_raises(AssertionError):\n            with assert_warns(ImportWarning):\n                pass\n\n    def test_assert_warns__wrong_type(self):\n        with assert_raises(AssertionError):\n            with assert_warns(ImportWarning):\n                warn(\"foo\", UnicodeWarning)\n\n    def test_assert_warns__multiple_warnings(self):\n        with assert_succeeds(AssertionError):\n            with assert_warns(UserWarning):\n                warn(\"foo\", UnicodeWarning)\n                warn(\"bar\", UserWarning)\n                warn(\"baz\", FutureWarning)\n\n    def test_assert_warns__warning_handler_deinstalled_on_success(self):\n        with catch_warnings(record=True) as warnings:\n            with assert_warns(UserWarning):\n                warn(\"foo\", UserWarning)\n            assert warnings is not None\n            assert_equal(0, len(warnings))\n            warn(\"bar\", UserWarning)\n            assert_equal(1, len(warnings))\n\n    def test_assert_warns__warning_handler_deinstalled_on_failure(self):\n        with catch_warnings(record=True) as warnings:\n            try:\n                with assert_warns(UserWarning):\n                    pass\n            except AssertionError:\n                pass\n            assert warnings is not None\n            assert_equal(0, len(warnings))\n            warn(\"bar\", UserWarning)\n            assert_equal(1, len(warnings))\n\n    def test_assert_warns__add_test_called(self):\n        called = Box(False)\n\n        def extra_test(warning):\n            assert_is(warning.category, UserWarning)\n            called.value = True\n            return True\n\n        with assert_warns(UserWarning) as context:\n            context.add_test(extra_test)\n            warn(\"bar\", UserWarning)\n        assert_true(called.value, \"extra_test() was not called\")\n\n    def test_assert_warns__add_test_not_called(self):\n        called = Box(False)\n\n        def extra_test(_: Warning) -> bool:\n            called.value = True\n            return False\n\n        with assert_raises(AssertionError):\n            with assert_warns(UserWarning) as context:\n                context.add_test(extra_test)\n        assert_false(called.value, \"extra_test() was unexpectedly called\")\n\n    # assert_warns_regex()\n\n    def test_assert_warns_regex__warned(self):\n        with assert_succeeds(AssertionError):\n            with assert_warns_regex(FutureWarning, r\"fo+\"):\n                warn(\"foo\", FutureWarning)\n\n    def test_assert_warns_regex__warning_text_matches_in_the_middle(self):\n        with assert_succeeds(AssertionError):\n            with assert_warns_regex(FutureWarning, r\"o\"):\n                warn(\"foo\", FutureWarning)\n\n    def test_assert_warns_regex__not_warned(self):\n        with assert_raises(AssertionError):\n            with assert_warns_regex(UserWarning, r\"foo\"):\n                pass\n\n    def test_assert_warns_regex__wrong_type(self):\n        with assert_raises(AssertionError):\n            with assert_warns_regex(ImportWarning, r\"foo\"):\n                warn(\"foo\", UnicodeWarning)\n\n    def test_assert_warns_regex__wrong_message(self):\n        with assert_raises(AssertionError):\n            with assert_warns_regex(UnicodeWarning, r\"foo\"):\n                warn(\"bar\", UnicodeWarning)\n\n    def test_assert_warns_regex__multiple_warnings(self):\n        with assert_succeeds(AssertionError):\n            with assert_warns_regex(UserWarning, r\"bar2\"):\n                warn(\"foo\", UnicodeWarning)\n                warn(\"bar1\", UserWarning)\n                warn(\"bar2\", UserWarning)\n                warn(\"bar3\", UserWarning)\n                warn(\"baz\", FutureWarning)\n\n    def test_assert_warns_regex__warning_handler_deinstalled_on_success(self):\n        with catch_warnings(record=True) as warnings:\n            with assert_warns_regex(UserWarning, r\"foo\"):\n                warn(\"foo\", UserWarning)\n            assert warnings is not None\n            assert_equal(0, len(warnings))\n            warn(\"bar\", UserWarning)\n            assert_equal(1, len(warnings))\n\n    def test_assert_warns_regex__warning_handler_deinstalled_on_failure(self):\n        with catch_warnings(record=True) as warnings:\n            try:\n                with assert_warns_regex(UserWarning, r\"\"):\n                    pass\n            except AssertionError:\n                pass\n            assert warnings is not None\n            assert_equal(0, len(warnings))\n            warn(\"bar\", UserWarning)\n            assert_equal(1, len(warnings))\n\n    def test_assert_warns_regex__not_issued__default_message(self):\n        with _assert_raises_assertion(\n            \"no UserWarning matching 'foo.*bar' issued\"\n        ):\n            with assert_warns_regex(UserWarning, r\"foo.*bar\"):\n                pass\n\n    def test_assert_warns_regex__not_issued__custom_message(self):\n        expected = \"no ImportWarning matching 'abc' issued;ImportWarning;ImportWarning;abc\"\n        with _assert_raises_assertion(expected):\n            msg_fmt = \"{msg};{exc_type.__name__};{exc_name};{pattern}\"\n            with assert_warns_regex(ImportWarning, r\"abc\", msg_fmt=msg_fmt):\n                pass\n\n    def test_assert_warns_regex__wrong_message__default_message(self):\n        with _assert_raises_assertion(\n            \"no UserWarning matching 'foo.*bar' issued\"\n        ):\n            with assert_warns_regex(UserWarning, r\"foo.*bar\"):\n                pass\n\n    def test_assert_warns_regex__wrong_message__custom_message(self):\n        expected = (\n            \"no UserWarning matching 'foo.*bar' issued;UserWarning;\"\n            \"UserWarning;foo.*bar\"\n        )\n        with _assert_raises_assertion(expected):\n            msg_fmt = \"{msg};{exc_type.__name__};{exc_name};{pattern}\"\n            with assert_warns_regex(UserWarning, r\"foo.*bar\", msg_fmt=msg_fmt):\n                pass\n\n    # assert_json_subset()\n\n    def test_assert_json_subset__different_types(self):\n        with _assert_raises_assertion(\"element $ differs: {} != []\"):\n            assert_json_subset({}, [])\n\n    def test_assert_json_subset__empty_objects(self):\n        with assert_succeeds(AssertionError):\n            assert_json_subset({}, {})\n\n    def test_assert_json_subset__objects_equal(self):\n        with assert_succeeds(AssertionError):\n            assert_json_subset(\n                {\"foo\": 3, \"bar\": \"abc\"}, {\"bar\": \"abc\", \"foo\": 3}\n            )\n\n    def test_assert_json_subset__one_key_missing_from_first_object(self):\n        with assert_succeeds(AssertionError):\n            assert_json_subset({\"foo\": 3}, {\"foo\": 3, \"bar\": 3})\n\n    def test_assert_json_subset__one_key_missing_from_second_object(self):\n        with _assert_raises_assertion(\"element 'bar' missing from element $\"):\n            assert_json_subset({\"foo\": 3, \"bar\": 3}, {\"foo\": 3})\n\n    def test_assert_json_subset__multiple_keys_missing_from_second_object(\n        self,\n    ):\n        with _assert_raises_assertion(\n            \"elements 'bar', 'baz', and 'foo' missing from element $\"\n        ):\n            assert_json_subset({\"foo\": 3, \"bar\": 3, \"baz\": 3}, {})\n\n    def test_assert_json_subset__value_differs(self):\n        with _assert_raises_assertion(\"element $['foo'] differs: 3 != 4\"):\n            assert_json_subset({\"foo\": 3}, {\"foo\": 4})\n\n    def test_assert_json_subset__empty_lists(self):\n        with assert_succeeds(AssertionError):\n            assert_json_subset([], [])\n\n    def test_assert_json_subset__different_sized_lists(self):\n        with _assert_raises_assertion(\"JSON array $ differs in size: 2 != 1\"):\n            assert_json_subset([1, 2], [1])\n        with _assert_raises_assertion(\"JSON array $ differs in size: 1 != 2\"):\n            assert_json_subset([1], [1, 2])\n\n    def test_assert_json_subset__different_list_values(self):\n        with _assert_raises_assertion(\"element $[0] differs: {} != []\"):\n            assert_json_subset([{}], [[]])\n\n    def test_assert_json_subset__fundamental_types_differ(self):\n        with _assert_raises_assertion(\"element $[0] differs: 1 != 'foo'\"):\n            assert_json_subset([1], [\"foo\"])\n\n    def test_assert_json_subset__fundamental_values_differ(self):\n        with _assert_raises_assertion(\"element $[0] differs: 'bar' != 'foo'\"):\n            assert_json_subset([\"bar\"], [\"foo\"])\n\n    def test_assert_json_subset__none(self):\n        with assert_succeeds(AssertionError):\n            assert_json_subset([None], [None])\n        with _assert_raises_assertion(\"element $[0] differs: 42 != None\"):\n            assert_json_subset([42], [None])\n        with _assert_raises_assertion(\"element $[0] differs: None != 42\"):\n            assert_json_subset([None], [42])\n\n    def test_assert_json_subset__compare_int_and_float(self):\n        with assert_succeeds(AssertionError):\n            assert_json_subset([42], [42.0])\n            assert_json_subset([42.0], [42])\n\n    def test_assert_json_subset__unsupported_type(self):\n        msg = \"unsupported type <{} 'set'>\".format(self._type_string)\n        with assert_raises_regex(TypeError, msg):\n            assert_json_subset([set()], [set()])\n\n    def test_assert_json_subset__subtypes(self):\n        with assert_succeeds(AssertionError):\n            assert_json_subset(OrderedDict(), {})\n            assert_json_subset({}, OrderedDict())\n\n    def test_assert_json_subset__second_is_string(self):\n        with assert_succeeds(AssertionError):\n            assert_json_subset({}, \"{  }\")\n\n    def test_assert_json_subset__second_is_unsupported_json_string(self):\n        msg = \"second must decode to dict or list, not <{} 'int'>\".format(\n            self._type_string\n        )\n        with _assert_raises_assertion(msg):\n            assert_json_subset({}, \"42\")\n\n    def test_assert_json_subset__second_is_invalid_json_string(self):\n        with assert_raises(JSONDecodeError):\n            assert_json_subset({}, \",\")\n\n    def test_assert_json_subset__second_is_bytes(self):\n        with assert_succeeds(AssertionError):\n            assert_json_subset([\"föo\"], '[\"föo\"]'.encode(\"utf-8\"))\n\n    def test_assert_json_subset__second_is_latin1_bytes(self):\n        with assert_raises(UnicodeDecodeError):\n            assert_json_subset([\"föo\"], '[\"föo\"]'.encode(\"iso-8859-1\"))\n\n    def test_assert_json_subset__invalid_type(self):\n        with assert_raises_regex(\n            TypeError, \"second must be dict, list, str, or bytes\"\n        ):\n            assert_json_subset({}, 42)  # type: ignore[arg-type]\n\n    def test_assert_json_subset__element_name_not_str(self) -> None:\n        with assert_raises_regex(\n            TypeError,\n            \"12 is not a valid object member name\",\n        ):\n            assert_json_subset({12: 34}, \"{}\")\n\n    def test_assert_json_subset__presence_check(self) -> None:\n        with assert_succeeds(AssertionError):\n            assert_json_subset({\"foo\": Present}, {\"foo\": \"bar\"})\n        with assert_succeeds(AssertionError):\n            assert_json_subset({\"foo\": Present()}, {\"foo\": \"bar\"})\n        with assert_raises_regex(\n            AssertionError,\n            r\"element 'foo' missing from element \\$\",\n        ):\n            assert_json_subset({\"foo\": Present}, {})\n        with assert_raises_regex(\n            AssertionError,\n            r\"element 'foo' missing from element \\$\",\n        ):\n            assert_json_subset({\"foo\": Present()}, {})\n        with assert_succeeds(AssertionError):\n            assert_json_subset({\"foo\": Absent}, {})\n        with assert_succeeds(AssertionError):\n            assert_json_subset({\"foo\": Absent()}, {})\n        with assert_raises_regex(\n            AssertionError,\n            r\"spurious member 'foo' in object \\$\",\n        ):\n            assert_json_subset({\"foo\": Absent}, {\"foo\": \"bar\"})\n        with assert_raises_regex(\n            AssertionError,\n            r\"spurious member 'foo' in object \\$\",\n        ):\n            assert_json_subset({\"foo\": Absent()}, {\"foo\": \"bar\"})\n\n    def test_assert_json_subset__existence_check_old(self) -> None:\n        with catch_warnings():\n            simplefilter(\"ignore\")\n            with assert_succeeds(AssertionError):\n                assert_json_subset({Exists(\"foo\"): True}, {\"foo\": \"bar\"})\n            with assert_raises_regex(\n                AssertionError,\n                r\"element 'foo' missing from element \\$\",\n            ):\n                assert_json_subset({Exists(\"foo\"): True}, {})\n            with assert_succeeds(AssertionError):\n                assert_json_subset({Exists(\"foo\"): False}, {})\n            with assert_raises_regex(\n                AssertionError,\n                r\"spurious member 'foo' in object \\$\",\n            ):\n                assert_json_subset({Exists(\"foo\"): False}, {\"foo\": \"bar\"})\n"
    },
    {
      "path": "srittau_python-asserts/CHANGELOG.md",
      "content": "# Changelog for python-asserts\n\npython-asserts adheres to [semantic versioning](https://semver.org/).\n\n## UNRELEASED –\n\n## [0.13.1] – 2024-04-29\n\n### Fixed\n\nFixed Python 3.12 deprecation warnings.\n\n## [0.13.0] – 2024-03-13\n\n### Added\n\n- Add support for Python 3.12.\n- Add `Present` and `Absent` for absence checks in `assert_json_subset()`.\n\n### Removed\n\n- Drop support for Python 3.7.\n\n### Deprecated\n\n- Deprecate `Exists` in favor of `Present` and `Absent` in\n  `assert_json_subset()`.\n\n## [0.12.0]\n\n### Added\n\n- Add `assert_not_regex()`.\n\n### Changed\n\n- Modernize the type stubs.\n\n### Removed\n\n- Drop support for Python 3.6.\n\n## [0.11.1]\n\n### Added\n\n- `assert_json_subset()` can now check for the existence or non-existence\n  of object members using the new `Exists` helper.\n- Non-string (or `Exists`) object member names in the first argument to\n  `assert_json_subset()` now raise a `TypeError`.\n\n## [0.11.0]\n\n### Removed\n\n- Drop support for Python 2.7 and 3.5.\n\n## [0.10.0]\n\n### Added\n\n- `AssertRaisesContext` and `AssertWarnsContext` now return themselves\n  when `__enter__()` is called. By extension it now easier to call\n  `add_test()` with `assert_raises()` et al:\n\n```python\nwith assert_raises(KeyError) as context:\n    context.add_test(...)\n    ...\n```\n\n- Add `AssertRaisesContext.exc_val` property to access the caught\n  exception after leaving the context manager:\n\n```python\nwith assert_raises(KeyError) as context:\n    ...\nassert_equal(\"expected message\", str(context.exc_val))\n```\n\n### Removed\n\n- Drop support for Python 3.4.\n\n## [0.9.1]\n\n### Changed\n\n- `AssertRaisesContext` and sub-classes are now generic over the\n  exception type.\n\n## [0.9.0]\n\n### Added\n\n- Add `assert_json_subset()`.\n\n## [0.8.6]\n\n### Added\n\n- Add support for Python 3.7 (contributed by Frank Niessink).\n\n## [0.8.5]\n\n### Added\n\n- Add `assert_dict_equal()`.\n- Add `assert_dict_superset()`.\n\n### Changed\n\n- `assert_equal()`: Use `assert_dict_equal()` if applicable.\n\n## [0.8.4]\n\n### Changed\n\n- `fail()` is now marked with `NoReturn` in type stub.\n\n### Fixed\n\n- Improve type annotations for Python 2.\n\n## [0.8.3]\n\n### Fixed\n\n- Fix type signature of `AssertRaisesContext.__exit__()`.\n\n## [0.8.2]\n\n### Added\n\n- Add a py.typed file to signal that this package supports type hints.\n\n## [0.8.1]\n\n### Fixed\n\n- `assert_raises_regex()`: Handle exceptions without any message correctly.\n\n## [0.8.0]\n\n### Added\n\n- assert_count_equal(): Add `msg_fmt` argument.\n- Add AssertRaisesErrnoContext, AssertRaisesRegexContext, and\n  AssertWarnsRegexContext.\n\n### Changed\n\n- Replace `msg` argument with `msg_fmt` in all assertions (except `fail()`).\n  This allows you to customize error messages more easily than before, because\n  `format()` with appropriate keyword arguments is now called on these\n  strings. See the documentation of individual assertions for the supported\n  arguments.\n- Replace AssertRaisesContext.msg and AssertWarnsContext.msg with msg_fmt.\n- assert_almost_equal(), assert_not_almost_equal(): Place msg_fmt as third\n  argument.\n\n## [0.7.3]\n\n### Added\n\n- Add assert_not_almost_equal().\n\n### Changed\n\n- assert_almost_equal(): Raise ValueError if diff <= 0.\n\n### Fixed\n\n- assert_almost_equal() would never fail if a delta was supplied and the\n  second number was smaller than the first.\n- Use fail() instead of raise AssertionError in a few assertions.\n\n## [0.7.2]\n\n### Added\n\n- Add assert_warns() and assert_warns_regex().\n\n## [0.7.1]\n\n### Changed\n\n- Distribute a wheel.\n- asserts is now a package, instead of a module.\n\n## [0.7.0]\n\n### Added\n\n- Add a stub file.\n- Add assert_count_equal().\n\n## [0.6]\n\n### Added\n\n- Add assert_less(), assert_less_equal(), assert_greater(), and\n  assert_greater_equal().\n- Add assert_not_is_instance().\n\n### Changed\n\n- assert_datetime_about_now()/assert_datetime_about_now_utc(): Handle\n  comparison with None more gracefully.\n\n## [0.5.1]\n\n### Added\n\n- Add the LICENSE file to the distribution.\n\n## [0.5]\n\nInitial release.\n"
    },
    {
      "path": "srittau_python-asserts/mypy.ini",
      "content": "[mypy]\ncheck_untyped_defs = True\ndisallow_subclassing_any = True\ndisallow_untyped_decorators = True\nno_implicit_optional = True\nno_implicit_reexport = True\nwarn_redundant_casts = True\nwarn_return_any = True\nwarn_unused_configs = True\nwarn_unused_ignores = True\n"
    },
    {
      "path": "srittau_python-asserts/poetry.lock",
      "content": "# This file is automatically @generated by Poetry 1.8.2 and should not be changed by hand.\n\n[[package]]\nname = \"mypy\"\nversion = \"1.10.0\"\ndescription = \"Optional static typing for Python\"\noptional = false\npython-versions = \">=3.8\"\nfiles = [\n    {file = \"mypy-1.10.0-cp310-cp310-macosx_10_9_x86_64.whl\", hash = \"sha256:da1cbf08fb3b851ab3b9523a884c232774008267b1f83371ace57f412fe308c2\"},\n    {file = \"mypy-1.10.0-cp310-cp310-macosx_11_0_arm64.whl\", hash = \"sha256:12b6bfc1b1a66095ab413160a6e520e1dc076a28f3e22f7fb25ba3b000b4ef99\"},\n    {file = \"mypy-1.10.0-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl\", hash = \"sha256:9e36fb078cce9904c7989b9693e41cb9711e0600139ce3970c6ef814b6ebc2b2\"},\n    {file = \"mypy-1.10.0-cp310-cp310-musllinux_1_1_x86_64.whl\", hash = \"sha256:2b0695d605ddcd3eb2f736cd8b4e388288c21e7de85001e9f85df9187f2b50f9\"},\n    {file = \"mypy-1.10.0-cp310-cp310-win_amd64.whl\", hash = \"sha256:cd777b780312ddb135bceb9bc8722a73ec95e042f911cc279e2ec3c667076051\"},\n    {file = \"mypy-1.10.0-cp311-cp311-macosx_10_9_x86_64.whl\", hash = \"sha256:3be66771aa5c97602f382230165b856c231d1277c511c9a8dd058be4784472e1\"},\n    {file = \"mypy-1.10.0-cp311-cp311-macosx_11_0_arm64.whl\", hash = \"sha256:8b2cbaca148d0754a54d44121b5825ae71868c7592a53b7292eeb0f3fdae95ee\"},\n    {file = \"mypy-1.10.0-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl\", hash = \"sha256:1ec404a7cbe9fc0e92cb0e67f55ce0c025014e26d33e54d9e506a0f2d07fe5de\"},\n    {file = \"mypy-1.10.0-cp311-cp311-musllinux_1_1_x86_64.whl\", hash = \"sha256:e22e1527dc3d4aa94311d246b59e47f6455b8729f4968765ac1eacf9a4760bc7\"},\n    {file = \"mypy-1.10.0-cp311-cp311-win_amd64.whl\", hash = \"sha256:a87dbfa85971e8d59c9cc1fcf534efe664d8949e4c0b6b44e8ca548e746a8d53\"},\n    {file = \"mypy-1.10.0-cp312-cp312-macosx_10_9_x86_64.whl\", hash = \"sha256:a781f6ad4bab20eef8b65174a57e5203f4be627b46291f4589879bf4e257b97b\"},\n    {file = \"mypy-1.10.0-cp312-cp312-macosx_11_0_arm64.whl\", hash = \"sha256:b808e12113505b97d9023b0b5e0c0705a90571c6feefc6f215c1df9381256e30\"},\n    {file = \"mypy-1.10.0-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl\", hash = \"sha256:8f55583b12156c399dce2df7d16f8a5095291354f1e839c252ec6c0611e86e2e\"},\n    {file = \"mypy-1.10.0-cp312-cp312-musllinux_1_1_x86_64.whl\", hash = \"sha256:4cf18f9d0efa1b16478c4c129eabec36148032575391095f73cae2e722fcf9d5\"},\n    {file = \"mypy-1.10.0-cp312-cp312-win_amd64.whl\", hash = \"sha256:bc6ac273b23c6b82da3bb25f4136c4fd42665f17f2cd850771cb600bdd2ebeda\"},\n    {file = \"mypy-1.10.0-cp38-cp38-macosx_10_9_x86_64.whl\", hash = \"sha256:9fd50226364cd2737351c79807775136b0abe084433b55b2e29181a4c3c878c0\"},\n    {file = \"mypy-1.10.0-cp38-cp38-macosx_11_0_arm64.whl\", hash = \"sha256:f90cff89eea89273727d8783fef5d4a934be2fdca11b47def50cf5d311aff727\"},\n    {file = \"mypy-1.10.0-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl\", hash = \"sha256:fcfc70599efde5c67862a07a1aaf50e55bce629ace26bb19dc17cece5dd31ca4\"},\n    {file = \"mypy-1.10.0-cp38-cp38-musllinux_1_1_x86_64.whl\", hash = \"sha256:075cbf81f3e134eadaf247de187bd604748171d6b79736fa9b6c9685b4083061\"},\n    {file = \"mypy-1.10.0-cp38-cp38-win_amd64.whl\", hash = \"sha256:3f298531bca95ff615b6e9f2fc0333aae27fa48052903a0ac90215021cdcfa4f\"},\n    {file = \"mypy-1.10.0-cp39-cp39-macosx_10_9_x86_64.whl\", hash = \"sha256:fa7ef5244615a2523b56c034becde4e9e3f9b034854c93639adb667ec9ec2976\"},\n    {file = \"mypy-1.10.0-cp39-cp39-macosx_11_0_arm64.whl\", hash = \"sha256:3236a4c8f535a0631f85f5fcdffba71c7feeef76a6002fcba7c1a8e57c8be1ec\"},\n    {file = \"mypy-1.10.0-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl\", hash = \"sha256:4a2b5cdbb5dd35aa08ea9114436e0d79aceb2f38e32c21684dcf8e24e1e92821\"},\n    {file = \"mypy-1.10.0-cp39-cp39-musllinux_1_1_x86_64.whl\", hash = \"sha256:92f93b21c0fe73dc00abf91022234c79d793318b8a96faac147cd579c1671746\"},\n    {file = \"mypy-1.10.0-cp39-cp39-win_amd64.whl\", hash = \"sha256:28d0e038361b45f099cc086d9dd99c15ff14d0188f44ac883010e172ce86c38a\"},\n    {file = \"mypy-1.10.0-py3-none-any.whl\", hash = \"sha256:f8c083976eb530019175aabadb60921e73b4f45736760826aa1689dda8208aee\"},\n    {file = \"mypy-1.10.0.tar.gz\", hash = \"sha256:3d087fcbec056c4ee34974da493a826ce316947485cef3901f511848e687c131\"},\n]\n\n[package.dependencies]\nmypy-extensions = \">=1.0.0\"\ntomli = {version = \">=1.1.0\", markers = \"python_version < \\\"3.11\\\"\"}\ntyping-extensions = \">=4.1.0\"\n\n[package.extras]\ndmypy = [\"psutil (>=4.0)\"]\ninstall-types = [\"pip\"]\nmypyc = [\"setuptools (>=50)\"]\nreports = [\"lxml\"]\n\n[[package]]\nname = \"mypy-extensions\"\nversion = \"1.0.0\"\ndescription = \"Type system extensions for programs checked with the mypy type checker.\"\noptional = false\npython-versions = \">=3.5\"\nfiles = [\n    {file = \"mypy_extensions-1.0.0-py3-none-any.whl\", hash = \"sha256:4392f6c0eb8a5668a69e23d168ffa70f0be9ccfd32b5cc2d26a34ae5b844552d\"},\n    {file = \"mypy_extensions-1.0.0.tar.gz\", hash = \"sha256:75dbf8955dc00442a438fc4d0666508a9a97b6bd41aa2f0ffe9d2f2725af0782\"},\n]\n\n[[package]]\nname = \"pastel\"\nversion = \"0.2.1\"\ndescription = \"Bring colors to your terminal.\"\noptional = false\npython-versions = \">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*\"\nfiles = [\n    {file = \"pastel-0.2.1-py2.py3-none-any.whl\", hash = \"sha256:4349225fcdf6c2bb34d483e523475de5bb04a5c10ef711263452cb37d7dd4364\"},\n    {file = \"pastel-0.2.1.tar.gz\", hash = \"sha256:e6581ac04e973cac858828c6202c1e1e81fee1dc7de7683f3e1ffe0bfd8a573d\"},\n]\n\n[[package]]\nname = \"poethepoet\"\nversion = \"0.26.1\"\ndescription = \"A task runner that works well with poetry.\"\noptional = false\npython-versions = \">=3.8\"\nfiles = [\n    {file = \"poethepoet-0.26.1-py3-none-any.whl\", hash = \"sha256:aa43b443fec5d17d7e76771cccd484e5285805301721a74f059c483ad3276edd\"},\n    {file = \"poethepoet-0.26.1.tar.gz\", hash = \"sha256:aaad8541f6072617a60bcff2562d00779b58b353bd0f1847b06d8d0f2b6dc192\"},\n]\n\n[package.dependencies]\npastel = \">=0.2.1,<0.3.0\"\ntomli = \">=1.2.2\"\n\n[package.extras]\npoetry-plugin = [\"poetry (>=1.0,<2.0)\"]\n\n[[package]]\nname = \"ruff\"\nversion = \"0.4.8\"\ndescription = \"An extremely fast Python linter and code formatter, written in Rust.\"\noptional = false\npython-versions = \">=3.7\"\nfiles = [\n    {file = \"ruff-0.4.8-py3-none-macosx_10_12_x86_64.whl\", hash = \"sha256:7663a6d78f6adb0eab270fa9cf1ff2d28618ca3a652b60f2a234d92b9ec89066\"},\n    {file = \"ruff-0.4.8-py3-none-macosx_11_0_arm64.whl\", hash = \"sha256:eeceb78da8afb6de0ddada93112869852d04f1cd0f6b80fe464fd4e35c330913\"},\n    {file = \"ruff-0.4.8-py3-none-manylinux_2_17_aarch64.manylinux2014_aarch64.whl\", hash = \"sha256:aad360893e92486662ef3be0a339c5ca3c1b109e0134fcd37d534d4be9fb8de3\"},\n    {file = \"ruff-0.4.8-py3-none-manylinux_2_17_armv7l.manylinux2014_armv7l.whl\", hash = \"sha256:284c2e3f3396fb05f5f803c9fffb53ebbe09a3ebe7dda2929ed8d73ded736deb\"},\n    {file = \"ruff-0.4.8-py3-none-manylinux_2_17_i686.manylinux2014_i686.whl\", hash = \"sha256:a7354f921e3fbe04d2a62d46707e569f9315e1a613307f7311a935743c51a764\"},\n    {file = \"ruff-0.4.8-py3-none-manylinux_2_17_ppc64.manylinux2014_ppc64.whl\", hash = \"sha256:72584676164e15a68a15778fd1b17c28a519e7a0622161eb2debdcdabdc71883\"},\n    {file = \"ruff-0.4.8-py3-none-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl\", hash = \"sha256:9678d5c9b43315f323af2233a04d747409d1e3aa6789620083a82d1066a35199\"},\n    {file = \"ruff-0.4.8-py3-none-manylinux_2_17_s390x.manylinux2014_s390x.whl\", hash = \"sha256:704977a658131651a22b5ebeb28b717ef42ac6ee3b11e91dc87b633b5d83142b\"},\n    {file = \"ruff-0.4.8-py3-none-manylinux_2_17_x86_64.manylinux2014_x86_64.whl\", hash = \"sha256:d05f8d6f0c3cce5026cecd83b7a143dcad503045857bc49662f736437380ad45\"},\n    {file = \"ruff-0.4.8-py3-none-musllinux_1_2_aarch64.whl\", hash = \"sha256:6ea874950daca5697309d976c9afba830d3bf0ed66887481d6bca1673fc5b66a\"},\n    {file = \"ruff-0.4.8-py3-none-musllinux_1_2_armv7l.whl\", hash = \"sha256:fc95aac2943ddf360376be9aa3107c8cf9640083940a8c5bd824be692d2216dc\"},\n    {file = \"ruff-0.4.8-py3-none-musllinux_1_2_i686.whl\", hash = \"sha256:384154a1c3f4bf537bac69f33720957ee49ac8d484bfc91720cc94172026ceed\"},\n    {file = \"ruff-0.4.8-py3-none-musllinux_1_2_x86_64.whl\", hash = \"sha256:e9d5ce97cacc99878aa0d084c626a15cd21e6b3d53fd6f9112b7fc485918e1fa\"},\n    {file = \"ruff-0.4.8-py3-none-win32.whl\", hash = \"sha256:6d795d7639212c2dfd01991259460101c22aabf420d9b943f153ab9d9706e6a9\"},\n    {file = \"ruff-0.4.8-py3-none-win_amd64.whl\", hash = \"sha256:e14a3a095d07560a9d6769a72f781d73259655919d9b396c650fc98a8157555d\"},\n    {file = \"ruff-0.4.8-py3-none-win_arm64.whl\", hash = \"sha256:14019a06dbe29b608f6b7cbcec300e3170a8d86efaddb7b23405cb7f7dcaf780\"},\n    {file = \"ruff-0.4.8.tar.gz\", hash = \"sha256:16d717b1d57b2e2fd68bd0bf80fb43931b79d05a7131aa477d66fc40fbd86268\"},\n]\n\n[[package]]\nname = \"tomli\"\nversion = \"2.0.1\"\ndescription = \"A lil' TOML parser\"\noptional = false\npython-versions = \">=3.7\"\nfiles = [\n    {file = \"tomli-2.0.1-py3-none-any.whl\", hash = \"sha256:939de3e7a6161af0c887ef91b7d41a53e7c5a1ca976325f429cb46ea9bc30ecc\"},\n    {file = \"tomli-2.0.1.tar.gz\", hash = \"sha256:de526c12914f0c550d15924c62d72abc48d6fe7364aa87328337a31007fe8a4f\"},\n]\n\n[[package]]\nname = \"typing-extensions\"\nversion = \"4.12.2\"\ndescription = \"Backported and Experimental Type Hints for Python 3.8+\"\noptional = false\npython-versions = \">=3.8\"\nfiles = [\n    {file = \"typing_extensions-4.12.2-py3-none-any.whl\", hash = \"sha256:04e5ca0351e0f3f85c6853954072df659d0d13fac324d0072316b67d7794700d\"},\n    {file = \"typing_extensions-4.12.2.tar.gz\", hash = \"sha256:1a7ead55c7e559dd4dee8856e3a88b41225abfe1ce8df57b7c13915fe121ffb8\"},\n]\n\n[metadata]\nlock-version = \"2.0\"\npython-versions = \">=3.8.1\"\ncontent-hash = \"c010d5b718fed30569338ba631a7631751f5f6562c78491c16aa79927662e3f0\"\n"
    },
    {
      "path": "srittau_python-asserts/README.md",
      "content": "# Python Asserts\n\n[![License](https://img.shields.io/pypi/l/asserts.svg)](https://pypi.python.org/pypi/asserts/)\n[![PyPI - Python Version](https://img.shields.io/pypi/pyversions/asserts)](https://pypi.python.org/pypi/asserts/)\n[![GitHub](https://img.shields.io/github/release/srittau/python-asserts/all.svg)](https://github.com/srittau/python-asserts/releases/)\n[![pypi](https://img.shields.io/pypi/v/asserts.svg)](https://pypi.python.org/pypi/asserts/)\n[![GitHub Actions](https://img.shields.io/github/actions/workflow/status/srittau/python-asserts/test-and-lint.yml)](https://github.com/srittau/python-asserts/actions/workflows/test-and-lint.yml)\n\nStand-alone Assertions for Python\n\nThis package provides a few advantages over the assertions provided by\nunittest.TestCase:\n\n- Can be used stand-alone, for example:\n  - In test cases, not derived from TestCase.\n  - In fake and mock classes.\n  - In implementations as rich alternative to the assert statement.\n- PEP 8 compliance.\n- Custom stand-alone assertions can be written easily.\n- Arguably a better separation of concerns, since TestCase is responsible\n  for test running only, if assertion functions are used exclusively.\n\nThere are a few regressions compared to assertions from TestCase:\n\n- The default assertion class (`AssertionError`) can not be overwritten. This\n  is rarely a problem in practice.\n- asserts does not support the `addTypeEqualityFunc()` functionality.\n\nUsage:\n\n```python\n>>> from asserts import assert_true, assert_equal, assert_raises\n>>> my_var = 13\n>>> assert_equal(13, my_var)\n>>> assert_true(True, msg=\"custom failure message\")\n>>> with assert_raises(KeyError):\n...     raise KeyError()\n```\n\nFailure messages can be customized:\n\n```python\n>>> assert_equal(13, 14, msg_fmt=\"{got} is wrong, expected {expected}\")\nTraceback (most recent call last):\n  ...\nAssertionError: 14 is wrong, expected 13\n```\n"
    },
    {
      "path": "srittau_python-asserts/LICENSE",
      "content": "The MIT License (MIT)\n\nCopyright (c) 2014 Sebastian Rittau\n\nPermission is hereby granted, free of charge, to any person obtaining a copy of\nthis software and associated documentation files (the \"Software\"), to deal in\nthe Software without restriction, including without limitation the rights to\nuse, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of\nthe Software, and to permit persons to whom the Software is furnished to do so,\nsubject to the following conditions:\n\nThe above copyright notice and this permission notice shall be included in all\ncopies or substantial portions of the Software.\n\nTHE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\nIMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS\nFOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR\nCOPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER\nIN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN\nCONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.\n"
    },
    {
      "path": "srittau_python-asserts/asserts/__init__.py",
      "content": "\"\"\"\nRich Assertions.\n\nThis module contains several rich standard assertions that can be used in unit\ntests and in implementations. Users are encouraged to define their own\nassertions, possibly using assertions from this package as a basis.\n\n    >>> assert_equal(13, 13)\n    >>> assert_equal(13, 14)\n    Traceback (most recent call last):\n        ...\n    AssertionError: 13 != 14\n    >>> with assert_raises(KeyError):\n    ...     raise KeyError()\n    >>> with assert_raises(KeyError):\n    ...     pass\n    Traceback (most recent call last):\n        ...\n    AssertionError: KeyError not raised\n\n\"\"\"\n\nfrom __future__ import annotations\n\nimport re\nimport sys\nfrom datetime import datetime, timedelta, timezone\nfrom json import loads as json_loads\nfrom typing import Any, Callable, Set\nfrom warnings import WarningMessage, catch_warnings\n\nfrom typing_extensions import deprecated\n\n\ndef fail(msg=None):\n    \"\"\"Raise an AssertionError with the given message.\n\n    >>> fail(\"my message\")\n    Traceback (most recent call last):\n        ...\n    AssertionError: my message\n\n    \"\"\"\n    raise AssertionError(msg or \"assertion failure\")\n\n\ndef assert_true(expr, msg_fmt=\"{msg}\"):\n    \"\"\"Fail the test unless the expression is truthy.\n\n    >>> assert_true(\"Hello World!\")\n    >>> assert_true(\"\")\n    Traceback (most recent call last):\n        ...\n    AssertionError: '' is not truthy\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * expr - tested expression\n    \"\"\"\n\n    if not expr:\n        msg = \"{!r} is not truthy\".format(expr)\n        fail(msg_fmt.format(msg=msg, expr=expr))\n\n\ndef assert_false(expr, msg_fmt=\"{msg}\"):\n    \"\"\"Fail the test unless the expression is falsy.\n\n    >>> assert_false(\"\")\n    >>> assert_false(\"Hello World!\")\n    Traceback (most recent call last):\n        ...\n    AssertionError: 'Hello World!' is not falsy\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * expr - tested expression\n    \"\"\"\n\n    if expr:\n        msg = \"{!r} is not falsy\".format(expr)\n        fail(msg_fmt.format(msg=msg, expr=expr))\n\n\ndef assert_boolean_true(expr, msg_fmt=\"{msg}\"):\n    \"\"\"Fail the test unless the expression is the constant True.\n\n    >>> assert_boolean_true(True)\n    >>> assert_boolean_true(\"Hello World!\")\n    Traceback (most recent call last):\n        ...\n    AssertionError: 'Hello World!' is not True\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * expr - tested expression\n    \"\"\"\n\n    if expr is not True:\n        msg = \"{!r} is not True\".format(expr)\n        fail(msg_fmt.format(msg=msg, expr=expr))\n\n\ndef assert_boolean_false(expr, msg_fmt=\"{msg}\"):\n    \"\"\"Fail the test unless the expression is the constant False.\n\n    >>> assert_boolean_false(False)\n    >>> assert_boolean_false(0)\n    Traceback (most recent call last):\n        ...\n    AssertionError: 0 is not False\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * expr - tested expression\n    \"\"\"\n\n    if expr is not False:\n        msg = \"{!r} is not False\".format(expr)\n        fail(msg_fmt.format(msg=msg, expr=expr))\n\n\ndef assert_is_none(expr, msg_fmt=\"{msg}\"):\n    \"\"\"Fail if the expression is not None.\n\n    >>> assert_is_none(None)\n    >>> assert_is_none(False)\n    Traceback (most recent call last):\n        ...\n    AssertionError: False is not None\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * expr - tested expression\n    \"\"\"\n\n    if expr is not None:\n        msg = \"{!r} is not None\".format(expr)\n        fail(msg_fmt.format(msg=msg, expr=expr))\n\n\ndef assert_is_not_none(expr, msg_fmt=\"{msg}\"):\n    \"\"\"Fail if the expression is None.\n\n    >>> assert_is_not_none(0)\n    >>> assert_is_not_none(None)\n    Traceback (most recent call last):\n        ...\n    AssertionError: expression is None\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * expr - tested expression\n    \"\"\"\n    if expr is None:\n        msg = \"expression is None\"\n        fail(msg_fmt.format(msg=msg, expr=expr))\n\n\ndef assert_equal(first, second, msg_fmt=\"{msg}\"):\n    \"\"\"Fail unless first equals second, as determined by the '==' operator.\n\n    >>> assert_equal(5, 5.0)\n    >>> assert_equal(\"Hello World!\", \"Goodbye!\")\n    Traceback (most recent call last):\n        ...\n    AssertionError: 'Hello World!' != 'Goodbye!'\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * first - the first argument\n    * second - the second argument\n    \"\"\"\n\n    if isinstance(first, dict) and isinstance(second, dict):\n        assert_dict_equal(first, second, msg_fmt)\n    elif not first == second:\n        msg = \"{!r} != {!r}\".format(first, second)\n        fail(msg_fmt.format(msg=msg, first=first, second=second))\n\n\ndef assert_not_equal(first, second, msg_fmt=\"{msg}\"):\n    \"\"\"Fail if first equals second, as determined by the '==' operator.\n\n    >>> assert_not_equal(5, 8)\n    >>> assert_not_equal(-7, -7.0)\n    Traceback (most recent call last):\n        ...\n    AssertionError: -7 == -7.0\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * first - the first argument\n    * second - the second argument\n    \"\"\"\n\n    if first == second:\n        msg = \"{!r} == {!r}\".format(first, second)\n        fail(msg_fmt.format(msg=msg, first=first, second=second))\n\n\ndef assert_almost_equal(\n    first, second, msg_fmt=\"{msg}\", places=None, delta=None\n):\n    \"\"\"Fail if first and second are not equal after rounding.\n\n    By default, the difference between first and second is rounded to\n    7 decimal places. This can be configured with the places argument.\n    Alternatively, delta can be used to specify the maximum allowed\n    difference between first and second.\n\n    If first and second can not be rounded or both places and delta are\n    supplied, a TypeError is raised.\n\n    >>> assert_almost_equal(5, 5.00000001)\n    >>> assert_almost_equal(5, 5.001)\n    Traceback (most recent call last):\n        ...\n    AssertionError: 5 != 5.001 within 7 places\n    >>> assert_almost_equal(5, 5.001, places=2)\n    >>> assert_almost_equal(5, 5.001, delta=0.1)\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * first - the first argument\n    * second - the second argument\n    * places - number of places to compare or None\n    * delta - delta or None\n    \"\"\"\n\n    if delta is not None and places is not None:\n        raise TypeError(\"'places' and 'delta' are mutually exclusive\")\n    if delta is not None:\n        if delta <= 0:\n            raise ValueError(\"delta must be larger than 0\")\n        diff = abs(second - first)\n        success = diff < delta\n        detail_msg = \"with delta={}\".format(delta)\n    else:\n        if places is None:\n            places = 7\n        success = not round(second - first, places)\n        detail_msg = \"within {} places\".format(places)\n    if not success:\n        msg = \"{!r} != {!r} {}\".format(first, second, detail_msg)\n        fail(\n            msg_fmt.format(\n                msg=msg, first=first, second=second, places=places, delta=delta\n            )\n        )\n\n\ndef assert_not_almost_equal(\n    first, second, msg_fmt=\"{msg}\", places=None, delta=None\n):\n    \"\"\"Fail if first and second are equal after rounding.\n\n    By default, the difference between first and second is rounded to\n    7 decimal places. This can be configured with the places argument.\n    Alternatively, delta can be used to specify the maximum allowed\n    difference between first and second.\n\n    If first and second can not be rounded or both places and delta are\n    supplied, a TypeError is raised.\n\n    >>> assert_not_almost_equal(5, 5.001)\n    >>> assert_not_almost_equal(5, 5.00000001)\n    Traceback (most recent call last):\n        ...\n    AssertionError: 5 == 5.00000001 within 7 places\n    >>> assert_not_almost_equal(5, 5.001, places=2)\n    Traceback (most recent call last):\n        ...\n    AssertionError: 5 == 5.001 within 2 places\n    >>> assert_not_almost_equal(5, 5.001, delta=0.1)\n    Traceback (most recent call last):\n        ...\n    AssertionError: 5 == 5.001 with delta=0.1\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * first - the first argument\n    * second - the second argument\n    * places - number of places to compare or None\n    * delta - delta or None\n    \"\"\"\n\n    if delta is not None and places is not None:\n        raise TypeError(\"'places' and 'delta' are mutually exclusive\")\n    if delta is not None:\n        if delta <= 0:\n            raise ValueError(\"delta must be larger than 0\")\n        diff = abs(second - first)\n        success = diff >= delta\n        detail_msg = \"with delta={}\".format(delta)\n    else:\n        if places is None:\n            places = 7\n        success = bool(round(second - first, places))\n        detail_msg = \"within {} places\".format(places)\n    if not success:\n        msg = \"{!r} == {!r} {}\".format(first, second, detail_msg)\n        fail(\n            msg_fmt.format(\n                msg=msg, first=first, second=second, places=places, delta=delta\n            )\n        )\n\n\ndef assert_dict_equal(\n    first, second, key_msg_fmt=\"{msg}\", value_msg_fmt=\"{msg}\"\n):\n    \"\"\"Fail unless first dictionary equals second.\n\n    The dictionaries are considered equal, if they both contain the same\n    keys, and their respective values are also equal.\n\n    >>> assert_dict_equal({\"foo\": 5}, {\"foo\": 5})\n    >>> assert_dict_equal({\"foo\": 5}, {})\n    Traceback (most recent call last):\n        ...\n    AssertionError: key 'foo' missing from right dict\n\n    The following key_msg_fmt arguments are supported, if the keys do not\n    match:\n    * msg - the default error message\n    * first - the first dict\n    * second - the second dict\n    * missing_keys - list of keys missing from right\n    * extra_keys - list of keys missing from left\n\n    The following value_msg_fmt arguments are supported, if a value does not\n    match:\n    * msg - the default error message\n    * first - the first dict\n    * second - the second dict\n    * key - the key where the value does not match\n    * first_value - the value in the first dict\n    * second_value - the value in the second dict\n    \"\"\"\n    first_keys = set(first.keys())\n    second_keys = set(second.keys())\n    missing_keys = list(first_keys - second_keys)\n    extra_keys = list(second_keys - first_keys)\n    if missing_keys or extra_keys:\n        if missing_keys:\n            if len(missing_keys) == 1:\n                msg = \"key {!r} missing from right dict\".format(\n                    missing_keys[0]\n                )\n            else:\n                keys = \", \".join(sorted(repr(k) for k in missing_keys))\n                msg = \"keys {} missing from right dict\".format(keys)\n        else:\n            if len(extra_keys) == 1:\n                msg = \"extra key {!r} in right dict\".format(extra_keys[0])\n            else:\n                keys = \", \".join(sorted(repr(k) for k in extra_keys))\n                msg = \"extra keys {} in right dict\".format(keys)\n        if key_msg_fmt:\n            msg = key_msg_fmt.format(\n                msg=msg,\n                first=first,\n                second=second,\n                missing_keys=missing_keys,\n                extra_keys=extra_keys,\n            )\n        raise AssertionError(msg)\n    for key in first:\n        first_value = first[key]\n        second_value = second[key]\n        msg = \"key '{}' differs: {!r} != {!r}\".format(\n            key, first_value, second_value\n        )\n        if value_msg_fmt:\n            msg = value_msg_fmt.format(\n                msg=msg,\n                first=first,\n                second=second,\n                key=key,\n                first_value=first_value,\n                second_value=second_value,\n            )\n        msg = msg.replace(\"{\", \"{{\").replace(\"}\", \"}}\")\n        assert_equal(first_value, second_value, msg_fmt=msg)\n\n\ndef assert_dict_superset(\n    first, second, key_msg_fmt=\"{msg}\", value_msg_fmt=\"{msg}\"\n):\n    \"\"\"Fail unless second dictionary is a superset of the first.\n\n    The second dictionary must contain all keys of the first and their\n    values are equal (or a superset in case of dicts). But the second\n    dictionary can contain additional keys.\n\n    >>> assert_dict_superset({\"foo\": 5}, {\"foo\": 5, \"bar\": 10})\n    >>> assert_dict_superset({\"foo\": 5}, {})\n    Traceback (most recent call last):\n        ...\n    AssertionError: key 'foo' missing from right dict\n\n    The following key_msg_fmt arguments are supported, if the keys do not\n    match:\n    * msg - the default error message\n    * first - the first dict\n    * second - the second dict\n    * missing_keys - list of keys missing from right\n\n    The following value_msg_fmt arguments are supported, if a value does not\n    match:\n    * msg - the default error message\n    * first - the first dict\n    * second - the second dict\n    * key - the key where the value does not match\n    * first_value - the value in the first dict\n    * second_value - the value in the second dict\n    \"\"\"\n    first_keys = set(first.keys())\n    second_keys = set(second.keys())\n    missing_keys = list(first_keys - second_keys)\n    if missing_keys:\n        if len(missing_keys) == 1:\n            msg = \"key {!r} missing from right dict\".format(missing_keys[0])\n        else:\n            keys = \", \".join(sorted(repr(k) for k in missing_keys))\n            msg = \"keys {} missing from right dict\".format(keys)\n        if key_msg_fmt:\n            msg = key_msg_fmt.format(\n                msg=msg, first=first, second=second, missing_keys=missing_keys\n            )\n        raise AssertionError(msg)\n    for key in first:\n        first_value = first[key]\n        second_value = second[key]\n        msg = \"key '{}' differs: {!r} != {!r}\".format(\n            key, first_value, second_value\n        )\n        if value_msg_fmt:\n            msg = value_msg_fmt.format(\n                msg=msg,\n                first=first,\n                second=second,\n                key=key,\n                first_value=first_value,\n                second_value=second_value,\n            )\n        msg = msg.replace(\"{\", \"{{\").replace(\"}\", \"}}\")\n        assert_equal(first_value, second_value, msg_fmt=msg)\n\n\ndef assert_less(first, second, msg_fmt=\"{msg}\"):\n    \"\"\"Fail if first is not less than second.\n\n    >>> assert_less('bar', 'foo')\n    >>> assert_less(5, 5)\n    Traceback (most recent call last):\n        ...\n    AssertionError: 5 is not less than 5\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * first - the first argument\n    * second - the second argument\n    \"\"\"\n\n    if not first < second:\n        msg = \"{!r} is not less than {!r}\".format(first, second)\n        fail(msg_fmt.format(msg=msg, first=first, second=second))\n\n\ndef assert_less_equal(first, second, msg_fmt=\"{msg}\"):\n    \"\"\"Fail if first is not less than or equal to second.\n\n    >>> assert_less_equal('bar', 'foo')\n    >>> assert_less_equal(5, 5)\n    >>> assert_less_equal(6, 5)\n    Traceback (most recent call last):\n        ...\n    AssertionError: 6 is not less than or equal to 5\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * first - the first argument\n    * second - the second argument\n    \"\"\"\n\n    if not first <= second:\n        msg = \"{!r} is not less than or equal to {!r}\".format(first, second)\n        fail(msg_fmt.format(msg=msg, first=first, second=second))\n\n\ndef assert_greater(first, second, msg_fmt=\"{msg}\"):\n    \"\"\"Fail if first is not greater than second.\n\n    >>> assert_greater('foo', 'bar')\n    >>> assert_greater(5, 5)\n    Traceback (most recent call last):\n        ...\n    AssertionError: 5 is not greater than 5\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * first - the first argument\n    * second - the second argument\n    \"\"\"\n\n    if not first > second:\n        msg = \"{!r} is not greater than {!r}\".format(first, second)\n        fail(msg_fmt.format(msg=msg, first=first, second=second))\n\n\ndef assert_greater_equal(first, second, msg_fmt=\"{msg}\"):\n    \"\"\"Fail if first is not greater than or equal to second.\n\n    >>> assert_greater_equal('foo', 'bar')\n    >>> assert_greater_equal(5, 5)\n    >>> assert_greater_equal(5, 6)\n    Traceback (most recent call last):\n        ...\n    AssertionError: 5 is not greater than or equal to 6\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * first - the first argument\n    * second - the second argument\n    \"\"\"\n\n    if not first >= second:\n        msg = \"{!r} is not greater than or equal to {!r}\".format(first, second)\n        fail(msg_fmt.format(msg=msg, first=first, second=second))\n\n\ndef assert_regex(text, regex, msg_fmt=\"{msg}\"):\n    \"\"\"Fail if text does not match the regular expression.\n\n    regex can be either a regular expression string or a compiled regular\n    expression object.\n\n    >>> assert_regex(\"Hello World!\", r\"llo.*rld!$\")\n    >>> assert_regex(\"Hello World!\", r\"\\\\d\")\n    Traceback (most recent call last):\n        ...\n    AssertionError: 'Hello World!' does not match '\\\\\\\\d'\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * text - text that is matched\n    * pattern - regular expression pattern as string\n    \"\"\"\n\n    compiled = re.compile(regex)\n    if not compiled.search(text):\n        msg = \"{!r} does not match {!r}\".format(text, compiled.pattern)\n        fail(msg_fmt.format(msg=msg, text=text, pattern=compiled.pattern))\n\n\ndef assert_not_regex(text, regex, msg_fmt=\"{msg}\"):\n    \"\"\"Fail if text does match the regular expression.\n\n    regex can be either a regular expression string or a compiled regular\n    expression object.\n\n    >>> assert_regex(\"Hello World!\", r\"llo.*rld!$\")\n    >>> assert_regex(\"Hello World!\", r\"\\\\d\")\n    Traceback (most recent call last):\n        ...\n    AssertionError: 'Hello World!' does not match '\\\\\\\\d'\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * text - text that is matched\n    * pattern - regular expression pattern as string\n    \"\"\"\n\n    compiled = re.compile(regex)\n    if compiled.search(text):\n        msg = \"{!r} matches {!r}\".format(text, compiled.pattern)\n        fail(msg_fmt.format(msg=msg, text=text, pattern=compiled.pattern))\n\n\ndef assert_is(first, second, msg_fmt=\"{msg}\"):\n    \"\"\"Fail if first and second do not refer to the same object.\n\n    >>> list1 = [5, \"foo\"]\n    >>> list2 = [5, \"foo\"]\n    >>> assert_is(list1, list1)\n    >>> assert_is(list1, list2)\n    Traceback (most recent call last):\n        ...\n    AssertionError: [5, 'foo'] is not [5, 'foo']\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * first - the first argument\n    * second - the second argument\n    \"\"\"\n\n    if first is not second:\n        msg = \"{!r} is not {!r}\".format(first, second)\n        fail(msg_fmt.format(msg=msg, first=first, second=second))\n\n\ndef assert_is_not(first, second, msg_fmt=\"{msg}\"):\n    \"\"\"Fail if first and second refer to the same object.\n\n    >>> list1 = [5, \"foo\"]\n    >>> list2 = [5, \"foo\"]\n    >>> assert_is_not(list1, list2)\n    >>> assert_is_not(list1, list1)\n    Traceback (most recent call last):\n        ...\n    AssertionError: both arguments refer to [5, 'foo']\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * first - the first argument\n    * second - the second argument\n    \"\"\"\n\n    if first is second:\n        msg = \"both arguments refer to {!r}\".format(first)\n        fail(msg_fmt.format(msg=msg, first=first, second=second))\n\n\ndef assert_in(first, second, msg_fmt=\"{msg}\"):\n    \"\"\"Fail if first is not in collection second.\n\n    >>> assert_in(\"foo\", [4, \"foo\", {}])\n    >>> assert_in(\"bar\", [4, \"foo\", {}])\n    Traceback (most recent call last):\n        ...\n    AssertionError: 'bar' not in [4, 'foo', {}]\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * first - the element looked for\n    * second - the container looked in\n    \"\"\"\n\n    if first not in second:\n        msg = \"{!r} not in {!r}\".format(first, second)\n        fail(msg_fmt.format(msg=msg, first=first, second=second))\n\n\ndef assert_not_in(first, second, msg_fmt=\"{msg}\"):\n    \"\"\"Fail if first is in a collection second.\n\n    >>> assert_not_in(\"bar\", [4, \"foo\", {}])\n    >>> assert_not_in(\"foo\", [4, \"foo\", {}])\n    Traceback (most recent call last):\n        ...\n    AssertionError: 'foo' is in [4, 'foo', {}]\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * first - the element looked for\n    * second - the container looked in\n    \"\"\"\n    if first in second:\n        msg = \"{!r} is in {!r}\".format(first, second)\n        fail(msg_fmt.format(msg=msg, first=first, second=second))\n\n\ndef assert_count_equal(sequence1, sequence2, msg_fmt=\"{msg}\"):\n    \"\"\"Compare the items of two sequences, ignoring order.\n\n    >>> assert_count_equal([1, 2], {2, 1})\n\n    Items missing in either sequence will be listed:\n\n    >>> assert_count_equal([\"a\", \"b\", \"c\"], [\"a\", \"d\"])\n    Traceback (most recent call last):\n        ...\n    AssertionError: missing from sequence 1: 'd'; missing from sequence 2: 'b', 'c'\n\n    Items are counted in each sequence. This makes it useful to detect\n    duplicates:\n\n    >>> assert_count_equal({\"a\", \"b\"}, [\"a\", \"a\", \"b\"])\n    Traceback (most recent call last):\n        ...\n    AssertionError: missing from sequence 1: 'a'\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * first - first sequence\n    * second - second sequence\n    \"\"\"\n\n    def compare():\n        missing1 = list(sequence2)\n        missing2 = []\n        for item in sequence1:\n            try:\n                missing1.remove(item)\n            except ValueError:\n                missing2.append(item)\n        return missing1, missing2\n\n    def build_message():\n        msg = \"\"\n        if missing_from_1:\n            msg += \"missing from sequence 1: \" + \", \".join(\n                repr(i) for i in missing_from_1\n            )\n        if missing_from_1 and missing_from_2:\n            msg += \"; \"\n        if missing_from_2:\n            msg += \"missing from sequence 2: \" + \", \".join(\n                repr(i) for i in missing_from_2\n            )\n        return msg\n\n    missing_from_1, missing_from_2 = compare()\n    if missing_from_1 or missing_from_2:\n        fail(\n            msg_fmt.format(\n                msg=build_message(), first=sequence1, second=sequence2\n            )\n        )\n\n\ndef assert_between(lower_bound, upper_bound, expr, msg_fmt=\"{msg}\"):\n    \"\"\"Fail if an expression is not between certain bounds (inclusive).\n\n    >>> assert_between(5, 15, 5)\n    >>> assert_between(5, 15, 15)\n    >>> assert_between(5, 15, 4.9)\n    Traceback (most recent call last):\n        ...\n    AssertionError: 4.9 is not between 5 and 15\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * lower - lower bound\n    * upper - upper bound\n    * expr - tested expression\n    \"\"\"\n\n    if not lower_bound <= expr <= upper_bound:\n        msg = \"{!r} is not between {} and {}\".format(\n            expr, lower_bound, upper_bound\n        )\n        fail(\n            msg_fmt.format(\n                msg=msg, lower=lower_bound, upper=upper_bound, expr=expr\n            )\n        )\n\n\ndef assert_is_instance(obj, cls, msg_fmt=\"{msg}\"):\n    \"\"\"Fail if an object is not an instance of a class or tuple of classes.\n\n    >>> assert_is_instance(5, int)\n    >>> assert_is_instance('foo', (str, bytes))\n    >>> assert_is_instance(5, str)\n    Traceback (most recent call last):\n        ...\n    AssertionError: 5 is an instance of <class 'int'>, expected <class 'str'>\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * obj - object to test\n    * types - tuple of types tested against\n    \"\"\"\n    if not isinstance(obj, cls):\n        msg = \"{!r} is an instance of {!r}, expected {!r}\".format(\n            obj, obj.__class__, cls\n        )\n        types = cls if isinstance(cls, tuple) else (cls,)\n        fail(msg_fmt.format(msg=msg, obj=obj, types=types))\n\n\ndef assert_not_is_instance(obj, cls, msg_fmt=\"{msg}\"):\n    \"\"\"Fail if an object is an instance of a class or tuple of classes.\n\n    >>> assert_not_is_instance(5, str)\n    >>> assert_not_is_instance(5, (str, bytes))\n    >>> assert_not_is_instance('foo', str)\n    Traceback (most recent call last):\n        ...\n    AssertionError: 'foo' is an instance of <class 'str'>\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * obj - object to test\n    * types - tuple of types tested against\n    \"\"\"\n    if isinstance(obj, cls):\n        msg = \"{!r} is an instance of {!r}\".format(obj, obj.__class__)\n        types = cls if isinstance(cls, tuple) else (cls,)\n        fail(msg_fmt.format(msg=msg, obj=obj, types=types))\n\n\ndef assert_has_attr(obj, attribute, msg_fmt=\"{msg}\"):\n    \"\"\"Fail is an object does not have an attribute.\n\n    >>> assert_has_attr([], \"index\")\n    >>> assert_has_attr([], \"i_do_not_have_this\")\n    Traceback (most recent call last):\n        ...\n    AssertionError: [] does not have attribute 'i_do_not_have_this'\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * obj - object to test\n    * attribute - name of the attribute to check\n    \"\"\"\n\n    if not hasattr(obj, attribute):\n        msg = \"{!r} does not have attribute '{}'\".format(obj, attribute)\n        fail(msg_fmt.format(msg=msg, obj=obj, attribute=attribute))\n\n\n_EPSILON_SECONDS = 5\n\n\ndef assert_datetime_about_now(actual, msg_fmt=\"{msg}\"):\n    \"\"\"Fail if a datetime object is not within 5 seconds of the local time.\n\n    >>> assert_datetime_about_now(datetime.now())\n    >>> assert_datetime_about_now(datetime(1900, 1, 1, 12, 0, 0))\n    Traceback (most recent call last):\n        ...\n    AssertionError: datetime.datetime(1900, 1, 1, 12, 0) is not close to current date/time\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * actual - datetime object to check\n    * now - current datetime that was tested against\n    \"\"\"\n\n    now = datetime.now()\n    if actual is None:\n        msg = \"None is not a valid date/time\"\n        fail(msg_fmt.format(msg=msg, actual=actual, now=now))\n    lower_bound = now - timedelta(seconds=_EPSILON_SECONDS)\n    upper_bound = now + timedelta(seconds=_EPSILON_SECONDS)\n    if not lower_bound <= actual <= upper_bound:\n        msg = \"{!r} is not close to current date/time\".format(actual)\n        fail(msg_fmt.format(msg=msg, actual=actual, now=now))\n\n\ndef assert_datetime_about_now_utc(actual, msg_fmt=\"{msg}\"):\n    \"\"\"Fail if a datetime object is not within 5 seconds of UTC.\n\n    >>> assert_datetime_about_now_utc(datetime.now(timezone.utc).replace(tzinfo=None))\n    >>> assert_datetime_about_now_utc(datetime(1900, 1, 1, 12, 0, 0))\n    Traceback (most recent call last):\n        ...\n    AssertionError: datetime.datetime(1900, 1, 1, 12, 0) is not close to current UTC date/time\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * actual - datetime object to check\n    * now - current datetime that was tested against\n    \"\"\"\n\n    now = datetime.now(timezone.utc).replace(tzinfo=None)\n    if actual is None:\n        msg = \"None is not a valid date/time\"\n        fail(msg_fmt.format(msg=msg, actual=actual, now=now))\n    lower_bound = now - timedelta(seconds=_EPSILON_SECONDS)\n    upper_bound = now + timedelta(seconds=_EPSILON_SECONDS)\n    if not lower_bound <= actual <= upper_bound:\n        msg = \"{!r} is not close to current UTC date/time\".format(actual)\n        fail(msg_fmt.format(msg=msg, actual=actual, now=now))\n\n\nclass AssertRaisesContext:\n    \"\"\"A context manager to test for exceptions with certain properties.\n\n    When the context is left and no exception has been raised, an\n    AssertionError will be raised:\n\n        >>> context = AssertRaisesContext(TypeError)\n        >>> with context:\n        ...    pass\n        Traceback (most recent call last):\n            ...\n        AssertionError: TypeError not raised\n\n    If an exception that is not a sub-class of the exception class provided\n    to the constructor is raised, it will be passed on:\n\n        >>> with context:\n        ...    raise ValueError(\"Wrong Class\")\n        Traceback (most recent call last):\n            ...\n        ValueError: Wrong Class\n\n    If the exception has the right class, any additional tests that have been\n    configured on the context, will be called:\n\n        >>> def test(exc):\n        ...     assert_equal(\"Hello World!\", str(exc))\n        >>> context.add_test(test)\n        >>> with context:\n        ...     raise TypeError(\"Wrong Message\")\n        Traceback (most recent call last):\n            ...\n        AssertionError: 'Hello World!' != 'Wrong Message'\n\n    \"\"\"\n\n    def __init__(self, exception, msg_fmt=\"{msg}\"):\n        self.exception = exception\n        self.msg_fmt = msg_fmt\n        self._exc_type = exception\n        self._exc_val = None\n        self._exception_name = getattr(exception, \"__name__\", str(exception))\n        self._tests: list[Callable[[Any], object]] = []\n\n    def __enter__(self):\n        return self\n\n    def __exit__(self, exc_type, exc_val, exc_tb):\n        if not exc_type or not exc_val:\n            msg = \"{} not raised\".format(self._exception_name)\n            fail(self.format_message(msg))\n        self._exc_val = exc_val\n        if not issubclass(exc_type, self.exception):\n            return False\n        for test in self._tests:\n            test(exc_val)\n        return True\n\n    def format_message(self, default_msg):\n        return self.msg_fmt.format(\n            msg=default_msg,\n            exc_type=self._exc_type,\n            exc_name=self._exception_name,\n        )\n\n    def add_test(self, cb: Callable[[Any], object]) -> None:\n        \"\"\"Add a test callback.\n\n        This callback is called after determining that the right exception\n        class was raised. The callback will get the raised exception as only\n        argument.\n\n        \"\"\"\n        self._tests.append(cb)\n\n    @property\n    def exc_val(self):\n        if self._exc_val is None:\n            raise RuntimeError(\"must be called after leaving the context\")\n        return self._exc_val\n\n\nclass AssertRaisesRegexContext(AssertRaisesContext):\n    \"\"\"A context manager to test for exceptions and their messages.\"\"\"\n\n    def __init__(self, exception, pattern, msg_fmt=\"{msg}\"):\n        super(AssertRaisesRegexContext, self).__init__(exception, msg_fmt)\n        self.pattern = pattern\n\n    def format_message(self, default_msg):\n        return self.msg_fmt.format(\n            msg=default_msg,\n            exc_type=self._exc_type,\n            exc_name=self._exception_name,\n            pattern=self.pattern,\n            text=\"\",\n        )\n\n\nclass AssertRaisesErrnoContext(AssertRaisesContext):\n    \"\"\"A context manager to test for exceptions with errnos.\"\"\"\n\n    def __init__(self, exception, expected_errno, msg_fmt=\"{msg}\"):\n        super(AssertRaisesErrnoContext, self).__init__(exception, msg_fmt)\n        self.expected_errno = expected_errno\n\n    def format_message(self, default_msg):\n        return self.msg_fmt.format(\n            msg=default_msg,\n            exc_type=self._exc_type,\n            exc_name=self._exception_name,\n            expected_errno=self.expected_errno,\n            actual_errno=None,\n        )\n\n\ndef assert_raises(exception, msg_fmt=\"{msg}\"):\n    \"\"\"Fail unless a specific exception is raised inside the context.\n\n    If a different type of exception is raised, it will not be caught.\n\n    >>> with assert_raises(TypeError):\n    ...     raise TypeError()\n    ...\n    >>> with assert_raises(TypeError):\n    ...     pass\n    ...\n    Traceback (most recent call last):\n        ...\n    AssertionError: TypeError not raised\n    >>> with assert_raises(TypeError):\n    ...     raise ValueError(\"wrong error\")\n    ...\n    Traceback (most recent call last):\n        ...\n    ValueError: wrong error\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * exc_type - exception type that is expected\n    * exc_name - expected exception type name\n    \"\"\"\n\n    return AssertRaisesContext(exception, msg_fmt)\n\n\ndef assert_raises_regex(exception, regex, msg_fmt=\"{msg}\"):\n    \"\"\"Fail unless an exception with a message that matches a regular\n     expression is raised within the context.\n\n    The regular expression can be a regular expression string or object.\n\n    >>> with assert_raises_regex(ValueError, r\"\\\\d+\"):\n    ...     raise ValueError(\"Error #42\")\n    ...\n    >>> with assert_raises_regex(ValueError, r\"\\\\d+\"):\n    ...     raise ValueError(\"Generic Error\")\n    ...\n    Traceback (most recent call last):\n        ...\n    AssertionError: 'Generic Error' does not match '\\\\\\\\d+'\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * exc_type - exception type that is expected\n    * exc_name - expected exception type name\n    * text - actual error text\n    * pattern - expected error message as regular expression string\n    \"\"\"\n\n    def test(exc):\n        compiled = re.compile(regex)\n        if not exc.args:\n            msg = \"{} without message\".format(exception.__name__)\n            fail(\n                msg_fmt.format(\n                    msg=msg,\n                    text=None,\n                    pattern=compiled.pattern,\n                    exc_type=exception,\n                    exc_name=exception.__name__,\n                )\n            )\n        text = exc.args[0]\n        if not compiled.search(text):\n            msg = \"{!r} does not match {!r}\".format(text, compiled.pattern)\n            fail(\n                msg_fmt.format(\n                    msg=msg,\n                    text=text,\n                    pattern=compiled.pattern,\n                    exc_type=exception,\n                    exc_name=exception.__name__,\n                )\n            )\n\n    context = AssertRaisesRegexContext(exception, regex, msg_fmt)\n    context.add_test(test)\n    return context\n\n\ndef assert_raises_errno(exception, errno, msg_fmt=\"{msg}\"):\n    \"\"\"Fail unless an exception with a specific errno is raised with the\n     context.\n\n    >>> with assert_raises_errno(OSError, 42):\n    ...     raise OSError(42, \"OS Error\")\n    ...\n    >>> with assert_raises_errno(OSError, 44):\n    ...     raise OSError(17, \"OS Error\")\n    ...\n    Traceback (most recent call last):\n        ...\n    AssertionError: wrong errno: 44 != 17\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * exc_type - exception type that is expected\n    * exc_name - expected exception type name\n    * expected_errno -\n    * actual_errno - raised errno or None if no matching exception was raised\n    \"\"\"\n\n    def check_errno(exc):\n        if errno != exc.errno:\n            msg = \"wrong errno: {!r} != {!r}\".format(errno, exc.errno)\n            fail(\n                msg_fmt.format(\n                    msg=msg,\n                    exc_type=exception,\n                    exc_name=exception.__name__,\n                    expected_errno=errno,\n                    actual_errno=exc.errno,\n                )\n            )\n\n    context = AssertRaisesErrnoContext(exception, errno, msg_fmt)\n    context.add_test(check_errno)\n    return context\n\n\ndef assert_succeeds(exception, msg_fmt=\"{msg}\"):\n    \"\"\"Fail if a specific exception is raised within the context.\n\n    This assertion should be used for cases, where successfully running a\n    function signals a successful test, and raising the exception of a\n    certain type signals a test failure. All other raised exceptions are\n    passed on and will usually still result in a test error. This can be\n    used to signal the intent of a block.\n\n    >>> l = [\"foo\", \"bar\"]\n    >>> with assert_succeeds(ValueError):\n    ...     i = l.index(\"foo\")\n    ...\n    >>> with assert_succeeds(ValueError):\n    ...     raise ValueError()\n    ...\n    Traceback (most recent call last):\n        ...\n    AssertionError: ValueError was unexpectedly raised\n    >>> with assert_succeeds(ValueError):\n    ...     raise TypeError(\"Wrong Error\")\n    ...\n    Traceback (most recent call last):\n        ...\n    TypeError: Wrong Error\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * exc_type - exception type\n    * exc_name - exception type name\n    * exception - exception that was raised\n    \"\"\"\n\n    class _AssertSucceeds(object):\n        def __enter__(self):\n            pass\n\n        def __exit__(self, exc_type, exc_val, exc_tb):\n            if exc_type and issubclass(exc_type, exception):\n                msg = exception.__name__ + \" was unexpectedly raised\"\n                fail(\n                    msg_fmt.format(\n                        msg=msg,\n                        exc_type=exception,\n                        exc_name=exception.__name__,\n                        exception=exc_val,\n                    )\n                )\n\n    return _AssertSucceeds()\n\n\nclass AssertWarnsContext(object):\n    \"\"\"A context manager to test for warnings with certain properties.\n\n    When the context is left and the expected warning has not been raised, an\n    AssertionError will be raised:\n\n        >>> context = AssertWarnsContext(DeprecationWarning)\n        >>> with context:\n        ...    pass\n        Traceback (most recent call last):\n            ...\n        AssertionError: DeprecationWarning not issued\n\n    If the warning has the right class, any additional tests that have been\n    configured on the context, will be called:\n\n        >>> from warnings import warn\n        >>> def test(warning):\n        ...     return False\n        >>> context.add_test(test)\n        >>> with context:\n        ...     warn(\"Wrong Message\", DeprecationWarning)\n        Traceback (most recent call last):\n            ...\n        AssertionError: DeprecationWarning not issued\n\n    \"\"\"\n\n    def __init__(self, warning_class, msg_fmt=\"{msg}\"):\n        self._warning_class = warning_class\n        self._msg_fmt = msg_fmt\n        self._warning_context: catch_warnings[list[WarningMessage]] | None = (\n            None\n        )\n        self._warnings = []\n        self._tests: list[Callable[[Warning], bool]] = []\n\n    def __enter__(self):\n        self._warning_context = catch_warnings(record=True)\n        self._warnings = self._warning_context.__enter__()\n        return self\n\n    def __exit__(self, exc_type, exc_val, exc_tb):\n        assert self._warning_context is not None\n        self._warning_context.__exit__(exc_type, exc_val, exc_tb)\n        if not any(self._is_expected_warning(w) for w in self._warnings):\n            fail(self.format_message())\n\n    def format_message(self):\n        msg = \"{} not issued\".format(self._warning_class.__name__)\n        return self._msg_fmt.format(\n            msg=msg,\n            exc_type=self._warning_class,\n            exc_name=self._warning_class.__name__,\n        )\n\n    def _is_expected_warning(self, warning) -> bool:\n        if not issubclass(warning.category, self._warning_class):\n            return False\n        return all(test(warning) for test in self._tests)\n\n    def add_test(self, cb: Callable[[Warning], bool]) -> None:\n        \"\"\"Add a test callback.\n\n        This callback is called after determining that the right warning\n        class was issued. The callback will get the issued warning as only\n        argument and must return a boolean value.\n\n        \"\"\"\n        self._tests.append(cb)\n\n\nclass AssertWarnsRegexContext(AssertWarnsContext):\n    \"\"\"A context manager to test for warnings and their messages.\"\"\"\n\n    def __init__(self, warning_class, pattern, msg_fmt=\"{msg}\"):\n        super(AssertWarnsRegexContext, self).__init__(warning_class, msg_fmt)\n        self.pattern = pattern\n\n    def format_message(self):\n        msg = \"no {} matching {} issued\".format(\n            self._warning_class.__name__, repr(self.pattern)\n        )\n        return self._msg_fmt.format(\n            msg=msg,\n            exc_type=self._warning_class,\n            exc_name=self._warning_class.__name__,\n            pattern=self.pattern,\n        )\n\n\ndef assert_warns(warning_type, msg_fmt=\"{msg}\"):\n    \"\"\"Fail unless a specific warning is issued inside the context.\n\n    If a different type of warning is issued, it will not be caught.\n\n    >>> from warnings import warn\n    >>> with assert_warns(UserWarning):\n    ...     warn(\"warning message\", UserWarning)\n    ...\n    >>> with assert_warns(UserWarning):\n    ...     pass\n    ...\n    Traceback (most recent call last):\n        ...\n    AssertionError: UserWarning not issued\n    >>> with assert_warns(UserWarning):\n    ...     warn(\"warning message\", UnicodeWarning)\n    ...\n    Traceback (most recent call last):\n        ...\n    AssertionError: UserWarning not issued\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * exc_type - exception type\n    * exc_name - exception type name\n    \"\"\"\n    return AssertWarnsContext(warning_type, msg_fmt)\n\n\ndef assert_warns_regex(warning_type, regex, msg_fmt=\"{msg}\"):\n    \"\"\"Fail unless a warning with a message is issued inside the context.\n\n    The message can be a regular expression string or object.\n\n    >>> from warnings import warn\n    >>> with assert_warns_regex(UserWarning, r\"#\\\\d+\"):\n    ...     warn(\"Error #42\", UserWarning)\n    ...\n    >>> with assert_warns_regex(UserWarning, r\"Expected Error\"):\n    ...     warn(\"Generic Error\", UserWarning)\n    ...\n    Traceback (most recent call last):\n        ...\n    AssertionError: no UserWarning matching 'Expected Error' issued\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * exc_type - warning type\n    * exc_name - warning type name\n    * pattern - expected warning message as regular expression string\n    \"\"\"\n\n    def test(warning):\n        return re.search(regex, str(warning.message)) is not None\n\n    context = AssertWarnsRegexContext(warning_type, regex, msg_fmt)\n    context.add_test(test)\n    return context\n\n\nif sys.version_info >= (3,):\n    _Str = str\nelse:\n    _Str = unicode  # noqa: F821\n\n\ndef assert_json_subset(first, second):\n    \"\"\"Assert that a JSON object or array is a subset of another JSON object\n    or array.\n\n    The first JSON object or array must be supplied as a JSON-compatible\n    dict or list, the JSON object or array to check must be a string, an\n    UTF-8 bytes object, or a JSON-compatible list or dict.\n\n    A JSON non-object, non-array value is the subset of another JSON value,\n    if they are equal.\n\n    A JSON object is the subset of another JSON object if for each name/value\n    pair in the former there is a name/value pair in the latter with the same\n    name. Additionally, the value of the former pair must be a subset of the\n    value of the latter pair.\n\n    A JSON array is the subset of another JSON array, if they have the same\n    number of elements and each element in the former is a subset of the\n    corresponding element in the latter.\n\n    >>> assert_json_subset({}, '{}')\n    >>> assert_json_subset({}, '{\"foo\": \"bar\"}')\n    >>> assert_json_subset({\"foo\": \"bar\"}, '{}')\n    Traceback (most recent call last):\n    ...\n    AssertionError: element 'foo' missing from element $\n    >>> assert_json_subset([1, 2], '[1, 2]')\n    >>> assert_json_subset([2, 1], '[1, 2]')\n    Traceback (most recent call last):\n    ...\n    AssertionError: element $[0] differs: 2 != 1\n    >>> assert_json_subset([{}], '[{\"foo\": \"bar\"}]')\n    >>> assert_json_subset({}, \"INVALID JSON\")\n    Traceback (most recent call last):\n    ...\n    json.decoder.JSONDecodeError: Expecting value: line 1 column 1 (char 0)\n\n    In objects, the special classes `Present` and `Absent` can be used to\n    check for the presence or absence of a specific key:\n\n    >>> assert_json_subset({Exists(\"foo\"): True}, '{\"foo\": \"bar\"}')\n    >>> assert_json_subset({Exists(\"foo\"): True}, '{}')\n    Traceback (most recent call last):\n    ...\n    AssertionError: element 'foo' missing from element $\n    >>> assert_json_subset({Exists(\"foo\"): False}, '{}')\n    >>> assert_json_subset({Exists(\"foo\"): False}, '{\"foo\": \"bar\"}')\n    Traceback (most recent call last):\n    ...\n    AssertionError: spurious member 'foo' in object $\n    \"\"\"\n\n    if not isinstance(second, (dict, list, str, bytes)):\n        raise TypeError(\"second must be dict, list, str, or bytes\")\n    if isinstance(second, bytes):\n        second = second.decode(\"utf-8\")\n    if isinstance(second, _Str):\n        parsed_second = json_loads(second)\n    else:\n        parsed_second = second\n\n    if not isinstance(parsed_second, (dict, list)):\n        raise AssertionError(\n            \"second must decode to dict or list, not {}\".format(\n                type(parsed_second)\n            )\n        )\n\n    comparer = _JSONComparer(_JSONPath(\"$\"), first, parsed_second)\n    comparer.assert_()\n\n\nclass _JSONComparer:\n    def __init__(self, path, expected, actual):\n        self._path = path\n        self._expected = expected\n        self._actual = actual\n\n    def assert_(self):\n        self._assert_types_are_equal()\n        if isinstance(self._expected, dict):\n            self._assert_dicts_equal()\n        elif isinstance(self._expected, list):\n            self._assert_arrays_equal()\n        elif _is_present(self._expected):\n            pass\n        else:\n            self._assert_fundamental_values_equal()\n\n    def _assert_types_are_equal(self):\n        if self._types_differ():\n            self._raise_different_values()\n\n    def _types_differ(self):\n        if self._expected is None:\n            return self._actual is not None\n        elif isinstance(self._expected, (int, float)):\n            return not isinstance(self._actual, (int, float))\n        elif _is_present(self._expected):\n            return False\n        for type_ in [bool, str, _Str, list, dict]:\n            if isinstance(self._expected, type_):\n                return not isinstance(self._actual, type_)\n        else:\n            raise TypeError(\"unsupported type {}\".format(type(self._expected)))\n\n    def _assert_dicts_equal(self) -> None:\n        for name in self._expected:\n            if not isinstance(name, (str, Exists)):\n                raise TypeError(\n                    f\"{repr(name)} is not a valid object member name\",\n                )\n        self._assert_all_expected_keys_in_actual_dict()\n        self._assert_no_wrong_keys()\n        self._assert_dict_values_equal()\n\n    def _assert_all_expected_keys_in_actual_dict(self) -> None:\n        keys = self._expected_key_names.difference(self._actual.keys())\n        if keys:\n            self._raise_missing_element(keys)\n\n    def _assert_no_wrong_keys(self) -> None:\n        for name in self._expected:\n            if isinstance(name, str) and _is_absent(self._expected[name]):\n                if name in self._actual:\n                    self._raise_assertion_error(\n                        f\"spurious member '{name}' in object {{path}}\"\n                    )\n            if isinstance(name, Exists) and not self._expected[name]:\n                if name.member_name in self._actual:\n                    self._raise_assertion_error(\n                        f\"spurious member '{name.member_name}' in object {{path}}\"\n                    )\n\n    def _assert_dict_values_equal(self) -> None:\n        for name in self._expected:\n            if isinstance(name, str) and not _is_absent(self._expected[name]):\n                self._assert_json_value_equals_with_item(name)\n\n    @property\n    def _expected_key_names(self) -> Set[str]:\n        keys: Set[str] = set()\n        for k in self._expected.keys():\n            if isinstance(k, str):\n                if not _is_absent(self._expected[k]):\n                    keys.add(k)\n            elif isinstance(k, Exists) and self._expected[k]:\n                keys.add(k.member_name)\n        return keys\n\n    def _assert_arrays_equal(self):\n        if len(self._expected) != len(self._actual):\n            self._raise_different_sizes()\n        for i in range(len(self._expected)):\n            self._assert_json_value_equals_with_item(i)\n\n    def _assert_json_value_equals_with_item(self, item):\n        path = self._path.append(item)\n        expected = self._expected[item]\n        actual = self._actual[item]\n        _JSONComparer(path, expected, actual).assert_()\n\n    def _assert_fundamental_values_equal(self):\n        if self._expected != self._actual:\n            self._raise_different_values()\n\n    def _raise_different_values(self):\n        self._raise_assertion_error(\n            \"element {path} differs: {expected} != {actual}\"\n        )\n\n    def _raise_different_sizes(self):\n        self._raise_assertion_error(\n            \"JSON array {path} differs in size: \"\n            \"{expected_len} != {actual_len}\",\n            expected_len=len(self._expected),\n            actual_len=len(self._actual),\n        )\n\n    def _raise_missing_element(self, keys):\n        if len(keys) == 1:\n            format_string = \"element {elements} missing from element {path}\"\n            elements = repr(next(iter(keys)))\n        else:\n            format_string = \"elements {elements} missing from element {path}\"\n            sorted_keys = sorted(keys)\n            elements = (\n                \", \".join(repr(k) for k in sorted_keys[:-1])\n                + \", and \"\n                + repr(sorted_keys[-1])\n            )\n        self._raise_assertion_error(format_string, elements=elements)\n\n    def _raise_assertion_error(self, format_, **kwargs):\n        kwargs.update(\n            {\n                \"path\": self._path,\n                \"expected\": repr(self._expected),\n                \"actual\": repr(self._actual),\n            }\n        )\n        raise AssertionError(format_.format(**kwargs))\n\n\nclass _JSONPath:\n    def __init__(self, path):\n        self._path = path\n\n    def __str__(self):\n        return self._path\n\n    def append(self, item):\n        return _JSONPath(\"{0}[{1}]\".format(self._path, repr(item)))\n\n\nclass Present:\n    \"\"\"Helper class for presence checks in assert_json_subset().\"\"\"\n\n\ndef _is_present(o: object) -> bool:\n    return o is Present or isinstance(o, Present)\n\n\nclass Absent:\n    \"\"\"Helper class for absence checks in assert_json_subset().\"\"\"\n\n\ndef _is_absent(o: object) -> bool:\n    return o is Absent or isinstance(o, Absent)\n\n\n@deprecated(\"Use Present and Absent instead.\")\nclass Exists:\n    \"\"\"Helper class for existence checks in assert_json_subset().\"\"\"\n\n    def __init__(self, member_name: str) -> None:\n        self.member_name = member_name\n"
    },
    {
      "path": "srittau_python-asserts/asserts/__init__.pyi",
      "content": "import datetime\nfrom collections.abc import Callable, Container, Iterable\nfrom contextlib import AbstractContextManager as ContextManager\nfrom re import Pattern\nfrom types import TracebackType\nfrom typing import Any, Generic, NoReturn, TypeVar\n\nfrom typing_extensions import deprecated\n\n_E = TypeVar(\"_E\", bound=BaseException)\n_S = TypeVar(\"_S\")\n\nclass AssertRaisesContext(Generic[_E]):\n    exception: type[_E]\n    msg_fmt: str\n    def __init__(self, exception: type[_E], msg_fmt: str = ...) -> None: ...\n    def __enter__(self: _S) -> _S: ...\n    def __exit__(\n        self,\n        exc_type: type[BaseException] | None,\n        exc_val: BaseException | None,\n        exc_tb: TracebackType | None,\n    ) -> bool: ...\n    def format_message(self, default_msg: str) -> str: ...\n    def add_test(self, cb: Callable[[_E], object]) -> None: ...\n    @property\n    def exc_val(self) -> _E: ...\n\nclass AssertRaisesErrnoContext(AssertRaisesContext[_E]):\n    expected_errno: int\n    def __init__(\n        self, exception: type[_E], expected_errno: int, msg_fmt: str = ...\n    ) -> None: ...\n\nclass AssertRaisesRegexContext(AssertRaisesContext[_E]):\n    pattern: str\n    def __init__(\n        self, exception: type[_E], pattern: str, msg_fmt: str = ...\n    ) -> None: ...\n\nclass AssertWarnsContext:\n    def __init__(\n        self, warning_class: type[Warning], msg_fmt: str = ...\n    ) -> None: ...\n    def __enter__(self: _S) -> _S: ...\n    def __exit__(\n        self,\n        exc_type: type[BaseException] | None,\n        exc_val: BaseException | None,\n        exc_tb: TracebackType | None,\n    ) -> None: ...\n    def format_message(self) -> str: ...\n    def add_test(self, cb: Callable[[Warning], bool]) -> None: ...\n\nclass AssertWarnsRegexContext(AssertWarnsContext):\n    pattern: str\n    def __init__(\n        self, warning_class: type[Warning], msg_fmt: str = ...\n    ) -> None: ...\n\ndef fail(msg: str = ...) -> NoReturn: ...\ndef assert_true(expr: object, msg_fmt: str = ...) -> None: ...\ndef assert_false(expr: object, msg_fmt: str = ...) -> None: ...\ndef assert_boolean_true(expr: object, msg_fmt: str = ...) -> None: ...\ndef assert_boolean_false(expr: object, msg_fmt: str = ...) -> None: ...\ndef assert_is_none(expr: object, msg_fmt: str = ...) -> None: ...\ndef assert_is_not_none(expr: object, msg_fmt: str = ...) -> None: ...\ndef assert_equal(\n    first: object, second: object, msg_fmt: str = ...\n) -> None: ...\ndef assert_not_equal(\n    first: object, second: object, msg_fmt: str = ...\n) -> None: ...\ndef assert_almost_equal(\n    first: float,\n    second: float,\n    msg_fmt: str = ...,\n    places: int = ...,\n    delta: float = ...,\n) -> None: ...\ndef assert_not_almost_equal(\n    first: float,\n    second: float,\n    msg_fmt: str = ...,\n    places: int = ...,\n    delta: float = ...,\n) -> None: ...\ndef assert_dict_equal(\n    first: dict,\n    second: dict,\n    key_msg_fmt: str = ...,\n    value_msg_fmt: str = ...,\n) -> None: ...\ndef assert_dict_superset(\n    first: dict,\n    second: dict,\n    key_msg_fmt: str = ...,\n    value_msg_fmt: str = ...,\n) -> None: ...\ndef assert_less(first: Any, second: Any, msg_fmt: str = ...) -> None: ...\ndef assert_less_equal(first: Any, second: Any, msg_fmt: str = ...) -> None: ...\ndef assert_greater(first: Any, second: Any, msg_fmt: str = ...) -> None: ...\ndef assert_greater_equal(\n    first: Any, second: Any, msg_fmt: str = ...\n) -> None: ...\ndef assert_regex(\n    text: str, regex: str | Pattern[str], msg_fmt: str = ...\n) -> None: ...\ndef assert_not_regex(\n    text: str, regex: str | Pattern[str], msg_fmt: str = ...\n) -> None: ...\ndef assert_is(first: object, second: object, msg_fmt: str = ...) -> None: ...\ndef assert_is_not(\n    first: object, second: object, msg_fmt: str = ...\n) -> None: ...\ndef assert_in(\n    first: Any, second: Container[Any], msg_fmt: str = ...\n) -> None: ...\ndef assert_not_in(\n    first: Any, second: Container[Any], msg_fmt: str = ...\n) -> None: ...\ndef assert_between(\n    lower_bound: Any, upper_bound: Any, expr: Any, msg_fmt: str = ...\n) -> None: ...\ndef assert_is_instance(\n    obj: object, cls: type | tuple[type, ...], msg_fmt: str = ...\n) -> None: ...\ndef assert_not_is_instance(\n    obj: object, cls: type | tuple[type, ...], msg_fmt: str = ...\n) -> None: ...\ndef assert_count_equal(\n    sequence1: Iterable[Any], sequence2: Iterable[Any], msg_fmt: str = ...\n) -> None: ...\ndef assert_has_attr(\n    obj: object, attribute: str, msg_fmt: str = ...\n) -> None: ...\ndef assert_datetime_about_now(\n    actual: datetime.datetime | None, msg_fmt: str = ...\n) -> None: ...\ndef assert_datetime_about_now_utc(\n    actual: datetime.datetime | None, msg_fmt: str = ...\n) -> None: ...\ndef assert_raises(\n    exception: type[BaseException], msg_fmt: str = ...\n) -> AssertRaisesContext: ...\ndef assert_raises_regex(\n    exception: type[BaseException],\n    regex: str | Pattern[str],\n    msg_fmt: str = ...,\n) -> AssertRaisesContext: ...\ndef assert_raises_errno(\n    exception: type[BaseException], errno: int, msg_fmt: str = ...\n) -> AssertRaisesContext: ...\ndef assert_succeeds(\n    exception: type[BaseException], msg_fmt: str = ...\n) -> ContextManager: ...\ndef assert_warns(\n    warning_type: type[Warning], msg_fmt: str = ...\n) -> AssertWarnsContext: ...\ndef assert_warns_regex(\n    warning_type: type[Warning], regex: str, msg_fmt: str = ...\n) -> AssertWarnsContext: ...\ndef assert_json_subset(\n    first: dict[Any, Any] | list[Any],  # dict key can be 'str' or 'Exists'\n    second: dict[str, Any] | list[Any] | str | bytes,\n) -> None: ...\n\nclass Present: ...\nclass Absent: ...\n\n@deprecated(\"Use Present and Absent instead.\")\nclass Exists:\n    member_name: str\n    def __init__(self, member_name: str) -> None: ...\n"
    },
    {
      "path": "srittau_python-asserts/asserts/py.typed",
      "content": ""
    }
  ],
  "BuggyCode": [
    {
      "path": "srittau_python-asserts/pyproject.toml",
      "content": "[tool.poetry]\nname = \"asserts\"\nversion = \"0.13.2.dev0\"\ndescription = \"Stand-alone Assertions\"\nreadme = \"README.md\"\nauthors = [\"Sebastian Rittau <srittau@rittau.biz>\"]\nlicense = \"MIT\"\nhomepage = \"https://github.com/srittau/python-asserts\"\nrepository = \"https://github.com/srittau/python-asserts\"\nclassifiers = [\n    \"Development Status :: 3 - Alpha\",\n    \"Intended Audience :: Developers\",\n    \"Topic :: Software Development :: Quality Assurance\",\n    \"Topic :: Software Development :: Testing\",\n]\npackages = [{ include = \"asserts\" }]\ninclude = [\"*/py.typed\", \"*.pyi\"]\n\n[tool.poetry.urls]\n\"GitHub\" = \"https://github.com/srittau/python-asserts\"\n\"Bug Tracker\" = \"https://github.com/srittau/python-asserts/issues\"\n\"Changes\" = \"https://github.com/srittau/python-asserts/blob/main/CHANGELOG.md\"\n\n[tool.poetry.dependencies]\npython = \">=3.8.1\"\ntyping-extensions = \"^4.10.0\"\n\n[tool.poetry.group.dev.dependencies]\nmypy = \">=1.9,<1.11\"\npoethepoet = \">=0.25,<0.27\"\nruff = \">=0.3.2,<0.5.0\"\n\n[tool.ruff]\nline-length = 79\ntarget-version = \"py38\"\n\n[tool.poe.tasks]\ntest = \"python3 -Wall -m unittest test_asserts\"\ndoctest = \"python3 -m doctest asserts/__init__.py\"\nlint = \"ruff check asserts test_asserts.py\"\ntypecheck = \"mypy asserts test_asserts.py\"\n\n[build-system]\nrequires = [\"poetry-core>=1.0.0\"]\nbuild-backend = \"poetry.core.masonry.api\"\n"
    },
    {
      "path": "srittau_python-asserts/test_asserts.py",
      "content": "# -*- coding: utf-8 -*-\n\nimport re\nimport sys\nfrom collections import OrderedDict\nfrom datetime import datetime, timedelta, timezone\nfrom json import JSONDecodeError\nfrom unittest import TestCase\nfrom warnings import catch_warnings, simplefilter, warn\n\nfrom asserts import (\n    Absent,\n    Exists,\n    Present,\n    assert_almost_equal,\n    assert_between,\n    assert_boolean_false,\n    assert_boolean_true,\n    assert_count_equal,\n    assert_datetime_about_now,\n    assert_datetime_about_now_utc,\n    assert_dict_equal,\n    assert_dict_superset,\n    assert_equal,\n    assert_false,\n    assert_greater,\n    assert_greater_equal,\n    assert_has_attr,\n    assert_in,\n    assert_is,\n    assert_is_instance,\n    assert_is_none,\n    assert_is_not,\n    assert_is_not_none,\n    assert_json_subset,\n    assert_less,\n    assert_less_equal,\n    assert_not_almost_equal,\n    assert_not_equal,\n    assert_not_in,\n    assert_not_is_instance,\n    assert_not_regex,\n    assert_raises,\n    assert_raises_errno,\n    assert_raises_regex,\n    assert_regex,\n    assert_succeeds,\n    assert_true,\n    assert_warns,\n    assert_warns_regex,\n    fail,\n)\n\n\nclass Box:\n    def __init__(self, initial_value):\n        self.value = initial_value\n\n\nclass _DummyObject(object):\n    def __init__(self, value=\"x\"):\n        self.value = value\n\n    def __repr__(self):\n        return \"<Dummy>\"\n\n\ndef _assert_raises_assertion(expected_message):\n    \"\"\"Fail if the context does not raise an AssertionError or the exception\n    message does not match.\n\n    This is used to test assertions, without using those assertions.\n\n    \"\"\"\n\n    class Context(object):\n        def __enter__(self):\n            pass\n\n        def __exit__(self, exc_type, exc_val, exc_tb):\n            if exc_type is None:\n                raise AssertionError(\"no AssertionError raised\")\n            if exc_type != AssertionError:\n                return False\n            if str(exc_val) != expected_message:\n                raise AssertionError(\n                    \"expected exception message {!r}, got {!r}\".format(\n                        expected_message, str(exc_val)\n                    )\n                )\n            return True\n\n    return Context()\n\n\nclass AssertTest(TestCase):\n    _type_string = \"type\" if sys.version_info[0] < 3 else \"class\"\n\n    # fail()\n\n    def test_fail__default_message(self):\n        with _assert_raises_assertion(\"assertion failure\"):\n            fail()\n\n    def test_fail__with_message(self):\n        with _assert_raises_assertion(\"test message\"):\n            fail(\"test message\")\n\n    # assert_true()\n\n    def test_assert_true__truthy_value(self):\n        assert_true(\"Hello World!\")\n\n    def test_assert_true__falsy_value__default_message(self):\n        with _assert_raises_assertion(\"'' is not truthy\"):\n            assert_true(\"\")\n\n    def test_assert_true__falsy_value__custom_message(self):\n        with _assert_raises_assertion(\"0 is not truthy;0\"):\n            assert_true(0, \"{msg};{expr}\")\n\n    # assert_false()\n\n    def test_assert_false__falsy_value(self):\n        assert_false(\"\")\n\n    def test_assert_false__truthy_value__default_message(self):\n        with _assert_raises_assertion(\"25 is not falsy\"):\n            assert_false(25)\n\n    def test_assert_false__truthy_value__custom_message(self):\n        with _assert_raises_assertion(\"'foo' is not falsy;foo\"):\n            assert_false(\"foo\", \"{msg};{expr}\")\n\n    # assert_boolean_true()\n\n    def test_assert_boolean_true__true(self):\n        assert_boolean_true(True)\n\n    def test_assert_boolean_true__false__custom_message(self):\n        with _assert_raises_assertion(\"'Foo' is not True;Foo\"):\n            assert_boolean_true(\"Foo\", \"{msg};{expr}\")\n\n    def test_assert_boolean_true__truthy__default_message(self):\n        with _assert_raises_assertion(\"1 is not True\"):\n            assert_boolean_true(1)\n\n    # assert_boolean_false()\n\n    def test_assert_boolean_false__false(self):\n        assert_boolean_false(False)\n\n    def test_assert_boolean_false__true__default_message(self):\n        with _assert_raises_assertion(\"'foo' is not False\"):\n            assert_boolean_false(\"foo\")\n\n    def test_assert_boolean_false__falsy__custom_message(self):\n        with _assert_raises_assertion(\"0 is not False;0\"):\n            assert_boolean_false(0, \"{msg};{expr}\")\n\n    # assert_is_none()\n\n    def test_assert_is_none__none(self):\n        assert_is_none(None)\n\n    def test_assert_is_none__string__default_message(self):\n        with _assert_raises_assertion(\"'' is not None\"):\n            assert_is_none(\"\")\n\n    def test_assert_is_none__int__custom_message(self):\n        with _assert_raises_assertion(\"55 is not None;55\"):\n            assert_is_none(55, \"{msg};{expr}\")\n\n    # assert_is_not_none()\n\n    def test_assert_is_not_none__string(self):\n        assert_is_not_none(\"\")\n\n    def test_assert_is_not_none__none__default_message(self):\n        with _assert_raises_assertion(\"expression is None\"):\n            assert_is_not_none(None)\n\n    def test_assert_is_not_none__none__custom_message(self):\n        with _assert_raises_assertion(\"expression is None;None\"):\n            assert_is_not_none(None, \"{msg};{expr!r}\")\n\n    # assert_equal()\n\n    def test_assert_equal__equal_strings(self):\n        assert_equal(\"foo\", \"foo\")\n\n    def test_assert_equal__equal_objects(self):\n        class MyClass(object):\n            def __eq__(self, other):\n                return True\n\n        assert_equal(MyClass(), MyClass())\n\n    def test_assert_equal__not_equal__default_message(self):\n        with _assert_raises_assertion(\"'string' != 55\"):\n            assert_equal(\"string\", 55)\n\n    def test_assert_equal__not_equal__custom_message(self):\n        with _assert_raises_assertion(\"'string' != 55;'string';55\"):\n            assert_equal(\"string\", 55, \"{msg};{first!r};{second!r}\")\n\n    def test_assert_equal__dict(self):\n        with _assert_raises_assertion(\"key 'foo' missing from right dict\"):\n            assert_equal({\"foo\": 5}, {})\n\n    # assert_not_equal()\n\n    def test_assert_not_equal__not_equal(self):\n        assert_not_equal(\"abc\", \"def\")\n\n    def test_assert_not_equal__equal__default_message(self):\n        with _assert_raises_assertion(\"'abc' == 'abc'\"):\n            assert_not_equal(\"abc\", \"abc\")\n\n    def test_assert_not_equal__equal__custom_message(self):\n        with _assert_raises_assertion(\"1.0 == 1;1.0;1\"):\n            assert_not_equal(1.0, 1, \"{msg};{first};{second}\")\n\n    # assert_almost_equal()\n\n    def test_assert_almost_equal__same(self):\n        assert_almost_equal(5, 5)\n\n    def test_assert_almost_equal__similar__defaults(self):\n        assert_almost_equal(5, 5.00000001)\n\n    def test_assert_almost_equal__similar__places(self):\n        assert_almost_equal(5, 5.0001, places=3)\n\n    def test_assert_almost_equal__similar__delta(self):\n        assert_almost_equal(5, 5.001, delta=0.1)\n\n    def test_assert_almost_equal__similar__delta_reverse(self):\n        assert_almost_equal(5, 5.001, delta=0.1)\n\n    def test_assert_almost_equal__not_similar__default_message(self):\n        with _assert_raises_assertion(\"5 != 5.0001 within 7 places\"):\n            assert_almost_equal(5, 5.0001)\n\n    def test_assert_almost_equal__not_similar__places__default_message(self):\n        with _assert_raises_assertion(\"5 != 6 within 3 places\"):\n            assert_almost_equal(5, 6, places=3)\n\n    def test_assert_almost_equal__not_similar__delta__default_message(self):\n        with _assert_raises_assertion(\"5 != 6 with delta=0.1\"):\n            assert_almost_equal(5, 6, delta=0.1)\n\n    def test_assert_almost_equal__not_similar__delta_reverse(self):\n        with _assert_raises_assertion(\"6 != 5 with delta=0.3\"):\n            assert_almost_equal(6, 5, delta=0.3)\n\n    def test_assert_almost_equal__not_similar__custom_message(self):\n        with _assert_raises_assertion(\"5 != -5 within 7 places;5;-5;7;None\"):\n            assert_almost_equal(\n                5, -5, msg_fmt=\"{msg};{first};{second};{places};{delta!r}\"\n            )\n\n    def test_assert_almost_equal__not_similar__places__custom_message(self):\n        with _assert_raises_assertion(\"5 != -5 within 3 places;5;-5;3;None\"):\n            assert_almost_equal(\n                5,\n                -5,\n                places=3,\n                msg_fmt=\"{msg};{first};{second};{places};{delta!r}\",\n            )\n\n    def test_assert_almost_equal__not_similar__delta__custom_message(self):\n        with _assert_raises_assertion(\"5 != 6 with delta=0.1;5;6;None;0.1\"):\n            assert_almost_equal(\n                5,\n                6,\n                delta=0.1,\n                msg_fmt=\"{msg};{first};{second};{places!r};{delta}\",\n            )\n\n    def test_assert_almost_equal__wrong_types(self):\n        try:\n            assert_almost_equal(\"5\", \"5\")  # type: ignore[arg-type]\n        except TypeError:\n            pass\n        else:\n            raise AssertionError(\"TypeError not raised\")\n\n    def test_assert_almost_equal__places_and_delta(self):\n        try:\n            assert_almost_equal(5, 5, places=3, delta=0.0003)\n        except TypeError:\n            pass\n        else:\n            raise AssertionError(\"TypeError not raised\")\n\n    def test_assert_almost_equal__delta_eq_0(self):\n        try:\n            assert_almost_equal(5, 5, delta=0)\n        except ValueError:\n            pass\n        else:\n            raise AssertionError(\"ValueError not raised\")\n\n    def test_assert_almost_equal__delta_lt_0(self):\n        try:\n            assert_almost_equal(5, 5, delta=-1)\n        except ValueError:\n            pass\n        else:\n            raise AssertionError(\"ValueError not raised\")\n\n    # assert_not_almost_equal()\n\n    def test_assert_not_almost_equal__same(self):\n        with _assert_raises_assertion(\"5 == 5 within 7 places\"):\n            assert_not_almost_equal(5, 5)\n\n    def test_assert_not_almost_equal__similar__defaults(self):\n        with _assert_raises_assertion(\"5 == 5.00000001 within 7 places\"):\n            assert_not_almost_equal(5, 5.00000001)\n\n    def test_assert_not_almost_equal__similar__places(self):\n        with _assert_raises_assertion(\"5 == 5.0001 within 3 places\"):\n            assert_not_almost_equal(5, 5.0001, places=3)\n\n    def test_assert_not_almost_equal__similar__delta(self):\n        with _assert_raises_assertion(\"5 == 5.1 with delta=0.1\"):\n            assert_not_almost_equal(5, 5.1, delta=0.1)\n\n    def test_assert_not_almost_equal__similar__delta_reverse(self):\n        with _assert_raises_assertion(\"5 != 6 with delta=0.3\"):\n            assert_almost_equal(5, 6, delta=0.3)\n\n    def test_assert_not_almost_equal__not_similar(self):\n        assert_not_almost_equal(5, 5.0001)\n\n    def test_assert_not_almost_equal__not_similar__delta(self):\n        assert_not_almost_equal(5, 5.1, delta=0.05)\n\n    def test_assert_not_almost_equal__not_similar__delta_reverse(self):\n        assert_not_almost_equal(5.1, 5, delta=0.05)\n\n    def test_assert_not_almost_equal__similar__custom_message(self):\n        with _assert_raises_assertion(\n            \"5 == 5.00000001 within 7 places;5;5.00000001;7;None\"\n        ):\n            assert_not_almost_equal(\n                5,\n                5.00000001,\n                msg_fmt=\"{msg};{first};{second};{places};{delta!r}\",\n            )\n\n    def test_assert_not_almost_equal__similar__places__custom_message(self):\n        with _assert_raises_assertion(\n            \"5 == 5.0001 within 3 places;5;5.0001;3;None\"\n        ):\n            assert_not_almost_equal(\n                5,\n                5.0001,\n                places=3,\n                msg_fmt=\"{msg};{first};{second};{places};{delta!r}\",\n            )\n\n    def test_assert_not_almost_equal__similar__delta__custom_message(self):\n        with _assert_raises_assertion(\"5 == 6 with delta=1.1;5;6;None;1.1\"):\n            assert_not_almost_equal(\n                5,\n                6,\n                delta=1.1,\n                msg_fmt=\"{msg};{first};{second};{places!r};{delta}\",\n            )\n\n    def test_assert_not_almost_equal__wrong_types(self):\n        try:\n            assert_not_almost_equal(\"5\", \"5\")  # type: ignore[arg-type]\n        except TypeError:\n            pass\n        else:\n            raise AssertionError(\"TypeError not raised\")\n\n    def test_assert_not_almost_equal__places_and_delta(self):\n        try:\n            assert_not_almost_equal(5, 5, places=3, delta=0.0003)\n        except TypeError:\n            pass\n        else:\n            raise AssertionError(\"TypeError not raised\")\n\n    def test_not_assert_almost_equal__delta_eq_0(self):\n        try:\n            assert_not_almost_equal(5, 5, delta=0)\n        except ValueError:\n            pass\n        else:\n            raise AssertionError(\"ValueError not raised\")\n\n    def test_not_assert_almost_equal__delta_lt_0(self):\n        try:\n            assert_not_almost_equal(5, 5, delta=-1)\n        except ValueError:\n            pass\n        else:\n            raise AssertionError(\"ValueError not raised\")\n\n    # assert_dict_equal()\n\n    def test_assert_dict_equal__empty_dicts(self):\n        assert_dict_equal({}, {})\n\n    def test_assert_dict_equal__dicts_are_equal(self):\n        assert_dict_equal({\"foo\": 5}, {\"foo\": 5})\n\n    def test_assert_dict_equal__one_key_missing_from_right(self):\n        with _assert_raises_assertion(\"key 'foo' missing from right dict\"):\n            assert_dict_equal({\"bar\": 10, \"foo\": 5}, {\"bar\": 10})\n\n    def test_assert_dict_equal__multiple_keys_missing_from_right(self):\n        with _assert_raises_assertion(\n            \"keys 'bar', 'foo' missing from right dict\"\n        ):\n            assert_dict_equal({\"foo\": 5, \"bar\": 10, \"baz\": 15}, {\"baz\": 15})\n\n    def test_assert_dict_equal__one_key_missing_from_left(self):\n        with _assert_raises_assertion(\"extra key 'foo' in right dict\"):\n            assert_dict_equal({\"bar\": 10}, {\"bar\": 10, \"foo\": 5})\n\n    def test_assert_dict_equal__multiple_keys_missing_from_left(self):\n        with _assert_raises_assertion(\"extra keys 'bar', 'foo' in right dict\"):\n            assert_dict_equal({\"baz\": 15}, {\"foo\": 5, \"bar\": 10, \"baz\": 15})\n\n    def test_assert_dict_equal__values_do_not_match(self):\n        with _assert_raises_assertion(\"key 'foo' differs: 15 != 10\"):\n            assert_dict_equal({\"foo\": 15}, {\"foo\": 10})\n\n    def test_assert_dict_equal__not_string_keys(self):\n        with _assert_raises_assertion(\"key 10 missing from right dict\"):\n            assert_dict_equal({10: \"foo\"}, {})\n        with _assert_raises_assertion(\"keys 'foo', 5 missing from right dict\"):\n            assert_dict_equal({5: \"\", \"foo\": \"\"}, {})\n        with _assert_raises_assertion(\"extra key 10 in right dict\"):\n            assert_dict_equal({}, {10: \"foo\"})\n        with _assert_raises_assertion(\"extra keys 'foo', 5 in right dict\"):\n            assert_dict_equal({}, {5: \"\", \"foo\": \"\"})\n\n    def test_assert_dict_equal__message_precedence(self):\n        with _assert_raises_assertion(\"key 'foo' missing from right dict\"):\n            assert_dict_equal(\n                {\"foo\": \"\", \"bar\": \"\", \"baz\": 5},\n                {\"bar\": \"\", \"baz\": 10, \"extra\": \"\"},\n            )\n        with _assert_raises_assertion(\"extra key 'extra' in right dict\"):\n            assert_dict_equal(\n                {\"bar\": \"\", \"baz\": 5}, {\"bar\": \"\", \"baz\": 10, \"extra\": \"\"}\n            )\n\n    def test_assert_dict_equal__custom_key_message(self):\n        with _assert_raises_assertion(\n            \"key 'foo' missing from right dict;\"\n            \"{'foo': ''};{'bar': ''};['foo'];['bar']\"\n        ):\n            assert_dict_equal(\n                {\"foo\": \"\"},\n                {\"bar\": \"\"},\n                key_msg_fmt=\"{msg};{first!r};{second!r};\"\n                \"{missing_keys!r};{extra_keys!r}\",\n            )\n\n    def test_assert_dict_equal__custom_value_message(self):\n        with _assert_raises_assertion(\n            \"key 'foo' differs: 5 != 10;{'foo': 5};{'foo': 10};\" \"'foo';5;10\"\n        ):\n            assert_dict_equal(\n                {\"foo\": 5},\n                {\"foo\": 10},\n                value_msg_fmt=\"{msg};{first!r};{second!r};\"\n                \"{key!r};{first_value};{second_value}\",\n            )\n\n    # assert_dict_superset()\n\n    def test_assert_dict_superset__empty_dicts(self):\n        assert_dict_superset({}, {})\n\n    def test_assert_dict_superset__dicts_are_equal(self):\n        assert_dict_superset({\"foo\": 5}, {\"foo\": 5})\n\n    def test_assert_dict_superset__dicts_is_superset(self):\n        assert_dict_superset({\"foo\": 5}, {\"foo\": 5, \"bar\": 10})\n\n    def test_assert_dict_superset__one_key_missing_from_right(self):\n        with _assert_raises_assertion(\"key 'foo' missing from right dict\"):\n            assert_dict_superset({\"bar\": 10, \"foo\": 5}, {\"bar\": 10})\n\n    def test_assert_dict_superset__multiple_keys_missing_from_right(self):\n        with _assert_raises_assertion(\n            \"keys 'bar', 'foo' missing from right dict\"\n        ):\n            assert_dict_superset({\"foo\": 5, \"bar\": 10, \"baz\": 15}, {\"baz\": 15})\n\n    def test_assert_dict_superset__values_do_not_match(self):\n        with _assert_raises_assertion(\"key 'foo' differs: 15 != 10\"):\n            assert_dict_superset({\"foo\": 15}, {\"foo\": 10, \"bar\": 15})\n\n    def test_assert_dict_superset__not_string_keys(self):\n        with _assert_raises_assertion(\"key 10 missing from right dict\"):\n            assert_dict_superset({10: \"foo\"}, {})\n        with _assert_raises_assertion(\"keys 'foo', 5 missing from right dict\"):\n            assert_dict_superset({5: \"\", \"foo\": \"\"}, {})\n\n    def test_assert_dict_superset__message_precedence(self):\n        with _assert_raises_assertion(\"key 'foo' missing from right dict\"):\n            assert_dict_superset({\"foo\": \"\", \"bar\": 5}, {\"bar\": 1})\n\n    def test_assert_dict_superset__custom_key_message(self):\n        with _assert_raises_assertion(\n            \"key 'foo' missing from right dict;\"\n            \"{'foo': ''};{'bar': ''};['foo']\"\n        ):\n            assert_dict_superset(\n                {\"foo\": \"\"},\n                {\"bar\": \"\"},\n                key_msg_fmt=\"{msg};{first!r};{second!r};\" \"{missing_keys!r}\",\n            )\n\n    def test_assert_dict_superset__custom_value_message(self):\n        with _assert_raises_assertion(\n            \"key 'foo' differs: 5 != 10;{'foo': 5};{'foo': 10};\" \"'foo';5;10\"\n        ):\n            assert_dict_superset(\n                {\"foo\": 5},\n                {\"foo\": 10},\n                value_msg_fmt=\"{msg};{first!r};{second!r};\"\n                \"{key!r};{first_value};{second_value}\",\n            )\n\n    # assert_less()\n\n    def test_assert_less(self):\n        assert_less(4, 5)\n        with _assert_raises_assertion(\"5 is not less than 5\"):\n            assert_less(5, 5)\n        with _assert_raises_assertion(\"'foo' is not less than 'bar'\"):\n            assert_less(\"foo\", \"bar\")\n        with _assert_raises_assertion(\"6 is not less than 5;6;5\"):\n            assert_less(6, 5, \"{msg};{first};{second}\")\n\n    # assert_less_equal()\n\n    def test_assert_less_equal(self):\n        assert_less_equal(4, 5)\n        assert_less_equal(5, 5)\n        with _assert_raises_assertion(\n            \"'foo' is not less than or equal to 'bar'\"\n        ):\n            assert_less_equal(\"foo\", \"bar\")\n        with _assert_raises_assertion(\"6 is not less than or equal to 5;6;5\"):\n            assert_less_equal(6, 5, \"{msg};{first};{second}\")\n\n    # assert_greater()\n\n    def test_assert_greater(self):\n        assert_greater(5, 4)\n        with _assert_raises_assertion(\"5 is not greater than 5\"):\n            assert_greater(5, 5)\n        with _assert_raises_assertion(\"'bar' is not greater than 'foo'\"):\n            assert_greater(\"bar\", \"foo\")\n        with _assert_raises_assertion(\"5 is not greater than 6;5;6\"):\n            assert_greater(5, 6, \"{msg};{first};{second}\")\n\n    # assert_greater_equal()\n\n    def test_assert_greater_equal(self):\n        assert_greater_equal(5, 4)\n        assert_greater_equal(5, 5)\n        with _assert_raises_assertion(\n            \"'bar' is not greater than or equal to 'foo'\"\n        ):\n            assert_greater_equal(\"bar\", \"foo\")\n        with _assert_raises_assertion(\n            \"5 is not greater than or equal to 6;5;6\"\n        ):\n            assert_greater_equal(5, 6, \"{msg};{first};{second}\")\n\n    # assert_regex()\n\n    def test_assert_regex__matches_string(self):\n        assert_regex(\"This is a test text\", \"is.*test\")\n\n    def test_assert_regex__matches_regex(self):\n        regex = re.compile(\"is.*test\")\n        assert_regex(\"This is a test text\", regex)\n\n    def test_assert_regex__does_not_match_string__default_message(self):\n        with _assert_raises_assertion(\n            \"'This is a test text' does not match 'not found'\"\n        ):\n            assert_regex(\"This is a test text\", \"not found\")\n\n    def test_assert_regex__does_not_match_regex__default_message(self):\n        regex = re.compile(r\"not found\")\n        with _assert_raises_assertion(\n            \"'This is a test text' does not match 'not found'\"\n        ):\n            assert_regex(\"This is a test text\", regex)\n\n    def test_assert_regex__does_not_match_string__custom_message(self):\n        with _assert_raises_assertion(\n            \"'Wrong text' does not match 'not found';\"\n            \"'Wrong text';'not found'\"\n        ):\n            assert_regex(\n                \"Wrong text\", r\"not found\", \"{msg};{text!r};{pattern!r}\"\n            )\n\n    def test_assert_regex__does_not_match_regex__custom_message(self):\n        regex = re.compile(r\"not found\")\n        with _assert_raises_assertion(\n            \"'Wrong text' does not match 'not found';'Wrong text';\"\n            \"'not found'\"\n        ):\n            assert_regex(\"Wrong text\", regex, \"{msg};{text!r};{pattern!r}\")\n\n    # assert_not_regex()\n\n    def test_assert_not_regex__does_not_match_string(self):\n        assert_not_regex(\"This is a test text\", \"no match\")\n\n    def test_assert_not_regex__does_not_match_regex(self):\n        regex = re.compile(\"no match\")\n        assert_not_regex(\"This is a test text\", regex)\n\n    def test_assert_not_regex__matches_string__default_message(self):\n        with _assert_raises_assertion(\n            \"'This is a test text' matches 'is.*test'\"\n        ):\n            assert_not_regex(\"This is a test text\", \"is.*test\")\n\n    def test_assert_not_regex__matches_regex__default_message(self):\n        regex = re.compile(\"is.*test\")\n        with _assert_raises_assertion(\n            \"'This is a test text' matches 'is.*test'\"\n        ):\n            assert_not_regex(\"This is a test text\", regex)\n\n    def test_assert_not_regex__matches_string__custom_message(self):\n        with _assert_raises_assertion(\n            \"'This is a test text' matches 'is.*test';\"\n            \"'This is a test text';'is.*test'\"\n        ):\n            assert_not_regex(\n                \"This is a test text\",\n                \"is.*test\",\n                \"{msg};{text!r};{pattern!r}\",\n            )\n\n    def test_assert_not_regex__matches_regex__custom_message(self):\n        regex = re.compile(\"is.*test\")\n        with _assert_raises_assertion(\n            \"'This is a test text' matches 'is.*test';'This is a test text';\"\n            \"'is.*test'\"\n        ):\n            assert_not_regex(\n                \"This is a test text\", regex, \"{msg};{text!r};{pattern!r}\"\n            )\n\n    # assert_is()\n\n    def test_assert_is__same(self):\n        x = _DummyObject()\n        assert_is(x, x)\n\n    def test_assert_is__not_same__default_message(self):\n        with _assert_raises_assertion(\"'x' is not 'y'\"):\n            assert_is(\"x\", \"y\")\n\n    def test_assert_is__equal_but_not_same__custom_message(self):\n        x = \"x\"\n        y = _DummyObject(\"y\")\n        with _assert_raises_assertion(\"'x' is not <Dummy>;'x';y\"):\n            assert_is(x, y, \"{msg};{first!r};{second.value}\")\n\n    # assert_is_not()\n\n    def test_assert_is_not__not_same(self):\n        x = _DummyObject()\n        y = _DummyObject()\n        assert_is_not(x, y)\n\n    def test_assert_is_not__same__default_message(self):\n        x = _DummyObject(\"x\")\n        with _assert_raises_assertion(\"both arguments refer to <Dummy>\"):\n            assert_is_not(x, x)\n\n    def test_assert_is_not__same__custom_message(self):\n        x = _DummyObject(\"x\")\n        with _assert_raises_assertion(\"both arguments refer to <Dummy>;x;x\"):\n            assert_is_not(x, x, \"{msg};{first.value};{second.value}\")\n\n    # assert_in()\n\n    def test_assert_in__contains(self):\n        assert_in(\"foo\", [\"foo\", \"bar\", \"baz\"])\n\n    def test_assert_in__does_not_contain__default_message(self):\n        with _assert_raises_assertion(\"'foo' not in []\"):\n            assert_in(\"foo\", [])\n\n    def test_assert_in__does_not_contain__custom_message(self):\n        with _assert_raises_assertion(\"'foo' not in [];'foo';[]\"):\n            assert_in(\"foo\", [], \"{msg};{first!r};{second!r}\")\n\n    # assert_not_in()\n\n    def test_assert_not_in__does_not_contain(self):\n        assert_not_in(\"foo\", [])\n\n    def test_assert_not_in__does_contain__default_message(self):\n        with _assert_raises_assertion(\"'foo' is in ['foo', 'bar', 'baz']\"):\n            assert_not_in(\"foo\", [\"foo\", \"bar\", \"baz\"])\n\n    def test_assert_not_in__does_contain__custom_message(self):\n        with _assert_raises_assertion(\"'foo' is in ['foo', 'bar'];'foo';bar\"):\n            assert_not_in(\"foo\", [\"foo\", \"bar\"], \"{msg};{first!r};{second[1]}\")\n\n    # assert_count_equal()\n\n    def test_assert_count_equal__equal(self):\n        with assert_succeeds(AssertionError):\n            assert_count_equal([\"a\"], [\"a\"])\n\n    def test_assert_count_equal__equal_differing_types(self):\n        with assert_succeeds(AssertionError):\n            assert_count_equal([\"a\"], {\"a\"})\n\n    def test_assert_count_equal__ignore_order(self):\n        with assert_succeeds(AssertionError):\n            assert_count_equal([\"a\", \"b\"], [\"b\", \"a\"])\n\n    def test_assert_count_equal__missing_from_sequence1(self):\n        with _assert_raises_assertion(\"missing from sequence 1: 'a'\"):\n            assert_count_equal([], {\"a\"})\n\n    def test_assert_count_equal__multiple_missing_from_sequence1(self):\n        with _assert_raises_assertion(\"missing from sequence 1: 'b', 'c'\"):\n            assert_count_equal([\"a\"], [\"a\", \"b\", \"c\"])\n\n    def test_assert_count_equal__respect_duplicates(self):\n        with _assert_raises_assertion(\"missing from sequence 1: 'a'\"):\n            assert_count_equal({\"a\"}, [\"a\", \"a\"])\n\n    def test_assert_count_equal__missing_from_sequence2(self):\n        with _assert_raises_assertion(\"missing from sequence 2: 'a', 'c'\"):\n            assert_count_equal([\"a\", \"b\", \"c\"], [\"b\"])\n\n    def test_assert_count_equal__missing_from_both(self):\n        msg = \"missing from sequence 1: 'd'; missing from sequence 2: 'b', 'c'\"\n        with _assert_raises_assertion(msg):\n            assert_count_equal([\"a\", \"b\", \"c\"], [\"a\", \"d\"])\n\n    def test_assert_count_equal__custom_message(self):\n        with _assert_raises_assertion(\"missing from sequence 1: 'a';[];['a']\"):\n            assert_count_equal([], [\"a\"], \"{msg};{first};{second}\")\n\n    # assert_between()\n\n    def test_assert_between__within_range(self):\n        assert_between(0, 10, 0)\n        assert_between(0, 10, 10)\n        assert_between(0, 10, 5)\n\n    def test_assert_between__too_low__default_message(self):\n        with _assert_raises_assertion(\"-1 is not between 0 and 10\"):\n            assert_between(0, 10, -1)\n\n    def test_assert_between__too_high__custom_message(self):\n        with _assert_raises_assertion(\"11 is not between 0 and 10;0;10;11\"):\n            assert_between(0, 10, 11, \"{msg};{lower};{upper};{expr}\")\n\n    # assert_is_instance()\n\n    def _is_instance_message(self, expr, expected_type, real_type):\n        expected_message = (\n            \"{!r} is an instance of <class {}>, expected {}\".format(\n                expr, real_type, expected_type\n            )\n        )\n        if sys.version_info[0] < 3:\n            return expected_message.replace(\"class\", \"type\")\n        else:\n            return expected_message\n\n    def test_assert_is_instance__single_type(self):\n        assert_is_instance(4, int)\n        assert_is_instance(OSError(), Exception)\n\n    def test_assert_is_instance__multiple_types(self):\n        assert_is_instance(4, (str, int))\n\n    def test_assert_is_instance__default_message(self):\n        expected_message = self._is_instance_message(\n            \"my string\", \"<class 'int'>\", \"'str'\"\n        )\n        with _assert_raises_assertion(expected_message):\n            assert_is_instance(\"my string\", int)\n\n    def test_assert_is_instance__custom_message_single_type(self):\n        expected_message = self._is_instance_message(\n            \"my string\", \"<class 'int'>\", \"'str'\"\n        )\n        expected = \"{};my string;(<class 'int'>,)\".format(expected_message)\n        expected = expected.replace(\"class\", self._type_string)\n        with _assert_raises_assertion(expected):\n            assert_is_instance(\"my string\", int, \"{msg};{obj};{types}\")\n\n    def test_assert_is_instance__custom_message_multiple_types(self):\n        expected_message = self._is_instance_message(\n            \"my string\", \"(<class 'int'>, <class 'float'>)\", \"'str'\"\n        )\n        expected = \"{};my string;(<class 'int'>, <class 'float'>)\".format(\n            expected_message\n        )\n        expected = expected.replace(\"class\", self._type_string)\n        with _assert_raises_assertion(expected):\n            assert_is_instance(\n                \"my string\", (int, float), \"{msg};{obj};{types}\"\n            )\n\n    # assert_not_is_instance()\n\n    def _not_is_instance_message(self, obj):\n        expected_message = \"{!r} is an instance of {}\".format(\n            obj, obj.__class__\n        )\n        if sys.version_info[0] < 3:\n            expected_message = expected_message.replace(\"class\", \"type\")\n            expected_message = expected_message.replace(\n                \"type 'OSError'\", \"type 'exceptions.OSError'\"\n            )\n        return expected_message\n\n    def test_assert_not_is_instance__single_type(self):\n        assert_not_is_instance(4, str)\n\n    def test_assert_not_is_instance__multiple_types(self):\n        assert_not_is_instance(4, (str, bytes))\n\n    def test_assert_not_is_instance__default_message(self):\n        obj = OSError()\n        expected_message = self._not_is_instance_message(obj)\n        with _assert_raises_assertion(expected_message):\n            assert_not_is_instance(obj, Exception)\n\n    def test_assert_not_is_instance__custom_message__single_type(self):\n        msg = self._not_is_instance_message(\"Foo\")\n        expected = \"{};Foo;(<class 'str'>,)\".format(msg)\n        expected = expected.replace(\"class\", self._type_string)\n        with _assert_raises_assertion(expected):\n            assert_not_is_instance(\"Foo\", str, \"{msg};{obj};{types!r}\")\n\n    def test_assert_not_is_instance__custom_message__multiple_types(self):\n        msg = self._not_is_instance_message(\"Foo\")\n        expected = \"{};Foo;(<class 'str'>, <class 'int'>)\".format(msg)\n        expected = expected.replace(\"class\", self._type_string)\n        with _assert_raises_assertion(expected):\n            assert_not_is_instance(\"Foo\", (str, int), \"{msg};{obj};{types!r}\")\n\n    # assert_has_attr()\n\n    def test_assert_has_attr__has_attribute(self):\n        d = _DummyObject()\n        assert_has_attr(d, \"value\")\n\n    def test_assert_has_attr__does_not_have_attribute__default_message(self):\n        d = _DummyObject()\n        with _assert_raises_assertion(\"<Dummy> does not have attribute 'foo'\"):\n            assert_has_attr(d, \"foo\")\n\n    def test_assert_has_attr__does_not_have_attribute__custom_message(self):\n        d = _DummyObject()\n        expected = \"<Dummy> does not have attribute 'foo';<Dummy>;foo\"\n        with _assert_raises_assertion(expected):\n            assert_has_attr(d, \"foo\", msg_fmt=\"{msg};{obj!r};{attribute}\")\n\n    # assert_datetime_about_now()\n\n    def test_assert_datetime_about_now__close(self):\n        assert_datetime_about_now(datetime.now())\n\n    def test_assert_datetime_about_now__none__default_message(self):\n        expected_message = r\"^None is not a valid date/time$\"\n        with assert_raises_regex(AssertionError, expected_message):\n            assert_datetime_about_now(None)\n\n    def test_assert_datetime_about_now__none__custom_message(self):\n        dt = datetime.now().date().isoformat()\n        expected = \"None is not a valid date/time;None;{}\".format(dt)\n        with _assert_raises_assertion(expected):\n            assert_datetime_about_now(\n                None, msg_fmt=\"{msg};{actual!r};{now:%Y-%m-%d}\"\n            )\n\n    def test_assert_datetime_about_now__too_low(self):\n        then = datetime.now() - timedelta(minutes=1)\n        with assert_raises(AssertionError):\n            assert_datetime_about_now(then)\n\n    def test_assert_datetime_about_now__too_high(self):\n        then = datetime.now() + timedelta(minutes=1)\n        with assert_raises(AssertionError):\n            assert_datetime_about_now(then)\n\n    def test_assert_datetime_about_now__default_message(self):\n        then = datetime(1990, 4, 13, 12, 30, 15)\n        expected_message = (\n            r\"^datetime.datetime\\(1990, 4, 13, 12, 30, 15\\) \"\n            \"is not close to current date/time$\"\n        )\n        with assert_raises_regex(AssertionError, expected_message):\n            assert_datetime_about_now(then)\n\n    def test_assert_datetime_about_now__custom_message(self):\n        then = datetime(1990, 4, 13, 12, 30, 15)\n        now = datetime.now().date().isoformat()\n        expected = (\n            \"datetime.datetime(1990, 4, 13, 12, 30, 15) \"\n            \"is not close to current date/time;12:30;{}\".format(now)\n        )\n        with _assert_raises_assertion(expected):\n            assert_datetime_about_now(\n                then, msg_fmt=\"{msg};{actual:%H:%M};{now:%Y-%m-%d}\"\n            )\n\n    # assert_datetime_about_now_utc()\n\n    def test_assert_datetime_about_now_utc__close(self):\n        assert_datetime_about_now_utc(\n            datetime.now(timezone.utc).replace(tzinfo=None)\n        )\n\n    def test_assert_datetime_about_now_utc__none__default_message(self):\n        expected_message = r\"^None is not a valid date/time$\"\n        with assert_raises_regex(AssertionError, expected_message):\n            assert_datetime_about_now_utc(None)\n\n    def test_assert_datetime_about_now_utc__none__custom_message(self):\n        dt = datetime.now(timezone.utc).date().isoformat()\n        expected = \"None is not a valid date/time;None;{}\".format(dt)\n        with _assert_raises_assertion(expected):\n            assert_datetime_about_now_utc(\n                None, msg_fmt=\"{msg};{actual!r};{now:%Y-%m-%d}\"\n            )\n\n    def test_assert_datetime_about_now_utc__too_low(self):\n        then = datetime.now(timezone.utc).replace(tzinfo=None) - timedelta(\n            minutes=1\n        )\n        with assert_raises(AssertionError):\n            assert_datetime_about_now_utc(then)\n\n    def test_assert_datetime_about_now_utc__too_high(self):\n        then = datetime.now(timezone.utc).replace(tzinfo=None) + timedelta(\n            minutes=1\n        )\n        with assert_raises(AssertionError):\n            assert_datetime_about_now_utc(then)\n\n    def test_assert_datetime_about_now_utc__default_message(self):\n        then = datetime(1990, 4, 13, 12, 30, 15)\n        expected_message = (\n            r\"datetime.datetime\\(1990, 4, 13, 12, 30, 15\\) \"\n            r\"is not close to current UTC date/time$\"\n        )\n        with assert_raises_regex(AssertionError, expected_message):\n            assert_datetime_about_now_utc(then)\n\n    def test_assert_datetime_about_now_utc__custom_message(self):\n        then = datetime(1990, 4, 13, 12, 30, 15)\n        now = datetime.now(timezone.utc).date().isoformat()\n        expected = (\n            \"datetime.datetime(1990, 4, 13, 12, 30, 15) \"\n            \"is not close to current UTC date/time;12:30;{}\".format(now)\n        )\n        with _assert_raises_assertion(expected):\n            assert_datetime_about_now_utc(\n                then, msg_fmt=\"{msg};{actual:%H:%M};{now:%Y-%m-%d}\"\n            )\n\n    # assert_raises()\n\n    def test_assert_raises__raises_right_exception(self):\n        with assert_raises(KeyError):\n            raise KeyError()\n\n    def test_assert_raises__exc_val(self):\n        exc = KeyError()\n        with assert_raises(KeyError) as context:\n            raise exc\n        assert_is(exc, context.exc_val)\n\n    def test_assert_raises__exc_val_within_context(self):\n        with assert_raises(RuntimeError):\n            with assert_raises(KeyError) as context:\n                context.exc_val\n\n    def test_assert_raises__raises_subclass(self):\n        class MyError(IndexError):\n            pass\n\n        with assert_raises(IndexError):\n            raise MyError()\n\n    def test_assert_raises__exception_not_raised__default_message(self):\n        with _assert_raises_assertion(\"KeyError not raised\"):\n            with assert_raises(KeyError):\n                pass\n\n    def test_assert_raises__exception_not_raised__custom_message(self):\n        expected = \"KeyError not raised;KeyError;KeyError\"\n        with _assert_raises_assertion(expected):\n            with assert_raises(\n                KeyError, msg_fmt=\"{msg};{exc_type.__name__};{exc_name}\"\n            ):\n                pass\n\n    def test_assert_raises__wrong_exception_raised(self):\n        try:\n            with assert_raises(IndexError):\n                raise KeyError()\n        except KeyError:\n            pass\n        except Exception as exc:\n            fail(str(exc) + \" was raised\")\n        else:\n            fail(\"no exception raised\")\n\n    def test_assert_raises__add_test_called(self):\n        called = Box(False)\n\n        def extra_test(exc):\n            assert_is_instance(exc, KeyError)\n            called.value = True\n\n        with assert_raises(KeyError) as context:\n            context.add_test(extra_test)\n            raise KeyError()\n        assert_true(called.value, \"extra_test() was not called\")\n\n    def test_assert_raises__add_test_not_called(self):\n        called = Box(False)\n\n        def extra_test(_):\n            called.value = True\n\n        with assert_raises(AssertionError):\n            with assert_raises(KeyError) as context:\n                context.add_test(extra_test)\n        assert_false(called.value, \"extra_test() was unexpectedly called\")\n\n    # assert_raises_regex()\n\n    def test_assert_raises_regex__raises_right_exception(self):\n        with assert_raises_regex(KeyError, r\"test.*\"):\n            raise KeyError(\"test message\")\n\n    def test_assert_raises_regex__raises_right_exception__compiled(self):\n        with assert_raises_regex(KeyError, re.compile(r\"test.*\")):\n            raise KeyError(\"test message\")\n\n    def test_assert_raises_regex__exception_not_raised__default_message(self):\n        with _assert_raises_assertion(\"KeyError not raised\"):\n            with assert_raises_regex(KeyError, r\"test\"):\n                pass\n\n    def test_assert_raises_regex__exception_not_raised__custom_message(self):\n        expected = \"KeyError not raised;KeyError;KeyError;'';test\"\n        with _assert_raises_assertion(expected):\n            msg_fmt = \"{msg};{exc_type.__name__};{exc_name};{text!r};{pattern}\"\n            with assert_raises_regex(KeyError, r\"test\", msg_fmt=msg_fmt):\n                pass\n\n    def test_assert_raises_regex__no_message__default_message(self):\n        with _assert_raises_assertion(\"KeyError without message\"):\n            with assert_raises_regex(KeyError, r\"test\"):\n                raise KeyError()\n\n    def test_assert_raises_regex__no_message__custom_message(self):\n        expected = \"KeyError without message;KeyError;KeyError;None;test\"\n        with _assert_raises_assertion(expected):\n            msg_fmt = \"{msg};{exc_type.__name__};{exc_name};{text!r};{pattern}\"\n            with assert_raises_regex(KeyError, r\"test\", msg_fmt=msg_fmt):\n                raise KeyError()\n\n    def test_assert_raises_regex__wrong_exception_raised(self):\n        try:\n            with assert_raises_regex(IndexError, \"test message\"):\n                raise KeyError(\"test message\")\n        except KeyError:\n            pass\n        except Exception as exc:\n            fail(str(exc) + \" was raised\")\n        else:\n            fail(\"no exception raised\")\n\n    def test_assert_raises_regex__wrong_error__default_message(self):\n        with _assert_raises_assertion(\"'wrong message' does not match 'test'\"):\n            with assert_raises_regex(KeyError, r\"test\"):\n                raise KeyError(\"wrong message\")\n\n    def test_assert_raises_regex__wrong_error__pattern_default_message(self):\n        with _assert_raises_assertion(\"'wrong message' does not match 'test'\"):\n            with assert_raises_regex(KeyError, re.compile(r\"test\")):\n                raise KeyError(\"wrong message\")\n\n    def test_assert_raises_regex__wrong_error__custom_message(self):\n        expected = (\n            \"'wrong message' does not match 'test';KeyError;KeyError;\"\n            \"'wrong message';test\"\n        )\n        with _assert_raises_assertion(expected):\n            msg_fmt = \"{msg};{exc_type.__name__};{exc_name};{text!r};{pattern}\"\n            with assert_raises_regex(KeyError, r\"test\", msg_fmt=msg_fmt):\n                raise KeyError(\"wrong message\")\n\n    # assert_raises_errno()\n\n    def test_assert_raises_errno__right_errno(self):\n        with assert_raises_errno(OSError, 20):\n            raise OSError(20, \"Test error\")\n\n    def test_assert_raises_errno__no_exception_raised__default_message(self):\n        with _assert_raises_assertion(\"OSError not raised\"):\n            with assert_raises_errno(OSError, 20):\n                pass\n\n    def test_assert_raises_errno__no_exception_raised__custom_message(self):\n        expected = \"OSError not raised;OSError;OSError;20;None\"\n        with _assert_raises_assertion(expected):\n            msg_fmt = (\n                \"{msg};{exc_type.__name__};{exc_name};{expected_errno};\"\n                \"{actual_errno}\"\n            )\n            with assert_raises_errno(OSError, 20, msg_fmt=msg_fmt):\n                pass\n\n    def test_assert_raises_errno__wrong_class_raised(self):\n        class RightClass(OSError):\n            pass\n\n        class WrongClass(OSError):\n            pass\n\n        try:\n            with assert_raises_errno(RightClass, 20):\n                raise WrongClass(20, \"Test error\")\n        except WrongClass:\n            pass\n        else:\n            raise AssertionError(\"WrongClass was not raised\")\n\n    def test_assert_raises_errno__wrong_errno__default_message(self):\n        with _assert_raises_assertion(\"wrong errno: 20 != 1\"):\n            with assert_raises_errno(OSError, 20):\n                raise OSError(1, \"Test error\")\n\n    def test_assert_raises_errno__wrong_errno__custom_message(self):\n        expected = \"wrong errno: 20 != 1;OSError;OSError;20;1\"\n        with _assert_raises_assertion(expected):\n            msg_fmt = (\n                \"{msg};{exc_type.__name__};{exc_name};{expected_errno};\"\n                \"{actual_errno}\"\n            )\n            with assert_raises_errno(OSError, 20, msg_fmt=msg_fmt):\n                raise OSError(1, \"Test error\")\n\n    # assert_succeeds()\n\n    def test_assert_succeeds__no_exception_raised(self):\n        with assert_succeeds(KeyError):\n            pass\n\n    def test_assert_succeeds__expected_exception__default_message(self):\n        with _assert_raises_assertion(\"KeyError was unexpectedly raised\"):\n            with assert_succeeds(KeyError):\n                raise KeyError()\n\n    def test_assert_succeeds__expected_exception__custom_message(self):\n        expected = (\n            \"KeyError was unexpectedly raised;KeyError;KeyError;test error\"\n        )\n        with _assert_raises_assertion(expected):\n            msg_fmt = (\n                \"{msg};{exc_type.__name__};{exc_name};{exception.args[0]}\"\n            )\n            with assert_succeeds(KeyError, msg_fmt=msg_fmt):\n                raise KeyError(\"test error\")\n\n    def test_assert_succeeds__unexpected_exception(self):\n        try:\n            with assert_succeeds(ValueError):\n                raise KeyError()\n        except KeyError:\n            pass\n        else:\n            raise AssertionError(\"KeyError was not raised\")\n\n    # assert_warns()\n\n    def test_assert_warns__default_message(self):\n        with assert_raises_regex(AssertionError, r\"^ImportWarning not issued\"):\n            with assert_warns(ImportWarning):\n                pass\n\n    def test_assert_warns__custom_message(self):\n        exception = \"ImportWarning not issued;ImportWarning;ImportWarning\"\n        with _assert_raises_assertion(exception):\n            msg_fmt = \"{msg};{exc_type.__name__};{exc_name}\"\n            with assert_warns(ImportWarning, msg_fmt=msg_fmt):\n                pass\n\n    def test_assert_warns__warned(self):\n        with assert_succeeds(AssertionError):\n            with assert_warns(FutureWarning):\n                warn(\"foo\", FutureWarning)\n\n    def test_assert_warns__not_warned(self):\n        with assert_raises(AssertionError):\n            with assert_warns(ImportWarning):\n                pass\n\n    def test_assert_warns__wrong_type(self):\n        with assert_raises(AssertionError):\n            with assert_warns(ImportWarning):\n                warn(\"foo\", UnicodeWarning)\n\n    def test_assert_warns__multiple_warnings(self):\n        with assert_succeeds(AssertionError):\n            with assert_warns(UserWarning):\n                warn(\"foo\", UnicodeWarning)\n                warn(\"bar\", UserWarning)\n                warn(\"baz\", FutureWarning)\n\n    def test_assert_warns__warning_handler_deinstalled_on_success(self):\n        with catch_warnings(record=True) as warnings:\n            with assert_warns(UserWarning):\n                warn(\"foo\", UserWarning)\n            assert warnings is not None\n            assert_equal(0, len(warnings))\n            warn(\"bar\", UserWarning)\n            assert_equal(1, len(warnings))\n\n    def test_assert_warns__warning_handler_deinstalled_on_failure(self):\n        with catch_warnings(record=True) as warnings:\n            try:\n                with assert_warns(UserWarning):\n                    pass\n            except AssertionError:\n                pass\n            assert warnings is not None\n            assert_equal(0, len(warnings))\n            warn(\"bar\", UserWarning)\n            assert_equal(1, len(warnings))\n\n    def test_assert_warns__add_test_called(self):\n        called = Box(False)\n\n        def extra_test(warning):\n            assert_is(warning.category, UserWarning)\n            called.value = True\n            return True\n\n        with assert_warns(UserWarning) as context:\n            context.add_test(extra_test)\n            warn(\"bar\", UserWarning)\n        assert_true(called.value, \"extra_test() was not called\")\n\n    def test_assert_warns__add_test_not_called(self):\n        called = Box(False)\n\n        def extra_test(_: Warning) -> bool:\n            called.value = True\n            return False\n\n        with assert_raises(AssertionError):\n            with assert_warns(UserWarning) as context:\n                context.add_test(extra_test)\n        assert_false(called.value, \"extra_test() was unexpectedly called\")\n\n    # assert_warns_regex()\n\n    def test_assert_warns_regex__warned(self):\n        with assert_succeeds(AssertionError):\n            with assert_warns_regex(FutureWarning, r\"fo+\"):\n                warn(\"foo\", FutureWarning)\n\n    def test_assert_warns_regex__warning_text_matches_in_the_middle(self):\n        with assert_succeeds(AssertionError):\n            with assert_warns_regex(FutureWarning, r\"o\"):\n                warn(\"foo\", FutureWarning)\n\n    def test_assert_warns_regex__not_warned(self):\n        with assert_raises(AssertionError):\n            with assert_warns_regex(UserWarning, r\"foo\"):\n                pass\n\n    def test_assert_warns_regex__wrong_type(self):\n        with assert_raises(AssertionError):\n            with assert_warns_regex(ImportWarning, r\"foo\"):\n                warn(\"foo\", UnicodeWarning)\n\n    def test_assert_warns_regex__wrong_message(self):\n        with assert_raises(AssertionError):\n            with assert_warns_regex(UnicodeWarning, r\"foo\"):\n                warn(\"bar\", UnicodeWarning)\n\n    def test_assert_warns_regex__multiple_warnings(self):\n        with assert_succeeds(AssertionError):\n            with assert_warns_regex(UserWarning, r\"bar2\"):\n                warn(\"foo\", UnicodeWarning)\n                warn(\"bar1\", UserWarning)\n                warn(\"bar2\", UserWarning)\n                warn(\"bar3\", UserWarning)\n                warn(\"baz\", FutureWarning)\n\n    def test_assert_warns_regex__warning_handler_deinstalled_on_success(self):\n        with catch_warnings(record=True) as warnings:\n            with assert_warns_regex(UserWarning, r\"foo\"):\n                warn(\"foo\", UserWarning)\n            assert warnings is not None\n            assert_equal(0, len(warnings))\n            warn(\"bar\", UserWarning)\n            assert_equal(1, len(warnings))\n\n    def test_assert_warns_regex__warning_handler_deinstalled_on_failure(self):\n        with catch_warnings(record=True) as warnings:\n            try:\n                with assert_warns_regex(UserWarning, r\"\"):\n                    pass\n            except AssertionError:\n                pass\n            assert warnings is not None\n            assert_equal(0, len(warnings))\n            warn(\"bar\", UserWarning)\n            assert_equal(1, len(warnings))\n\n    def test_assert_warns_regex__not_issued__default_message(self):\n        with _assert_raises_assertion(\n            \"no UserWarning matching 'foo.*bar' issued\"\n        ):\n            with assert_warns_regex(UserWarning, r\"foo.*bar\"):\n                pass\n\n    def test_assert_warns_regex__not_issued__custom_message(self):\n        expected = \"no ImportWarning matching 'abc' issued;ImportWarning;ImportWarning;abc\"\n        with _assert_raises_assertion(expected):\n            msg_fmt = \"{msg};{exc_type.__name__};{exc_name};{pattern}\"\n            with assert_warns_regex(ImportWarning, r\"abc\", msg_fmt=msg_fmt):\n                pass\n\n    def test_assert_warns_regex__wrong_message__default_message(self):\n        with _assert_raises_assertion(\n            \"no UserWarning matching 'foo.*bar' issued\"\n        ):\n            with assert_warns_regex(UserWarning, r\"foo.*bar\"):\n                pass\n\n    def test_assert_warns_regex__wrong_message__custom_message(self):\n        expected = (\n            \"no UserWarning matching 'foo.*bar' issued;UserWarning;\"\n            \"UserWarning;foo.*bar\"\n        )\n        with _assert_raises_assertion(expected):\n            msg_fmt = \"{msg};{exc_type.__name__};{exc_name};{pattern}\"\n            with assert_warns_regex(UserWarning, r\"foo.*bar\", msg_fmt=msg_fmt):\n                pass\n\n    # assert_json_subset()\n\n    def test_assert_json_subset__different_types(self):\n        with _assert_raises_assertion(\"element $ differs: {} != []\"):\n            assert_json_subset({}, [])\n\n    def test_assert_json_subset__empty_objects(self):\n        with assert_succeeds(AssertionError):\n            assert_json_subset({}, {})\n\n    def test_assert_json_subset__objects_equal(self):\n        with assert_succeeds(AssertionError):\n            assert_json_subset(\n                {\"foo\": 3, \"bar\": \"abc\"}, {\"bar\": \"abc\", \"foo\": 3}\n            )\n\n    def test_assert_json_subset__one_key_missing_from_first_object(self):\n        with assert_succeeds(AssertionError):\n            assert_json_subset({\"foo\": 3}, {\"foo\": 3, \"bar\": 3})\n\n    def test_assert_json_subset__one_key_missing_from_second_object(self):\n        with _assert_raises_assertion(\"element 'bar' missing from element $\"):\n            assert_json_subset({\"foo\": 3, \"bar\": 3}, {\"foo\": 3})\n\n    def test_assert_json_subset__multiple_keys_missing_from_second_object(\n        self,\n    ):\n        with _assert_raises_assertion(\n            \"elements 'bar', 'baz', and 'foo' missing from element $\"\n        ):\n            assert_json_subset({\"foo\": 3, \"bar\": 3, \"baz\": 3}, {})\n\n    def test_assert_json_subset__value_differs(self):\n        with _assert_raises_assertion(\"element $['foo'] differs: 3 != 4\"):\n            assert_json_subset({\"foo\": 3}, {\"foo\": 4})\n\n    def test_assert_json_subset__empty_lists(self):\n        with assert_succeeds(AssertionError):\n            assert_json_subset([], [])\n\n    def test_assert_json_subset__different_sized_lists(self):\n        with _assert_raises_assertion(\"JSON array $ differs in size: 2 != 1\"):\n            assert_json_subset([1, 2], [1])\n        with _assert_raises_assertion(\"JSON array $ differs in size: 1 != 2\"):\n            assert_json_subset([1], [1, 2])\n\n    def test_assert_json_subset__different_list_values(self):\n        with _assert_raises_assertion(\"element $[0] differs: {} != []\"):\n            assert_json_subset([{}], [[]])\n\n    def test_assert_json_subset__fundamental_types_differ(self):\n        with _assert_raises_assertion(\"element $[0] differs: 1 != 'foo'\"):\n            assert_json_subset([1], [\"foo\"])\n\n    def test_assert_json_subset__fundamental_values_differ(self):\n        with _assert_raises_assertion(\"element $[0] differs: 'bar' != 'foo'\"):\n            assert_json_subset([\"bar\"], [\"foo\"])\n\n    def test_assert_json_subset__none(self):\n        with assert_succeeds(AssertionError):\n            assert_json_subset([None], [None])\n        with _assert_raises_assertion(\"element $[0] differs: 42 != None\"):\n            assert_json_subset([42], [None])\n        with _assert_raises_assertion(\"element $[0] differs: None != 42\"):\n            assert_json_subset([None], [42])\n\n    def test_assert_json_subset__compare_int_and_float(self):\n        with assert_succeeds(AssertionError):\n            assert_json_subset([42], [42.0])\n            assert_json_subset([42.0], [42])\n\n    def test_assert_json_subset__unsupported_type(self):\n        msg = \"unsupported type <{} 'set'>\".format(self._type_string)\n        with assert_raises_regex(TypeError, msg):\n            assert_json_subset([set()], [set()])\n\n    def test_assert_json_subset__subtypes(self):\n        with assert_succeeds(AssertionError):\n            assert_json_subset(OrderedDict(), {})\n            assert_json_subset({}, OrderedDict())\n\n    def test_assert_json_subset__second_is_string(self):\n        with assert_succeeds(AssertionError):\n            assert_json_subset({}, \"{  }\")\n\n    def test_assert_json_subset__second_is_unsupported_json_string(self):\n        msg = \"second must decode to dict or list, not <{} 'int'>\".format(\n            self._type_string\n        )\n        with _assert_raises_assertion(msg):\n            assert_json_subset({}, \"42\")\n\n    def test_assert_json_subset__second_is_invalid_json_string(self):\n        with assert_raises(JSONDecodeError):\n            assert_json_subset({}, \",\")\n\n    def test_assert_json_subset__second_is_bytes(self):\n        with assert_succeeds(AssertionError):\n            assert_json_subset([\"föo\"], '[\"föo\"]'.encode(\"utf-8\"))\n\n    def test_assert_json_subset__second_is_latin1_bytes(self):\n        with assert_raises(UnicodeDecodeError):\n            assert_json_subset([\"föo\"], '[\"föo\"]'.encode(\"iso-8859-1\"))\n\n    def test_assert_json_subset__invalid_type(self):\n        with assert_raises_regex(\n            TypeError, \"second must be dict, list, str, or bytes\"\n        ):\n            assert_json_subset({}, 42)  # type: ignore[arg-type]\n\n    def test_assert_json_subset__element_name_not_str(self) -> None:\n        with assert_raises_regex(\n            TypeError,\n            \"12 is not a valid object member name\",\n        ):\n            assert_json_subset({12: 34}, \"{}\")\n\n    def test_assert_json_subset__presence_check(self) -> None:\n        with assert_succeeds(AssertionError):\n            assert_json_subset({\"foo\": Present}, {\"foo\": \"bar\"})\n        with assert_succeeds(AssertionError):\n            assert_json_subset({\"foo\": Present()}, {\"foo\": \"bar\"})\n        with assert_raises_regex(\n            AssertionError,\n            r\"element 'foo' missing from element \\$\",\n        ):\n            assert_json_subset({\"foo\": Present}, {})\n        with assert_raises_regex(\n            AssertionError,\n            r\"element 'foo' missing from element \\$\",\n        ):\n            assert_json_subset({\"foo\": Present()}, {})\n        with assert_succeeds(AssertionError):\n            assert_json_subset({\"foo\": Absent}, {})\n        with assert_succeeds(AssertionError):\n            assert_json_subset({\"foo\": Absent()}, {})\n        with assert_raises_regex(\n            AssertionError,\n            r\"spurious member 'foo' in object \\$\",\n        ):\n            assert_json_subset({\"foo\": Absent}, {\"foo\": \"bar\"})\n        with assert_raises_regex(\n            AssertionError,\n            r\"spurious member 'foo' in object \\$\",\n        ):\n            assert_json_subset({\"foo\": Absent()}, {\"foo\": \"bar\"})\n\n    def test_assert_json_subset__existence_check_old(self) -> None:\n        with catch_warnings():\n            simplefilter(\"ignore\")\n            with assert_succeeds(AssertionError):\n                assert_json_subset({Exists(\"foo\"): True}, {\"foo\": \"bar\"})\n            with assert_raises_regex(\n                AssertionError,\n                r\"element 'foo' missing from element \\$\",\n            ):\n                assert_json_subset({Exists(\"foo\"): True}, {})\n            with assert_succeeds(AssertionError):\n                assert_json_subset({Exists(\"foo\"): False}, {})\n            with assert_raises_regex(\n                AssertionError,\n                r\"spurious member 'foo' in object \\$\",\n            ):\n                assert_json_subset({Exists(\"foo\"): False}, {\"foo\": \"bar\"})\n"
    },
    {
      "path": "srittau_python-asserts/CHANGELOG.md",
      "content": "# Changelog for python-asserts\n\npython-asserts adheres to [semantic versioning](https://semver.org/).\n\n## UNRELEASED –\n\n## [0.13.1] – 2024-04-29\n\n### Fixed\n\nFixed Python 3.12 deprecation warnings.\n\n## [0.13.0] – 2024-03-13\n\n### Added\n\n- Add support for Python 3.12.\n- Add `Present` and `Absent` for absence checks in `assert_json_subset()`.\n\n### Removed\n\n- Drop support for Python 3.7.\n\n### Deprecated\n\n- Deprecate `Exists` in favor of `Present` and `Absent` in\n  `assert_json_subset()`.\n\n## [0.12.0]\n\n### Added\n\n- Add `assert_not_regex()`.\n\n### Changed\n\n- Modernize the type stubs.\n\n### Removed\n\n- Drop support for Python 3.6.\n\n## [0.11.1]\n\n### Added\n\n- `assert_json_subset()` can now check for the existence or non-existence\n  of object members using the new `Exists` helper.\n- Non-string (or `Exists`) object member names in the first argument to\n  `assert_json_subset()` now raise a `TypeError`.\n\n## [0.11.0]\n\n### Removed\n\n- Drop support for Python 2.7 and 3.5.\n\n## [0.10.0]\n\n### Added\n\n- `AssertRaisesContext` and `AssertWarnsContext` now return themselves\n  when `__enter__()` is called. By extension it now easier to call\n  `add_test()` with `assert_raises()` et al:\n\n```python\nwith assert_raises(KeyError) as context:\n    context.add_test(...)\n    ...\n```\n\n- Add `AssertRaisesContext.exc_val` property to access the caught\n  exception after leaving the context manager:\n\n```python\nwith assert_raises(KeyError) as context:\n    ...\nassert_equal(\"expected message\", str(context.exc_val))\n```\n\n### Removed\n\n- Drop support for Python 3.4.\n\n## [0.9.1]\n\n### Changed\n\n- `AssertRaisesContext` and sub-classes are now generic over the\n  exception type.\n\n## [0.9.0]\n\n### Added\n\n- Add `assert_json_subset()`.\n\n## [0.8.6]\n\n### Added\n\n- Add support for Python 3.7 (contributed by Frank Niessink).\n\n## [0.8.5]\n\n### Added\n\n- Add `assert_dict_equal()`.\n- Add `assert_dict_superset()`.\n\n### Changed\n\n- `assert_equal()`: Use `assert_dict_equal()` if applicable.\n\n## [0.8.4]\n\n### Changed\n\n- `fail()` is now marked with `NoReturn` in type stub.\n\n### Fixed\n\n- Improve type annotations for Python 2.\n\n## [0.8.3]\n\n### Fixed\n\n- Fix type signature of `AssertRaisesContext.__exit__()`.\n\n## [0.8.2]\n\n### Added\n\n- Add a py.typed file to signal that this package supports type hints.\n\n## [0.8.1]\n\n### Fixed\n\n- `assert_raises_regex()`: Handle exceptions without any message correctly.\n\n## [0.8.0]\n\n### Added\n\n- assert_count_equal(): Add `msg_fmt` argument.\n- Add AssertRaisesErrnoContext, AssertRaisesRegexContext, and\n  AssertWarnsRegexContext.\n\n### Changed\n\n- Replace `msg` argument with `msg_fmt` in all assertions (except `fail()`).\n  This allows you to customize error messages more easily than before, because\n  `format()` with appropriate keyword arguments is now called on these\n  strings. See the documentation of individual assertions for the supported\n  arguments.\n- Replace AssertRaisesContext.msg and AssertWarnsContext.msg with msg_fmt.\n- assert_almost_equal(), assert_not_almost_equal(): Place msg_fmt as third\n  argument.\n\n## [0.7.3]\n\n### Added\n\n- Add assert_not_almost_equal().\n\n### Changed\n\n- assert_almost_equal(): Raise ValueError if diff <= 0.\n\n### Fixed\n\n- assert_almost_equal() would never fail if a delta was supplied and the\n  second number was smaller than the first.\n- Use fail() instead of raise AssertionError in a few assertions.\n\n## [0.7.2]\n\n### Added\n\n- Add assert_warns() and assert_warns_regex().\n\n## [0.7.1]\n\n### Changed\n\n- Distribute a wheel.\n- asserts is now a package, instead of a module.\n\n## [0.7.0]\n\n### Added\n\n- Add a stub file.\n- Add assert_count_equal().\n\n## [0.6]\n\n### Added\n\n- Add assert_less(), assert_less_equal(), assert_greater(), and\n  assert_greater_equal().\n- Add assert_not_is_instance().\n\n### Changed\n\n- assert_datetime_about_now()/assert_datetime_about_now_utc(): Handle\n  comparison with None more gracefully.\n\n## [0.5.1]\n\n### Added\n\n- Add the LICENSE file to the distribution.\n\n## [0.5]\n\nInitial release.\n"
    },
    {
      "path": "srittau_python-asserts/mypy.ini",
      "content": "[mypy]\ncheck_untyped_defs = True\ndisallow_subclassing_any = True\ndisallow_untyped_decorators = True\nno_implicit_optional = True\nno_implicit_reexport = True\nwarn_redundant_casts = True\nwarn_return_any = True\nwarn_unused_configs = True\nwarn_unused_ignores = True\n"
    },
    {
      "path": "srittau_python-asserts/poetry.lock",
      "content": "# This file is automatically @generated by Poetry 1.8.2 and should not be changed by hand.\n\n[[package]]\nname = \"mypy\"\nversion = \"1.10.0\"\ndescription = \"Optional static typing for Python\"\noptional = false\npython-versions = \">=3.8\"\nfiles = [\n    {file = \"mypy-1.10.0-cp310-cp310-macosx_10_9_x86_64.whl\", hash = \"sha256:da1cbf08fb3b851ab3b9523a884c232774008267b1f83371ace57f412fe308c2\"},\n    {file = \"mypy-1.10.0-cp310-cp310-macosx_11_0_arm64.whl\", hash = \"sha256:12b6bfc1b1a66095ab413160a6e520e1dc076a28f3e22f7fb25ba3b000b4ef99\"},\n    {file = \"mypy-1.10.0-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl\", hash = \"sha256:9e36fb078cce9904c7989b9693e41cb9711e0600139ce3970c6ef814b6ebc2b2\"},\n    {file = \"mypy-1.10.0-cp310-cp310-musllinux_1_1_x86_64.whl\", hash = \"sha256:2b0695d605ddcd3eb2f736cd8b4e388288c21e7de85001e9f85df9187f2b50f9\"},\n    {file = \"mypy-1.10.0-cp310-cp310-win_amd64.whl\", hash = \"sha256:cd777b780312ddb135bceb9bc8722a73ec95e042f911cc279e2ec3c667076051\"},\n    {file = \"mypy-1.10.0-cp311-cp311-macosx_10_9_x86_64.whl\", hash = \"sha256:3be66771aa5c97602f382230165b856c231d1277c511c9a8dd058be4784472e1\"},\n    {file = \"mypy-1.10.0-cp311-cp311-macosx_11_0_arm64.whl\", hash = \"sha256:8b2cbaca148d0754a54d44121b5825ae71868c7592a53b7292eeb0f3fdae95ee\"},\n    {file = \"mypy-1.10.0-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl\", hash = \"sha256:1ec404a7cbe9fc0e92cb0e67f55ce0c025014e26d33e54d9e506a0f2d07fe5de\"},\n    {file = \"mypy-1.10.0-cp311-cp311-musllinux_1_1_x86_64.whl\", hash = \"sha256:e22e1527dc3d4aa94311d246b59e47f6455b8729f4968765ac1eacf9a4760bc7\"},\n    {file = \"mypy-1.10.0-cp311-cp311-win_amd64.whl\", hash = \"sha256:a87dbfa85971e8d59c9cc1fcf534efe664d8949e4c0b6b44e8ca548e746a8d53\"},\n    {file = \"mypy-1.10.0-cp312-cp312-macosx_10_9_x86_64.whl\", hash = \"sha256:a781f6ad4bab20eef8b65174a57e5203f4be627b46291f4589879bf4e257b97b\"},\n    {file = \"mypy-1.10.0-cp312-cp312-macosx_11_0_arm64.whl\", hash = \"sha256:b808e12113505b97d9023b0b5e0c0705a90571c6feefc6f215c1df9381256e30\"},\n    {file = \"mypy-1.10.0-cp312-cp312-manylinux_2_17_x86_64.manylinux2014_x86_64.whl\", hash = \"sha256:8f55583b12156c399dce2df7d16f8a5095291354f1e839c252ec6c0611e86e2e\"},\n    {file = \"mypy-1.10.0-cp312-cp312-musllinux_1_1_x86_64.whl\", hash = \"sha256:4cf18f9d0efa1b16478c4c129eabec36148032575391095f73cae2e722fcf9d5\"},\n    {file = \"mypy-1.10.0-cp312-cp312-win_amd64.whl\", hash = \"sha256:bc6ac273b23c6b82da3bb25f4136c4fd42665f17f2cd850771cb600bdd2ebeda\"},\n    {file = \"mypy-1.10.0-cp38-cp38-macosx_10_9_x86_64.whl\", hash = \"sha256:9fd50226364cd2737351c79807775136b0abe084433b55b2e29181a4c3c878c0\"},\n    {file = \"mypy-1.10.0-cp38-cp38-macosx_11_0_arm64.whl\", hash = \"sha256:f90cff89eea89273727d8783fef5d4a934be2fdca11b47def50cf5d311aff727\"},\n    {file = \"mypy-1.10.0-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl\", hash = \"sha256:fcfc70599efde5c67862a07a1aaf50e55bce629ace26bb19dc17cece5dd31ca4\"},\n    {file = \"mypy-1.10.0-cp38-cp38-musllinux_1_1_x86_64.whl\", hash = \"sha256:075cbf81f3e134eadaf247de187bd604748171d6b79736fa9b6c9685b4083061\"},\n    {file = \"mypy-1.10.0-cp38-cp38-win_amd64.whl\", hash = \"sha256:3f298531bca95ff615b6e9f2fc0333aae27fa48052903a0ac90215021cdcfa4f\"},\n    {file = \"mypy-1.10.0-cp39-cp39-macosx_10_9_x86_64.whl\", hash = \"sha256:fa7ef5244615a2523b56c034becde4e9e3f9b034854c93639adb667ec9ec2976\"},\n    {file = \"mypy-1.10.0-cp39-cp39-macosx_11_0_arm64.whl\", hash = \"sha256:3236a4c8f535a0631f85f5fcdffba71c7feeef76a6002fcba7c1a8e57c8be1ec\"},\n    {file = \"mypy-1.10.0-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl\", hash = \"sha256:4a2b5cdbb5dd35aa08ea9114436e0d79aceb2f38e32c21684dcf8e24e1e92821\"},\n    {file = \"mypy-1.10.0-cp39-cp39-musllinux_1_1_x86_64.whl\", hash = \"sha256:92f93b21c0fe73dc00abf91022234c79d793318b8a96faac147cd579c1671746\"},\n    {file = \"mypy-1.10.0-cp39-cp39-win_amd64.whl\", hash = \"sha256:28d0e038361b45f099cc086d9dd99c15ff14d0188f44ac883010e172ce86c38a\"},\n    {file = \"mypy-1.10.0-py3-none-any.whl\", hash = \"sha256:f8c083976eb530019175aabadb60921e73b4f45736760826aa1689dda8208aee\"},\n    {file = \"mypy-1.10.0.tar.gz\", hash = \"sha256:3d087fcbec056c4ee34974da493a826ce316947485cef3901f511848e687c131\"},\n]\n\n[package.dependencies]\nmypy-extensions = \">=1.0.0\"\ntomli = {version = \">=1.1.0\", markers = \"python_version < \\\"3.11\\\"\"}\ntyping-extensions = \">=4.1.0\"\n\n[package.extras]\ndmypy = [\"psutil (>=4.0)\"]\ninstall-types = [\"pip\"]\nmypyc = [\"setuptools (>=50)\"]\nreports = [\"lxml\"]\n\n[[package]]\nname = \"mypy-extensions\"\nversion = \"1.0.0\"\ndescription = \"Type system extensions for programs checked with the mypy type checker.\"\noptional = false\npython-versions = \">=3.5\"\nfiles = [\n    {file = \"mypy_extensions-1.0.0-py3-none-any.whl\", hash = \"sha256:4392f6c0eb8a5668a69e23d168ffa70f0be9ccfd32b5cc2d26a34ae5b844552d\"},\n    {file = \"mypy_extensions-1.0.0.tar.gz\", hash = \"sha256:75dbf8955dc00442a438fc4d0666508a9a97b6bd41aa2f0ffe9d2f2725af0782\"},\n]\n\n[[package]]\nname = \"pastel\"\nversion = \"0.2.1\"\ndescription = \"Bring colors to your terminal.\"\noptional = false\npython-versions = \">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*\"\nfiles = [\n    {file = \"pastel-0.2.1-py2.py3-none-any.whl\", hash = \"sha256:4349225fcdf6c2bb34d483e523475de5bb04a5c10ef711263452cb37d7dd4364\"},\n    {file = \"pastel-0.2.1.tar.gz\", hash = \"sha256:e6581ac04e973cac858828c6202c1e1e81fee1dc7de7683f3e1ffe0bfd8a573d\"},\n]\n\n[[package]]\nname = \"poethepoet\"\nversion = \"0.26.1\"\ndescription = \"A task runner that works well with poetry.\"\noptional = false\npython-versions = \">=3.8\"\nfiles = [\n    {file = \"poethepoet-0.26.1-py3-none-any.whl\", hash = \"sha256:aa43b443fec5d17d7e76771cccd484e5285805301721a74f059c483ad3276edd\"},\n    {file = \"poethepoet-0.26.1.tar.gz\", hash = \"sha256:aaad8541f6072617a60bcff2562d00779b58b353bd0f1847b06d8d0f2b6dc192\"},\n]\n\n[package.dependencies]\npastel = \">=0.2.1,<0.3.0\"\ntomli = \">=1.2.2\"\n\n[package.extras]\npoetry-plugin = [\"poetry (>=1.0,<2.0)\"]\n\n[[package]]\nname = \"ruff\"\nversion = \"0.4.8\"\ndescription = \"An extremely fast Python linter and code formatter, written in Rust.\"\noptional = false\npython-versions = \">=3.7\"\nfiles = [\n    {file = \"ruff-0.4.8-py3-none-macosx_10_12_x86_64.whl\", hash = \"sha256:7663a6d78f6adb0eab270fa9cf1ff2d28618ca3a652b60f2a234d92b9ec89066\"},\n    {file = \"ruff-0.4.8-py3-none-macosx_11_0_arm64.whl\", hash = \"sha256:eeceb78da8afb6de0ddada93112869852d04f1cd0f6b80fe464fd4e35c330913\"},\n    {file = \"ruff-0.4.8-py3-none-manylinux_2_17_aarch64.manylinux2014_aarch64.whl\", hash = \"sha256:aad360893e92486662ef3be0a339c5ca3c1b109e0134fcd37d534d4be9fb8de3\"},\n    {file = \"ruff-0.4.8-py3-none-manylinux_2_17_armv7l.manylinux2014_armv7l.whl\", hash = \"sha256:284c2e3f3396fb05f5f803c9fffb53ebbe09a3ebe7dda2929ed8d73ded736deb\"},\n    {file = \"ruff-0.4.8-py3-none-manylinux_2_17_i686.manylinux2014_i686.whl\", hash = \"sha256:a7354f921e3fbe04d2a62d46707e569f9315e1a613307f7311a935743c51a764\"},\n    {file = \"ruff-0.4.8-py3-none-manylinux_2_17_ppc64.manylinux2014_ppc64.whl\", hash = \"sha256:72584676164e15a68a15778fd1b17c28a519e7a0622161eb2debdcdabdc71883\"},\n    {file = \"ruff-0.4.8-py3-none-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl\", hash = \"sha256:9678d5c9b43315f323af2233a04d747409d1e3aa6789620083a82d1066a35199\"},\n    {file = \"ruff-0.4.8-py3-none-manylinux_2_17_s390x.manylinux2014_s390x.whl\", hash = \"sha256:704977a658131651a22b5ebeb28b717ef42ac6ee3b11e91dc87b633b5d83142b\"},\n    {file = \"ruff-0.4.8-py3-none-manylinux_2_17_x86_64.manylinux2014_x86_64.whl\", hash = \"sha256:d05f8d6f0c3cce5026cecd83b7a143dcad503045857bc49662f736437380ad45\"},\n    {file = \"ruff-0.4.8-py3-none-musllinux_1_2_aarch64.whl\", hash = \"sha256:6ea874950daca5697309d976c9afba830d3bf0ed66887481d6bca1673fc5b66a\"},\n    {file = \"ruff-0.4.8-py3-none-musllinux_1_2_armv7l.whl\", hash = \"sha256:fc95aac2943ddf360376be9aa3107c8cf9640083940a8c5bd824be692d2216dc\"},\n    {file = \"ruff-0.4.8-py3-none-musllinux_1_2_i686.whl\", hash = \"sha256:384154a1c3f4bf537bac69f33720957ee49ac8d484bfc91720cc94172026ceed\"},\n    {file = \"ruff-0.4.8-py3-none-musllinux_1_2_x86_64.whl\", hash = \"sha256:e9d5ce97cacc99878aa0d084c626a15cd21e6b3d53fd6f9112b7fc485918e1fa\"},\n    {file = \"ruff-0.4.8-py3-none-win32.whl\", hash = \"sha256:6d795d7639212c2dfd01991259460101c22aabf420d9b943f153ab9d9706e6a9\"},\n    {file = \"ruff-0.4.8-py3-none-win_amd64.whl\", hash = \"sha256:e14a3a095d07560a9d6769a72f781d73259655919d9b396c650fc98a8157555d\"},\n    {file = \"ruff-0.4.8-py3-none-win_arm64.whl\", hash = \"sha256:14019a06dbe29b608f6b7cbcec300e3170a8d86efaddb7b23405cb7f7dcaf780\"},\n    {file = \"ruff-0.4.8.tar.gz\", hash = \"sha256:16d717b1d57b2e2fd68bd0bf80fb43931b79d05a7131aa477d66fc40fbd86268\"},\n]\n\n[[package]]\nname = \"tomli\"\nversion = \"2.0.1\"\ndescription = \"A lil' TOML parser\"\noptional = false\npython-versions = \">=3.7\"\nfiles = [\n    {file = \"tomli-2.0.1-py3-none-any.whl\", hash = \"sha256:939de3e7a6161af0c887ef91b7d41a53e7c5a1ca976325f429cb46ea9bc30ecc\"},\n    {file = \"tomli-2.0.1.tar.gz\", hash = \"sha256:de526c12914f0c550d15924c62d72abc48d6fe7364aa87328337a31007fe8a4f\"},\n]\n\n[[package]]\nname = \"typing-extensions\"\nversion = \"4.12.2\"\ndescription = \"Backported and Experimental Type Hints for Python 3.8+\"\noptional = false\npython-versions = \">=3.8\"\nfiles = [\n    {file = \"typing_extensions-4.12.2-py3-none-any.whl\", hash = \"sha256:04e5ca0351e0f3f85c6853954072df659d0d13fac324d0072316b67d7794700d\"},\n    {file = \"typing_extensions-4.12.2.tar.gz\", hash = \"sha256:1a7ead55c7e559dd4dee8856e3a88b41225abfe1ce8df57b7c13915fe121ffb8\"},\n]\n\n[metadata]\nlock-version = \"2.0\"\npython-versions = \">=3.8.1\"\ncontent-hash = \"c010d5b718fed30569338ba631a7631751f5f6562c78491c16aa79927662e3f0\"\n"
    },
    {
      "path": "srittau_python-asserts/README.md",
      "content": "# Python Asserts\n\n[![License](https://img.shields.io/pypi/l/asserts.svg)](https://pypi.python.org/pypi/asserts/)\n[![PyPI - Python Version](https://img.shields.io/pypi/pyversions/asserts)](https://pypi.python.org/pypi/asserts/)\n[![GitHub](https://img.shields.io/github/release/srittau/python-asserts/all.svg)](https://github.com/srittau/python-asserts/releases/)\n[![pypi](https://img.shields.io/pypi/v/asserts.svg)](https://pypi.python.org/pypi/asserts/)\n[![GitHub Actions](https://img.shields.io/github/actions/workflow/status/srittau/python-asserts/test-and-lint.yml)](https://github.com/srittau/python-asserts/actions/workflows/test-and-lint.yml)\n\nStand-alone Assertions for Python\n\nThis package provides a few advantages over the assertions provided by\nunittest.TestCase:\n\n- Can be used stand-alone, for example:\n  - In test cases, not derived from TestCase.\n  - In fake and mock classes.\n  - In implementations as rich alternative to the assert statement.\n- PEP 8 compliance.\n- Custom stand-alone assertions can be written easily.\n- Arguably a better separation of concerns, since TestCase is responsible\n  for test running only, if assertion functions are used exclusively.\n\nThere are a few regressions compared to assertions from TestCase:\n\n- The default assertion class (`AssertionError`) can not be overwritten. This\n  is rarely a problem in practice.\n- asserts does not support the `addTypeEqualityFunc()` functionality.\n\nUsage:\n\n```python\n>>> from asserts import assert_true, assert_equal, assert_raises\n>>> my_var = 13\n>>> assert_equal(13, my_var)\n>>> assert_true(True, msg=\"custom failure message\")\n>>> with assert_raises(KeyError):\n...     raise KeyError()\n```\n\nFailure messages can be customized:\n\n```python\n>>> assert_equal(13, 14, msg_fmt=\"{got} is wrong, expected {expected}\")\nTraceback (most recent call last):\n  ...\nAssertionError: 14 is wrong, expected 13\n```\n"
    },
    {
      "path": "srittau_python-asserts/LICENSE",
      "content": "The MIT License (MIT)\n\nCopyright (c) 2014 Sebastian Rittau\n\nPermission is hereby granted, free of charge, to any person obtaining a copy of\nthis software and associated documentation files (the \"Software\"), to deal in\nthe Software without restriction, including without limitation the rights to\nuse, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of\nthe Software, and to permit persons to whom the Software is furnished to do so,\nsubject to the following conditions:\n\nThe above copyright notice and this permission notice shall be included in all\ncopies or substantial portions of the Software.\n\nTHE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\nIMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS\nFOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR\nCOPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER\nIN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN\nCONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.\n"
    },
    {
      "path": "srittau_python-asserts/asserts/__init__.py",
      "content": "\"\"\"\nRich Assertions.\n\nThis module contains several rich standard assertions that can be used in unit\ntests and in implementations. Users are encouraged to define their own\nassertions, possibly using assertions from this package as a basis.\n\n    >>> assert_equal(13, 13)\n    >>> assert_equal(13, 14)\n    Traceback (most recent call last):\n        ...\n    AssertionError: 13 != 14\n    >>> with assert_raises(KeyError):\n    ...     raise KeyError()\n    >>> with assert_raises(KeyError):\n    ...     pass\n    Traceback (most recent call last):\n        ...\n    AssertionError: KeyError not raised\n\n\"\"\"\n\nfrom __future__ import annotations\n\nimport re\nimport sys\nfrom datetime import datetime, timedelta, timezone\nfrom json import loads as json_loads\nfrom typing import Any, Callable, Set\nfrom warnings import WarningMessage, catch_warnings\n\nfrom typing_extensions import deprecated\n\n\ndef fail(msg=None):\n    \"\"\"Raise an AssertionError with the given message.\n\n    >>> fail(\"my message\")\n    Traceback (most recent call last):\n        ...\n    AssertionError: my message\n\n    \"\"\"\n    raise AssertionError(msg or \"assertion failure\")\n\n\ndef assert_true(expr, msg_fmt=\"{msg}\"):\n    \"\"\"Fail the test unless the expression is truthy.\n\n    >>> assert_true(\"Hello World!\")\n    >>> assert_true(\"\")\n    Traceback (most recent call last):\n        ...\n    AssertionError: '' is not truthy\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * expr - tested expression\n    \"\"\"\n\n    if not expr:\n        msg = \"{!r} is not truthy\".format(expr)\n        fail(msg_fmt.format(msg=msg, expr=expr))\n\n\ndef assert_false(expr, msg_fmt=\"{msg}\"):\n    \"\"\"Fail the test unless the expression is falsy.\n\n    >>> assert_false(\"\")\n    >>> assert_false(\"Hello World!\")\n    Traceback (most recent call last):\n        ...\n    AssertionError: 'Hello World!' is not falsy\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * expr - tested expression\n    \"\"\"\n\n    if expr:\n        msg = \"{!r} is not falsy\".format(expr)\n        fail(msg_fmt.format(msg=msg, expr=expr))\n\n\ndef assert_boolean_true(expr, msg_fmt=\"{msg}\"):\n    \"\"\"Fail the test unless the expression is the constant True.\n\n    >>> assert_boolean_true(True)\n    >>> assert_boolean_true(\"Hello World!\")\n    Traceback (most recent call last):\n        ...\n    AssertionError: 'Hello World!' is not True\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * expr - tested expression\n    \"\"\"\n\n    if expr is not True:\n        msg = \"{!r} is not True\".format(expr)\n        fail(msg_fmt.format(msg=msg, expr=expr))\n\n\ndef assert_boolean_false(expr, msg_fmt=\"{msg}\"):\n    \"\"\"Fail the test unless the expression is the constant False.\n\n    >>> assert_boolean_false(False)\n    >>> assert_boolean_false(0)\n    Traceback (most recent call last):\n        ...\n    AssertionError: 0 is not False\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * expr - tested expression\n    \"\"\"\n\n    if expr is not False:\n        msg = \"{!r} is not False\".format(expr)\n        fail(msg_fmt.format(msg=msg, expr=expr))\n\n\ndef assert_is_none(expr, msg_fmt=\"{msg}\"):\n    \"\"\"Fail if the expression is not None.\n\n    >>> assert_is_none(None)\n    >>> assert_is_none(False)\n    Traceback (most recent call last):\n        ...\n    AssertionError: False is not None\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * expr - tested expression\n    \"\"\"\n\n    if expr is not None:\n        msg = \"{!r} is not None\".format(expr)\n        fail(msg_fmt.format(msg=msg, expr=expr))\n\n\ndef assert_is_not_none(expr, msg_fmt=\"{msg}\"):\n    \"\"\"Fail if the expression is None.\n\n    >>> assert_is_not_none(0)\n    >>> assert_is_not_none(None)\n    Traceback (most recent call last):\n        ...\n    AssertionError: expression is None\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * expr - tested expression\n    \"\"\"\n    if expr is None:\n        msg = \"expression is None\"\n        fail(msg_fmt.format(msg=msg, expr=expr))\n\n\ndef assert_equal(first, second, msg_fmt=\"{msg}\"):\n    \"\"\"Fail unless first equals second, as determined by the '==' operator.\n\n    >>> assert_equal(5, 5.0)\n    >>> assert_equal(\"Hello World!\", \"Goodbye!\")\n    Traceback (most recent call last):\n        ...\n    AssertionError: 'Hello World!' != 'Goodbye!'\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * first - the first argument\n    * second - the second argument\n    \"\"\"\n\n    if isinstance(first, dict) and isinstance(second, dict):\n        assert_dict_equal(first, second, msg_fmt)\n    elif (not first) == second:\n        msg = \"{!r} != {!r}\".format(first, second)\n        fail(msg_fmt.format(msg=msg, first=first, second=second))\n\n\ndef assert_not_equal(first, second, msg_fmt=\"{msg}\"):\n    \"\"\"Fail if first equals second, as determined by the '==' operator.\n\n    >>> assert_not_equal(5, 8)\n    >>> assert_not_equal(-7, -7.0)\n    Traceback (most recent call last):\n        ...\n    AssertionError: -7 == -7.0\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * first - the first argument\n    * second - the second argument\n    \"\"\"\n\n    if first == second:\n        msg = \"{!r} == {!r}\".format(first, second)\n        fail(msg_fmt.format(msg=msg, first=first, second=second))\n\n\ndef assert_almost_equal(\n    first, second, msg_fmt=\"{msg}\", places=None, delta=None\n):\n    \"\"\"Fail if first and second are not equal after rounding.\n\n    By default, the difference between first and second is rounded to\n    7 decimal places. This can be configured with the places argument.\n    Alternatively, delta can be used to specify the maximum allowed\n    difference between first and second.\n\n    If first and second can not be rounded or both places and delta are\n    supplied, a TypeError is raised.\n\n    >>> assert_almost_equal(5, 5.00000001)\n    >>> assert_almost_equal(5, 5.001)\n    Traceback (most recent call last):\n        ...\n    AssertionError: 5 != 5.001 within 7 places\n    >>> assert_almost_equal(5, 5.001, places=2)\n    >>> assert_almost_equal(5, 5.001, delta=0.1)\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * first - the first argument\n    * second - the second argument\n    * places - number of places to compare or None\n    * delta - delta or None\n    \"\"\"\n\n    if delta is not None and places is not None:\n        raise TypeError(\"'places' and 'delta' are mutually exclusive\")\n    if delta is not None:\n        if delta <= 0:\n            raise ValueError(\"delta must be larger than 0\")\n        diff = abs(second - first)\n        success = diff < delta\n        detail_msg = \"with delta={}\".format(delta)\n    else:\n        if places is None:\n            places = 7\n        success = not round(second - first, places)\n        detail_msg = \"within {} places\".format(places)\n    if not success:\n        msg = \"{!r} != {!r} {}\".format(first, second, detail_msg)\n        fail(\n            msg_fmt.format(\n                msg=msg, first=first, second=second, places=places, delta=delta\n            )\n        )\n\n\ndef assert_not_almost_equal(\n    first, second, msg_fmt=\"{msg}\", places=None, delta=None\n):\n    \"\"\"Fail if first and second are equal after rounding.\n\n    By default, the difference between first and second is rounded to\n    7 decimal places. This can be configured with the places argument.\n    Alternatively, delta can be used to specify the maximum allowed\n    difference between first and second.\n\n    If first and second can not be rounded or both places and delta are\n    supplied, a TypeError is raised.\n\n    >>> assert_not_almost_equal(5, 5.001)\n    >>> assert_not_almost_equal(5, 5.00000001)\n    Traceback (most recent call last):\n        ...\n    AssertionError: 5 == 5.00000001 within 7 places\n    >>> assert_not_almost_equal(5, 5.001, places=2)\n    Traceback (most recent call last):\n        ...\n    AssertionError: 5 == 5.001 within 2 places\n    >>> assert_not_almost_equal(5, 5.001, delta=0.1)\n    Traceback (most recent call last):\n        ...\n    AssertionError: 5 == 5.001 with delta=0.1\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * first - the first argument\n    * second - the second argument\n    * places - number of places to compare or None\n    * delta - delta or None\n    \"\"\"\n\n    if delta is not None and places is not None:\n        raise TypeError(\"'places' and 'delta' are mutually exclusive\")\n    if delta is not None:\n        if delta <= 0:\n            raise ValueError(\"delta must be larger than 0\")\n        diff = abs(second - first)\n        success = diff >= delta\n        detail_msg = \"with delta={}\".format(delta)\n    else:\n        if places is None:\n            places = 7\n        success = bool(round(second - first, places))\n        detail_msg = \"within {} places\".format(places)\n    if not success:\n        msg = \"{!r} == {!r} {}\".format(first, second, detail_msg)\n        fail(\n            msg_fmt.format(\n                msg=msg, first=first, second=second, places=places, delta=delta\n            )\n        )\n\n\ndef assert_dict_equal(\n    first, second, key_msg_fmt=\"{msg}\", value_msg_fmt=\"{msg}\"\n):\n    \"\"\"Fail unless first dictionary equals second.\n\n    The dictionaries are considered equal, if they both contain the same\n    keys, and their respective values are also equal.\n\n    >>> assert_dict_equal({\"foo\": 5}, {\"foo\": 5})\n    >>> assert_dict_equal({\"foo\": 5}, {})\n    Traceback (most recent call last):\n        ...\n    AssertionError: key 'foo' missing from right dict\n\n    The following key_msg_fmt arguments are supported, if the keys do not\n    match:\n    * msg - the default error message\n    * first - the first dict\n    * second - the second dict\n    * missing_keys - list of keys missing from right\n    * extra_keys - list of keys missing from left\n\n    The following value_msg_fmt arguments are supported, if a value does not\n    match:\n    * msg - the default error message\n    * first - the first dict\n    * second - the second dict\n    * key - the key where the value does not match\n    * first_value - the value in the first dict\n    * second_value - the value in the second dict\n    \"\"\"\n    first_keys = set(first.keys())\n    second_keys = set(second.keys())\n    missing_keys = list(first_keys - second_keys)\n    extra_keys = list(second_keys - first_keys)\n    if missing_keys or extra_keys:\n        if missing_keys:\n            if len(missing_keys) == 1:\n                msg = \"key {!r} missing from right dict\".format(\n                    missing_keys[0]\n                )\n            else:\n                keys = \", \".join(sorted(repr(k) for k in missing_keys))\n                msg = \"keys {} missing from right dict\".format(keys)\n        else:\n            if len(extra_keys) == 1:\n                msg = \"extra key {!r} in right dict\".format(extra_keys[0])\n            else:\n                keys = \", \".join(sorted(repr(k) for k in extra_keys))\n                msg = \"extra keys {} in right dict\".format(keys)\n        if key_msg_fmt:\n            msg = key_msg_fmt.format(\n                msg=msg,\n                first=first,\n                second=second,\n                missing_keys=missing_keys,\n                extra_keys=extra_keys,\n            )\n        raise AssertionError(msg)\n    for key in first:\n        first_value = first[key]\n        second_value = second[key]\n        msg = \"key '{}' differs: {!r} != {!r}\".format(\n            key, first_value, second_value\n        )\n        if value_msg_fmt:\n            msg = value_msg_fmt.format(\n                msg=msg,\n                first=first,\n                second=second,\n                key=key,\n                first_value=first_value,\n                second_value=second_value,\n            )\n        msg = msg.replace(\"{\", \"{{\").replace(\"}\", \"}}\")\n        assert_equal(first_value, second_value, msg_fmt=msg)\n\n\ndef assert_dict_superset(\n    first, second, key_msg_fmt=\"{msg}\", value_msg_fmt=\"{msg}\"\n):\n    \"\"\"Fail unless second dictionary is a superset of the first.\n\n    The second dictionary must contain all keys of the first and their\n    values are equal (or a superset in case of dicts). But the second\n    dictionary can contain additional keys.\n\n    >>> assert_dict_superset({\"foo\": 5}, {\"foo\": 5, \"bar\": 10})\n    >>> assert_dict_superset({\"foo\": 5}, {})\n    Traceback (most recent call last):\n        ...\n    AssertionError: key 'foo' missing from right dict\n\n    The following key_msg_fmt arguments are supported, if the keys do not\n    match:\n    * msg - the default error message\n    * first - the first dict\n    * second - the second dict\n    * missing_keys - list of keys missing from right\n\n    The following value_msg_fmt arguments are supported, if a value does not\n    match:\n    * msg - the default error message\n    * first - the first dict\n    * second - the second dict\n    * key - the key where the value does not match\n    * first_value - the value in the first dict\n    * second_value - the value in the second dict\n    \"\"\"\n    first_keys = set(first.keys())\n    second_keys = set(second.keys())\n    missing_keys = list(first_keys - second_keys)\n    if missing_keys:\n        if len(missing_keys) == 1:\n            msg = \"key {!r} missing from right dict\".format(missing_keys[0])\n        else:\n            keys = \", \".join(sorted(repr(k) for k in missing_keys))\n            msg = \"keys {} missing from right dict\".format(keys)\n        if key_msg_fmt:\n            msg = key_msg_fmt.format(\n                msg=msg, first=first, second=second, missing_keys=missing_keys\n            )\n        raise AssertionError(msg)\n    for key in first:\n        first_value = first[key]\n        second_value = second[key]\n        msg = \"key '{}' differs: {!r} != {!r}\".format(\n            key, first_value, second_value\n        )\n        if value_msg_fmt:\n            msg = value_msg_fmt.format(\n                msg=msg,\n                first=first,\n                second=second,\n                key=key,\n                first_value=first_value,\n                second_value=second_value,\n            )\n        msg = msg.replace(\"{\", \"{{\").replace(\"}\", \"}}\")\n        assert_equal(first_value, second_value, msg_fmt=msg)\n\n\ndef assert_less(first, second, msg_fmt=\"{msg}\"):\n    \"\"\"Fail if first is not less than second.\n\n    >>> assert_less('bar', 'foo')\n    >>> assert_less(5, 5)\n    Traceback (most recent call last):\n        ...\n    AssertionError: 5 is not less than 5\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * first - the first argument\n    * second - the second argument\n    \"\"\"\n\n    if not first < second:\n        msg = \"{!r} is not less than {!r}\".format(first, second)\n        fail(msg_fmt.format(msg=msg, first=first, second=second))\n\n\ndef assert_less_equal(first, second, msg_fmt=\"{msg}\"):\n    \"\"\"Fail if first is not less than or equal to second.\n\n    >>> assert_less_equal('bar', 'foo')\n    >>> assert_less_equal(5, 5)\n    >>> assert_less_equal(6, 5)\n    Traceback (most recent call last):\n        ...\n    AssertionError: 6 is not less than or equal to 5\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * first - the first argument\n    * second - the second argument\n    \"\"\"\n\n    if not first <= second:\n        msg = \"{!r} is not less than or equal to {!r}\".format(first, second)\n        fail(msg_fmt.format(msg=msg, first=first, second=second))\n\n\ndef assert_greater(first, second, msg_fmt=\"{msg}\"):\n    \"\"\"Fail if first is not greater than second.\n\n    >>> assert_greater('foo', 'bar')\n    >>> assert_greater(5, 5)\n    Traceback (most recent call last):\n        ...\n    AssertionError: 5 is not greater than 5\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * first - the first argument\n    * second - the second argument\n    \"\"\"\n\n    if not first > second:\n        msg = \"{!r} is not greater than {!r}\".format(first, second)\n        fail(msg_fmt.format(msg=msg, first=first, second=second))\n\n\ndef assert_greater_equal(first, second, msg_fmt=\"{msg}\"):\n    \"\"\"Fail if first is not greater than or equal to second.\n\n    >>> assert_greater_equal('foo', 'bar')\n    >>> assert_greater_equal(5, 5)\n    >>> assert_greater_equal(5, 6)\n    Traceback (most recent call last):\n        ...\n    AssertionError: 5 is not greater than or equal to 6\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * first - the first argument\n    * second - the second argument\n    \"\"\"\n\n    if not first >= second:\n        msg = \"{!r} is not greater than or equal to {!r}\".format(first, second)\n        fail(msg_fmt.format(msg=msg, first=first, second=second))\n\n\ndef assert_regex(text, regex, msg_fmt=\"{msg}\"):\n    \"\"\"Fail if text does not match the regular expression.\n\n    regex can be either a regular expression string or a compiled regular\n    expression object.\n\n    >>> assert_regex(\"Hello World!\", r\"llo.*rld!$\")\n    >>> assert_regex(\"Hello World!\", r\"\\\\d\")\n    Traceback (most recent call last):\n        ...\n    AssertionError: 'Hello World!' does not match '\\\\\\\\d'\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * text - text that is matched\n    * pattern - regular expression pattern as string\n    \"\"\"\n\n    compiled = re.compile(regex)\n    if not compiled.search(text):\n        msg = \"{!r} does not match {!r}\".format(text, compiled.pattern)\n        fail(msg_fmt.format(msg=msg, text=text, pattern=compiled.pattern))\n\n\ndef assert_not_regex(text, regex, msg_fmt=\"{msg}\"):\n    \"\"\"Fail if text does match the regular expression.\n\n    regex can be either a regular expression string or a compiled regular\n    expression object.\n\n    >>> assert_regex(\"Hello World!\", r\"llo.*rld!$\")\n    >>> assert_regex(\"Hello World!\", r\"\\\\d\")\n    Traceback (most recent call last):\n        ...\n    AssertionError: 'Hello World!' does not match '\\\\\\\\d'\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * text - text that is matched\n    * pattern - regular expression pattern as string\n    \"\"\"\n\n    compiled = re.compile(regex)\n    if compiled.search(text):\n        msg = \"{!r} matches {!r}\".format(text, compiled.pattern)\n        fail(msg_fmt.format(msg=msg, text=text, pattern=compiled.pattern))\n\n\ndef assert_is(first, second, msg_fmt=\"{msg}\"):\n    \"\"\"Fail if first and second do not refer to the same object.\n\n    >>> list1 = [5, \"foo\"]\n    >>> list2 = [5, \"foo\"]\n    >>> assert_is(list1, list1)\n    >>> assert_is(list1, list2)\n    Traceback (most recent call last):\n        ...\n    AssertionError: [5, 'foo'] is not [5, 'foo']\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * first - the first argument\n    * second - the second argument\n    \"\"\"\n\n    if first is not second:\n        msg = \"{!r} is not {!r}\".format(first, second)\n        fail(msg_fmt.format(msg=msg, first=first, second=second))\n\n\ndef assert_is_not(first, second, msg_fmt=\"{msg}\"):\n    \"\"\"Fail if first and second refer to the same object.\n\n    >>> list1 = [5, \"foo\"]\n    >>> list2 = [5, \"foo\"]\n    >>> assert_is_not(list1, list2)\n    >>> assert_is_not(list1, list1)\n    Traceback (most recent call last):\n        ...\n    AssertionError: both arguments refer to [5, 'foo']\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * first - the first argument\n    * second - the second argument\n    \"\"\"\n\n    if first is second:\n        msg = \"both arguments refer to {!r}\".format(first)\n        fail(msg_fmt.format(msg=msg, first=first, second=second))\n\n\ndef assert_in(first, second, msg_fmt=\"{msg}\"):\n    \"\"\"Fail if first is not in collection second.\n\n    >>> assert_in(\"foo\", [4, \"foo\", {}])\n    >>> assert_in(\"bar\", [4, \"foo\", {}])\n    Traceback (most recent call last):\n        ...\n    AssertionError: 'bar' not in [4, 'foo', {}]\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * first - the element looked for\n    * second - the container looked in\n    \"\"\"\n\n    if first not in second:\n        msg = \"{!r} not in {!r}\".format(first, second)\n        fail(msg_fmt.format(msg=msg, first=first, second=second))\n\n\ndef assert_not_in(first, second, msg_fmt=\"{msg}\"):\n    \"\"\"Fail if first is in a collection second.\n\n    >>> assert_not_in(\"bar\", [4, \"foo\", {}])\n    >>> assert_not_in(\"foo\", [4, \"foo\", {}])\n    Traceback (most recent call last):\n        ...\n    AssertionError: 'foo' is in [4, 'foo', {}]\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * first - the element looked for\n    * second - the container looked in\n    \"\"\"\n    if first in second:\n        msg = \"{!r} is in {!r}\".format(first, second)\n        fail(msg_fmt.format(msg=msg, first=first, second=second))\n\n\ndef assert_count_equal(sequence1, sequence2, msg_fmt=\"{msg}\"):\n    \"\"\"Compare the items of two sequences, ignoring order.\n\n    >>> assert_count_equal([1, 2], {2, 1})\n\n    Items missing in either sequence will be listed:\n\n    >>> assert_count_equal([\"a\", \"b\", \"c\"], [\"a\", \"d\"])\n    Traceback (most recent call last):\n        ...\n    AssertionError: missing from sequence 1: 'd'; missing from sequence 2: 'b', 'c'\n\n    Items are counted in each sequence. This makes it useful to detect\n    duplicates:\n\n    >>> assert_count_equal({\"a\", \"b\"}, [\"a\", \"a\", \"b\"])\n    Traceback (most recent call last):\n        ...\n    AssertionError: missing from sequence 1: 'a'\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * first - first sequence\n    * second - second sequence\n    \"\"\"\n\n    def compare():\n        missing1 = list(sequence2)\n        missing2 = []\n        for item in sequence1:\n            try:\n                missing1.remove(item)\n            except ValueError:\n                missing2.append(item)\n        return missing1, missing2\n\n    def build_message():\n        msg = \"\"\n        if missing_from_1:\n            msg += \"missing from sequence 1: \" + \", \".join(\n                repr(i) for i in missing_from_1\n            )\n        if missing_from_1 and missing_from_2:\n            msg += \"; \"\n        if missing_from_2:\n            msg += \"missing from sequence 2: \" + \", \".join(\n                repr(i) for i in missing_from_2\n            )\n        return msg\n\n    missing_from_1, missing_from_2 = compare()\n    if missing_from_1 or missing_from_2:\n        fail(\n            msg_fmt.format(\n                msg=build_message(), first=sequence1, second=sequence2\n            )\n        )\n\n\ndef assert_between(lower_bound, upper_bound, expr, msg_fmt=\"{msg}\"):\n    \"\"\"Fail if an expression is not between certain bounds (inclusive).\n\n    >>> assert_between(5, 15, 5)\n    >>> assert_between(5, 15, 15)\n    >>> assert_between(5, 15, 4.9)\n    Traceback (most recent call last):\n        ...\n    AssertionError: 4.9 is not between 5 and 15\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * lower - lower bound\n    * upper - upper bound\n    * expr - tested expression\n    \"\"\"\n\n    if not lower_bound <= expr <= upper_bound:\n        msg = \"{!r} is not between {} and {}\".format(\n            expr, lower_bound, upper_bound\n        )\n        fail(\n            msg_fmt.format(\n                msg=msg, lower=lower_bound, upper=upper_bound, expr=expr\n            )\n        )\n\n\ndef assert_is_instance(obj, cls, msg_fmt=\"{msg}\"):\n    \"\"\"Fail if an object is not an instance of a class or tuple of classes.\n\n    >>> assert_is_instance(5, int)\n    >>> assert_is_instance('foo', (str, bytes))\n    >>> assert_is_instance(5, str)\n    Traceback (most recent call last):\n        ...\n    AssertionError: 5 is an instance of <class 'int'>, expected <class 'str'>\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * obj - object to test\n    * types - tuple of types tested against\n    \"\"\"\n    if not isinstance(obj, cls):\n        msg = \"{!r} is an instance of {!r}, expected {!r}\".format(\n            obj, obj.__class__, cls\n        )\n        types = cls if isinstance(cls, tuple) else (cls,)\n        fail(msg_fmt.format(msg=msg, obj=obj, types=types))\n\n\ndef assert_not_is_instance(obj, cls, msg_fmt=\"{msg}\"):\n    \"\"\"Fail if an object is an instance of a class or tuple of classes.\n\n    >>> assert_not_is_instance(5, str)\n    >>> assert_not_is_instance(5, (str, bytes))\n    >>> assert_not_is_instance('foo', str)\n    Traceback (most recent call last):\n        ...\n    AssertionError: 'foo' is an instance of <class 'str'>\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * obj - object to test\n    * types - tuple of types tested against\n    \"\"\"\n    if isinstance(obj, cls):\n        msg = \"{!r} is an instance of {!r}\".format(obj, obj.__class__)\n        types = cls if isinstance(cls, tuple) else (cls,)\n        fail(msg_fmt.format(msg=msg, obj=obj, types=types))\n\n\ndef assert_has_attr(obj, attribute, msg_fmt=\"{msg}\"):\n    \"\"\"Fail is an object does not have an attribute.\n\n    >>> assert_has_attr([], \"index\")\n    >>> assert_has_attr([], \"i_do_not_have_this\")\n    Traceback (most recent call last):\n        ...\n    AssertionError: [] does not have attribute 'i_do_not_have_this'\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * obj - object to test\n    * attribute - name of the attribute to check\n    \"\"\"\n\n    if not hasattr(obj, attribute):\n        msg = \"{!r} does not have attribute '{}'\".format(obj, attribute)\n        fail(msg_fmt.format(msg=msg, obj=obj, attribute=attribute))\n\n\n_EPSILON_SECONDS = 5\n\n\ndef assert_datetime_about_now(actual, msg_fmt=\"{msg}\"):\n    \"\"\"Fail if a datetime object is not within 5 seconds of the local time.\n\n    >>> assert_datetime_about_now(datetime.now())\n    >>> assert_datetime_about_now(datetime(1900, 1, 1, 12, 0, 0))\n    Traceback (most recent call last):\n        ...\n    AssertionError: datetime.datetime(1900, 1, 1, 12, 0) is not close to current date/time\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * actual - datetime object to check\n    * now - current datetime that was tested against\n    \"\"\"\n\n    now = datetime.now()\n    if actual is None:\n        msg = \"None is not a valid date/time\"\n        fail(msg_fmt.format(msg=msg, actual=actual, now=now))\n    lower_bound = now - timedelta(seconds=_EPSILON_SECONDS)\n    upper_bound = now + timedelta(seconds=_EPSILON_SECONDS)\n    if not lower_bound <= actual <= upper_bound:\n        msg = \"{!r} is not close to current date/time\".format(actual)\n        fail(msg_fmt.format(msg=msg, actual=actual, now=now))\n\n\ndef assert_datetime_about_now_utc(actual, msg_fmt=\"{msg}\"):\n    \"\"\"Fail if a datetime object is not within 5 seconds of UTC.\n\n    >>> assert_datetime_about_now_utc(datetime.now(timezone.utc).replace(tzinfo=None))\n    >>> assert_datetime_about_now_utc(datetime(1900, 1, 1, 12, 0, 0))\n    Traceback (most recent call last):\n        ...\n    AssertionError: datetime.datetime(1900, 1, 1, 12, 0) is not close to current UTC date/time\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * actual - datetime object to check\n    * now - current datetime that was tested against\n    \"\"\"\n\n    now = datetime.now(timezone.utc).replace(tzinfo=None)\n    if actual is None:\n        msg = \"None is not a valid date/time\"\n        fail(msg_fmt.format(msg=msg, actual=actual, now=now))\n    lower_bound = now - timedelta(seconds=_EPSILON_SECONDS)\n    upper_bound = now + timedelta(seconds=_EPSILON_SECONDS)\n    if not lower_bound <= actual <= upper_bound:\n        msg = \"{!r} is not close to current UTC date/time\".format(actual)\n        fail(msg_fmt.format(msg=msg, actual=actual, now=now))\n\n\nclass AssertRaisesContext:\n    \"\"\"A context manager to test for exceptions with certain properties.\n\n    When the context is left and no exception has been raised, an\n    AssertionError will be raised:\n\n        >>> context = AssertRaisesContext(TypeError)\n        >>> with context:\n        ...    pass\n        Traceback (most recent call last):\n            ...\n        AssertionError: TypeError not raised\n\n    If an exception that is not a sub-class of the exception class provided\n    to the constructor is raised, it will be passed on:\n\n        >>> with context:\n        ...    raise ValueError(\"Wrong Class\")\n        Traceback (most recent call last):\n            ...\n        ValueError: Wrong Class\n\n    If the exception has the right class, any additional tests that have been\n    configured on the context, will be called:\n\n        >>> def test(exc):\n        ...     assert_equal(\"Hello World!\", str(exc))\n        >>> context.add_test(test)\n        >>> with context:\n        ...     raise TypeError(\"Wrong Message\")\n        Traceback (most recent call last):\n            ...\n        AssertionError: 'Hello World!' != 'Wrong Message'\n\n    \"\"\"\n\n    def __init__(self, exception, msg_fmt=\"{msg}\"):\n        self.exception = exception\n        self.msg_fmt = msg_fmt\n        self._exc_type = exception\n        self._exc_val = None\n        self._exception_name = getattr(exception, \"__name__\", str(exception))\n        self._tests: list[Callable[[Any], object]] = []\n\n    def __enter__(self):\n        return self\n\n    def __exit__(self, exc_type, exc_val, exc_tb):\n        if not exc_type or not exc_val:\n            msg = \"{} not raised\".format(self._exception_name)\n            fail(self.format_message(msg))\n        self._exc_val = exc_val\n        if not issubclass(exc_type, self.exception):\n            return False\n        for test in self._tests:\n            test(exc_val)\n        return True\n\n    def format_message(self, default_msg):\n        return self.msg_fmt.format(\n            msg=default_msg,\n            exc_type=self._exc_type,\n            exc_name=self._exception_name,\n        )\n\n    def add_test(self, cb: Callable[[Any], object]) -> None:\n        \"\"\"Add a test callback.\n\n        This callback is called after determining that the right exception\n        class was raised. The callback will get the raised exception as only\n        argument.\n\n        \"\"\"\n        self._tests.append(cb)\n\n    @property\n    def exc_val(self):\n        if self._exc_val is None:\n            raise RuntimeError(\"must be called after leaving the context\")\n        return self._exc_val\n\n\nclass AssertRaisesRegexContext(AssertRaisesContext):\n    \"\"\"A context manager to test for exceptions and their messages.\"\"\"\n\n    def __init__(self, exception, pattern, msg_fmt=\"{msg}\"):\n        super(AssertRaisesRegexContext, self).__init__(exception, msg_fmt)\n        self.pattern = pattern\n\n    def format_message(self, default_msg):\n        return self.msg_fmt.format(\n            msg=default_msg,\n            exc_type=self._exc_type,\n            exc_name=self._exception_name,\n            pattern=self.pattern,\n            text=\"\",\n        )\n\n\nclass AssertRaisesErrnoContext(AssertRaisesContext):\n    \"\"\"A context manager to test for exceptions with errnos.\"\"\"\n\n    def __init__(self, exception, expected_errno, msg_fmt=\"{msg}\"):\n        super(AssertRaisesErrnoContext, self).__init__(exception, msg_fmt)\n        self.expected_errno = expected_errno\n\n    def format_message(self, default_msg):\n        return self.msg_fmt.format(\n            msg=default_msg,\n            exc_type=self._exc_type,\n            exc_name=self._exception_name,\n            expected_errno=self.expected_errno,\n            actual_errno=None,\n        )\n\n\ndef assert_raises(exception, msg_fmt=\"{msg}\"):\n    \"\"\"Fail unless a specific exception is raised inside the context.\n\n    If a different type of exception is raised, it will not be caught.\n\n    >>> with assert_raises(TypeError):\n    ...     raise TypeError()\n    ...\n    >>> with assert_raises(TypeError):\n    ...     pass\n    ...\n    Traceback (most recent call last):\n        ...\n    AssertionError: TypeError not raised\n    >>> with assert_raises(TypeError):\n    ...     raise ValueError(\"wrong error\")\n    ...\n    Traceback (most recent call last):\n        ...\n    ValueError: wrong error\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * exc_type - exception type that is expected\n    * exc_name - expected exception type name\n    \"\"\"\n\n    return AssertRaisesContext(exception, msg_fmt)\n\n\ndef assert_raises_regex(exception, regex, msg_fmt=\"{msg}\"):\n    \"\"\"Fail unless an exception with a message that matches a regular\n     expression is raised within the context.\n\n    The regular expression can be a regular expression string or object.\n\n    >>> with assert_raises_regex(ValueError, r\"\\\\d+\"):\n    ...     raise ValueError(\"Error #42\")\n    ...\n    >>> with assert_raises_regex(ValueError, r\"\\\\d+\"):\n    ...     raise ValueError(\"Generic Error\")\n    ...\n    Traceback (most recent call last):\n        ...\n    AssertionError: 'Generic Error' does not match '\\\\\\\\d+'\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * exc_type - exception type that is expected\n    * exc_name - expected exception type name\n    * text - actual error text\n    * pattern - expected error message as regular expression string\n    \"\"\"\n\n    def test(exc):\n        compiled = re.compile(regex)\n        if not exc.args:\n            msg = \"{} without message\".format(exception.__name__)\n            fail(\n                msg_fmt.format(\n                    msg=msg,\n                    text=None,\n                    pattern=compiled.pattern,\n                    exc_type=exception,\n                    exc_name=exception.__name__,\n                )\n            )\n        text = exc.args[0]\n        if not compiled.search(text):\n            msg = \"{!r} does not match {!r}\".format(text, compiled.pattern)\n            fail(\n                msg_fmt.format(\n                    msg=msg,\n                    text=text,\n                    pattern=compiled.pattern,\n                    exc_type=exception,\n                    exc_name=exception.__name__,\n                )\n            )\n\n    context = AssertRaisesRegexContext(exception, regex, msg_fmt)\n    context.add_test(test)\n    return context\n\n\ndef assert_raises_errno(exception, errno, msg_fmt=\"{msg}\"):\n    \"\"\"Fail unless an exception with a specific errno is raised with the\n     context.\n\n    >>> with assert_raises_errno(OSError, 42):\n    ...     raise OSError(42, \"OS Error\")\n    ...\n    >>> with assert_raises_errno(OSError, 44):\n    ...     raise OSError(17, \"OS Error\")\n    ...\n    Traceback (most recent call last):\n        ...\n    AssertionError: wrong errno: 44 != 17\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * exc_type - exception type that is expected\n    * exc_name - expected exception type name\n    * expected_errno -\n    * actual_errno - raised errno or None if no matching exception was raised\n    \"\"\"\n\n    def check_errno(exc):\n        if errno != exc.errno:\n            msg = \"wrong errno: {!r} != {!r}\".format(errno, exc.errno)\n            fail(\n                msg_fmt.format(\n                    msg=msg,\n                    exc_type=exception,\n                    exc_name=exception.__name__,\n                    expected_errno=errno,\n                    actual_errno=exc.errno,\n                )\n            )\n\n    context = AssertRaisesErrnoContext(exception, errno, msg_fmt)\n    context.add_test(check_errno)\n    return context\n\n\ndef assert_succeeds(exception, msg_fmt=\"{msg}\"):\n    \"\"\"Fail if a specific exception is raised within the context.\n\n    This assertion should be used for cases, where successfully running a\n    function signals a successful test, and raising the exception of a\n    certain type signals a test failure. All other raised exceptions are\n    passed on and will usually still result in a test error. This can be\n    used to signal the intent of a block.\n\n    >>> l = [\"foo\", \"bar\"]\n    >>> with assert_succeeds(ValueError):\n    ...     i = l.index(\"foo\")\n    ...\n    >>> with assert_succeeds(ValueError):\n    ...     raise ValueError()\n    ...\n    Traceback (most recent call last):\n        ...\n    AssertionError: ValueError was unexpectedly raised\n    >>> with assert_succeeds(ValueError):\n    ...     raise TypeError(\"Wrong Error\")\n    ...\n    Traceback (most recent call last):\n        ...\n    TypeError: Wrong Error\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * exc_type - exception type\n    * exc_name - exception type name\n    * exception - exception that was raised\n    \"\"\"\n\n    class _AssertSucceeds(object):\n        def __enter__(self):\n            pass\n\n        def __exit__(self, exc_type, exc_val, exc_tb):\n            if exc_type and issubclass(exc_type, exception):\n                msg = exception.__name__ + \" was unexpectedly raised\"\n                fail(\n                    msg_fmt.format(\n                        msg=msg,\n                        exc_type=exception,\n                        exc_name=exception.__name__,\n                        exception=exc_val,\n                    )\n                )\n\n    return _AssertSucceeds()\n\n\nclass AssertWarnsContext(object):\n    \"\"\"A context manager to test for warnings with certain properties.\n\n    When the context is left and the expected warning has not been raised, an\n    AssertionError will be raised:\n\n        >>> context = AssertWarnsContext(DeprecationWarning)\n        >>> with context:\n        ...    pass\n        Traceback (most recent call last):\n            ...\n        AssertionError: DeprecationWarning not issued\n\n    If the warning has the right class, any additional tests that have been\n    configured on the context, will be called:\n\n        >>> from warnings import warn\n        >>> def test(warning):\n        ...     return False\n        >>> context.add_test(test)\n        >>> with context:\n        ...     warn(\"Wrong Message\", DeprecationWarning)\n        Traceback (most recent call last):\n            ...\n        AssertionError: DeprecationWarning not issued\n\n    \"\"\"\n\n    def __init__(self, warning_class, msg_fmt=\"{msg}\"):\n        self._warning_class = warning_class\n        self._msg_fmt = msg_fmt\n        self._warning_context: catch_warnings[list[WarningMessage]] | None = (\n            None\n        )\n        self._warnings = []\n        self._tests: list[Callable[[Warning], bool]] = []\n\n    def __enter__(self):\n        self._warning_context = catch_warnings(record=True)\n        self._warnings = self._warning_context.__enter__()\n        return self\n\n    def __exit__(self, exc_type, exc_val, exc_tb):\n        assert self._warning_context is not None\n        self._warning_context.__exit__(exc_type, exc_val, exc_tb)\n        if not any(self._is_expected_warning(w) for w in self._warnings):\n            fail(self.format_message())\n\n    def format_message(self):\n        msg = \"{} not issued\".format(self._warning_class.__name__)\n        return self._msg_fmt.format(\n            msg=msg,\n            exc_type=self._warning_class,\n            exc_name=self._warning_class.__name__,\n        )\n\n    def _is_expected_warning(self, warning) -> bool:\n        if not issubclass(warning.category, self._warning_class):\n            return False\n        return all(test(warning) for test in self._tests)\n\n    def add_test(self, cb: Callable[[Warning], bool]) -> None:\n        \"\"\"Add a test callback.\n\n        This callback is called after determining that the right warning\n        class was issued. The callback will get the issued warning as only\n        argument and must return a boolean value.\n\n        \"\"\"\n        self._tests.append(cb)\n\n\nclass AssertWarnsRegexContext(AssertWarnsContext):\n    \"\"\"A context manager to test for warnings and their messages.\"\"\"\n\n    def __init__(self, warning_class, pattern, msg_fmt=\"{msg}\"):\n        super(AssertWarnsRegexContext, self).__init__(warning_class, msg_fmt)\n        self.pattern = pattern\n\n    def format_message(self):\n        msg = \"no {} matching {} issued\".format(\n            self._warning_class.__name__, repr(self.pattern)\n        )\n        return self._msg_fmt.format(\n            msg=msg,\n            exc_type=self._warning_class,\n            exc_name=self._warning_class.__name__,\n            pattern=self.pattern,\n        )\n\n\ndef assert_warns(warning_type, msg_fmt=\"{msg}\"):\n    \"\"\"Fail unless a specific warning is issued inside the context.\n\n    If a different type of warning is issued, it will not be caught.\n\n    >>> from warnings import warn\n    >>> with assert_warns(UserWarning):\n    ...     warn(\"warning message\", UserWarning)\n    ...\n    >>> with assert_warns(UserWarning):\n    ...     pass\n    ...\n    Traceback (most recent call last):\n        ...\n    AssertionError: UserWarning not issued\n    >>> with assert_warns(UserWarning):\n    ...     warn(\"warning message\", UnicodeWarning)\n    ...\n    Traceback (most recent call last):\n        ...\n    AssertionError: UserWarning not issued\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * exc_type - exception type\n    * exc_name - exception type name\n    \"\"\"\n    return AssertWarnsContext(warning_type, msg_fmt)\n\n\ndef assert_warns_regex(warning_type, regex, msg_fmt=\"{msg}\"):\n    \"\"\"Fail unless a warning with a message is issued inside the context.\n\n    The message can be a regular expression string or object.\n\n    >>> from warnings import warn\n    >>> with assert_warns_regex(UserWarning, r\"#\\\\d+\"):\n    ...     warn(\"Error #42\", UserWarning)\n    ...\n    >>> with assert_warns_regex(UserWarning, r\"Expected Error\"):\n    ...     warn(\"Generic Error\", UserWarning)\n    ...\n    Traceback (most recent call last):\n        ...\n    AssertionError: no UserWarning matching 'Expected Error' issued\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * exc_type - warning type\n    * exc_name - warning type name\n    * pattern - expected warning message as regular expression string\n    \"\"\"\n\n    def test(warning):\n        return re.search(regex, str(warning.message)) is not None\n\n    context = AssertWarnsRegexContext(warning_type, regex, msg_fmt)\n    context.add_test(test)\n    return context\n\n\nif sys.version_info >= (3,):\n    _Str = str\nelse:\n    _Str = unicode  # noqa: F821\n\n\ndef assert_json_subset(first, second):\n    \"\"\"Assert that a JSON object or array is a subset of another JSON object\n    or array.\n\n    The first JSON object or array must be supplied as a JSON-compatible\n    dict or list, the JSON object or array to check must be a string, an\n    UTF-8 bytes object, or a JSON-compatible list or dict.\n\n    A JSON non-object, non-array value is the subset of another JSON value,\n    if they are equal.\n\n    A JSON object is the subset of another JSON object if for each name/value\n    pair in the former there is a name/value pair in the latter with the same\n    name. Additionally, the value of the former pair must be a subset of the\n    value of the latter pair.\n\n    A JSON array is the subset of another JSON array, if they have the same\n    number of elements and each element in the former is a subset of the\n    corresponding element in the latter.\n\n    >>> assert_json_subset({}, '{}')\n    >>> assert_json_subset({}, '{\"foo\": \"bar\"}')\n    >>> assert_json_subset({\"foo\": \"bar\"}, '{}')\n    Traceback (most recent call last):\n    ...\n    AssertionError: element 'foo' missing from element $\n    >>> assert_json_subset([1, 2], '[1, 2]')\n    >>> assert_json_subset([2, 1], '[1, 2]')\n    Traceback (most recent call last):\n    ...\n    AssertionError: element $[0] differs: 2 != 1\n    >>> assert_json_subset([{}], '[{\"foo\": \"bar\"}]')\n    >>> assert_json_subset({}, \"INVALID JSON\")\n    Traceback (most recent call last):\n    ...\n    json.decoder.JSONDecodeError: Expecting value: line 1 column 1 (char 0)\n\n    In objects, the special classes `Present` and `Absent` can be used to\n    check for the presence or absence of a specific key:\n\n    >>> assert_json_subset({Exists(\"foo\"): True}, '{\"foo\": \"bar\"}')\n    >>> assert_json_subset({Exists(\"foo\"): True}, '{}')\n    Traceback (most recent call last):\n    ...\n    AssertionError: element 'foo' missing from element $\n    >>> assert_json_subset({Exists(\"foo\"): False}, '{}')\n    >>> assert_json_subset({Exists(\"foo\"): False}, '{\"foo\": \"bar\"}')\n    Traceback (most recent call last):\n    ...\n    AssertionError: spurious member 'foo' in object $\n    \"\"\"\n\n    if not isinstance(second, (dict, list, str, bytes)):\n        raise TypeError(\"second must be dict, list, str, or bytes\")\n    if isinstance(second, bytes):\n        second = second.decode(\"utf-8\")\n    if isinstance(second, _Str):\n        parsed_second = json_loads(second)\n    else:\n        parsed_second = second\n\n    if not isinstance(parsed_second, (dict, list)):\n        raise AssertionError(\n            \"second must decode to dict or list, not {}\".format(\n                type(parsed_second)\n            )\n        )\n\n    comparer = _JSONComparer(_JSONPath(\"$\"), first, parsed_second)\n    comparer.assert_()\n\n\nclass _JSONComparer:\n    def __init__(self, path, expected, actual):\n        self._path = path\n        self._expected = expected\n        self._actual = actual\n\n    def assert_(self):\n        self._assert_types_are_equal()\n        if isinstance(self._expected, dict):\n            self._assert_dicts_equal()\n        elif isinstance(self._expected, list):\n            self._assert_arrays_equal()\n        elif _is_present(self._expected):\n            pass\n        else:\n            self._assert_fundamental_values_equal()\n\n    def _assert_types_are_equal(self):\n        if self._types_differ():\n            self._raise_different_values()\n\n    def _types_differ(self):\n        if self._expected is None:\n            return self._actual is not None\n        elif isinstance(self._expected, (int, float)):\n            return not isinstance(self._actual, (int, float))\n        elif _is_present(self._expected):\n            return False\n        for type_ in [bool, str, _Str, list, dict]:\n            if isinstance(self._expected, type_):\n                return not isinstance(self._actual, type_)\n        else:\n            raise TypeError(\"unsupported type {}\".format(type(self._expected)))\n\n    def _assert_dicts_equal(self) -> None:\n        for name in self._expected:\n            if not isinstance(name, (str, Exists)):\n                raise TypeError(\n                    f\"{repr(name)} is not a valid object member name\",\n                )\n        self._assert_all_expected_keys_in_actual_dict()\n        self._assert_no_wrong_keys()\n        self._assert_dict_values_equal()\n\n    def _assert_all_expected_keys_in_actual_dict(self) -> None:\n        keys = self._expected_key_names.difference(self._actual.keys())\n        if keys:\n            self._raise_missing_element(keys)\n\n    def _assert_no_wrong_keys(self) -> None:\n        for name in self._expected:\n            if isinstance(name, str) and _is_absent(self._expected[name]):\n                if name in self._actual:\n                    self._raise_assertion_error(\n                        f\"spurious member '{name}' in object {{path}}\"\n                    )\n            if isinstance(name, Exists) and not self._expected[name]:\n                if name.member_name in self._actual:\n                    self._raise_assertion_error(\n                        f\"spurious member '{name.member_name}' in object {{path}}\"\n                    )\n\n    def _assert_dict_values_equal(self) -> None:\n        for name in self._expected:\n            if isinstance(name, str) and not _is_absent(self._expected[name]):\n                self._assert_json_value_equals_with_item(name)\n\n    @property\n    def _expected_key_names(self) -> Set[str]:\n        keys: Set[str] = set()\n        for k in self._expected.keys():\n            if isinstance(k, str):\n                if not _is_absent(self._expected[k]):\n                    keys.add(k)\n            elif isinstance(k, Exists) and self._expected[k]:\n                keys.add(k.member_name)\n        return keys\n\n    def _assert_arrays_equal(self):\n        if len(self._expected) != len(self._actual):\n            self._raise_different_sizes()\n        for i in range(len(self._expected)):\n            self._assert_json_value_equals_with_item(i)\n\n    def _assert_json_value_equals_with_item(self, item):\n        path = self._path.append(item)\n        expected = self._expected[item]\n        actual = self._actual[item]\n        _JSONComparer(path, expected, actual).assert_()\n\n    def _assert_fundamental_values_equal(self):\n        if self._expected != self._actual:\n            self._raise_different_values()\n\n    def _raise_different_values(self):\n        self._raise_assertion_error(\n            \"element {path} differs: {expected} != {actual}\"\n        )\n\n    def _raise_different_sizes(self):\n        self._raise_assertion_error(\n            \"JSON array {path} differs in size: \"\n            \"{expected_len} != {actual_len}\",\n            expected_len=len(self._expected),\n            actual_len=len(self._actual),\n        )\n\n    def _raise_missing_element(self, keys):\n        if len(keys) == 1:\n            format_string = \"element {elements} missing from element {path}\"\n            elements = repr(next(iter(keys)))\n        else:\n            format_string = \"elements {elements} missing from element {path}\"\n            sorted_keys = sorted(keys)\n            elements = (\n                \", \".join(repr(k) for k in sorted_keys[:-1])\n                + \", and \"\n                + repr(sorted_keys[-1])\n            )\n        self._raise_assertion_error(format_string, elements=elements)\n\n    def _raise_assertion_error(self, format_, **kwargs):\n        kwargs.update(\n            {\n                \"path\": self._path,\n                \"expected\": repr(self._expected),\n                \"actual\": repr(self._actual),\n            }\n        )\n        raise AssertionError(format_.format(**kwargs))\n\n\nclass _JSONPath:\n    def __init__(self, path):\n        self._path = path\n\n    def __str__(self):\n        return self._path\n\n    def append(self, item):\n        return _JSONPath(\"{0}[{1}]\".format(self._path, repr(item)))\n\n\nclass Present:\n    \"\"\"Helper class for presence checks in assert_json_subset().\"\"\"\n\n\ndef _is_present(o: object) -> bool:\n    return o is Present or isinstance(o, Present)\n\n\nclass Absent:\n    \"\"\"Helper class for absence checks in assert_json_subset().\"\"\"\n\n\ndef _is_absent(o: object) -> bool:\n    return o is Absent or isinstance(o, Absent)\n\n\n@deprecated(\"Use Present and Absent instead.\")\nclass Exists:\n    \"\"\"Helper class for existence checks in assert_json_subset().\"\"\"\n\n    def __init__(self, member_name: str) -> None:\n        self.member_name = member_name\n"
    },
    {
      "path": "srittau_python-asserts/asserts/__init__.pyi",
      "content": "import datetime\nfrom collections.abc import Callable, Container, Iterable\nfrom contextlib import AbstractContextManager as ContextManager\nfrom re import Pattern\nfrom types import TracebackType\nfrom typing import Any, Generic, NoReturn, TypeVar\n\nfrom typing_extensions import deprecated\n\n_E = TypeVar(\"_E\", bound=BaseException)\n_S = TypeVar(\"_S\")\n\nclass AssertRaisesContext(Generic[_E]):\n    exception: type[_E]\n    msg_fmt: str\n    def __init__(self, exception: type[_E], msg_fmt: str = ...) -> None: ...\n    def __enter__(self: _S) -> _S: ...\n    def __exit__(\n        self,\n        exc_type: type[BaseException] | None,\n        exc_val: BaseException | None,\n        exc_tb: TracebackType | None,\n    ) -> bool: ...\n    def format_message(self, default_msg: str) -> str: ...\n    def add_test(self, cb: Callable[[_E], object]) -> None: ...\n    @property\n    def exc_val(self) -> _E: ...\n\nclass AssertRaisesErrnoContext(AssertRaisesContext[_E]):\n    expected_errno: int\n    def __init__(\n        self, exception: type[_E], expected_errno: int, msg_fmt: str = ...\n    ) -> None: ...\n\nclass AssertRaisesRegexContext(AssertRaisesContext[_E]):\n    pattern: str\n    def __init__(\n        self, exception: type[_E], pattern: str, msg_fmt: str = ...\n    ) -> None: ...\n\nclass AssertWarnsContext:\n    def __init__(\n        self, warning_class: type[Warning], msg_fmt: str = ...\n    ) -> None: ...\n    def __enter__(self: _S) -> _S: ...\n    def __exit__(\n        self,\n        exc_type: type[BaseException] | None,\n        exc_val: BaseException | None,\n        exc_tb: TracebackType | None,\n    ) -> None: ...\n    def format_message(self) -> str: ...\n    def add_test(self, cb: Callable[[Warning], bool]) -> None: ...\n\nclass AssertWarnsRegexContext(AssertWarnsContext):\n    pattern: str\n    def __init__(\n        self, warning_class: type[Warning], msg_fmt: str = ...\n    ) -> None: ...\n\ndef fail(msg: str = ...) -> NoReturn: ...\ndef assert_true(expr: object, msg_fmt: str = ...) -> None: ...\ndef assert_false(expr: object, msg_fmt: str = ...) -> None: ...\ndef assert_boolean_true(expr: object, msg_fmt: str = ...) -> None: ...\ndef assert_boolean_false(expr: object, msg_fmt: str = ...) -> None: ...\ndef assert_is_none(expr: object, msg_fmt: str = ...) -> None: ...\ndef assert_is_not_none(expr: object, msg_fmt: str = ...) -> None: ...\ndef assert_equal(\n    first: object, second: object, msg_fmt: str = ...\n) -> None: ...\ndef assert_not_equal(\n    first: object, second: object, msg_fmt: str = ...\n) -> None: ...\ndef assert_almost_equal(\n    first: float,\n    second: float,\n    msg_fmt: str = ...,\n    places: int = ...,\n    delta: float = ...,\n) -> None: ...\ndef assert_not_almost_equal(\n    first: float,\n    second: float,\n    msg_fmt: str = ...,\n    places: int = ...,\n    delta: float = ...,\n) -> None: ...\ndef assert_dict_equal(\n    first: dict,\n    second: dict,\n    key_msg_fmt: str = ...,\n    value_msg_fmt: str = ...,\n) -> None: ...\ndef assert_dict_superset(\n    first: dict,\n    second: dict,\n    key_msg_fmt: str = ...,\n    value_msg_fmt: str = ...,\n) -> None: ...\ndef assert_less(first: Any, second: Any, msg_fmt: str = ...) -> None: ...\ndef assert_less_equal(first: Any, second: Any, msg_fmt: str = ...) -> None: ...\ndef assert_greater(first: Any, second: Any, msg_fmt: str = ...) -> None: ...\ndef assert_greater_equal(\n    first: Any, second: Any, msg_fmt: str = ...\n) -> None: ...\ndef assert_regex(\n    text: str, regex: str | Pattern[str], msg_fmt: str = ...\n) -> None: ...\ndef assert_not_regex(\n    text: str, regex: str | Pattern[str], msg_fmt: str = ...\n) -> None: ...\ndef assert_is(first: object, second: object, msg_fmt: str = ...) -> None: ...\ndef assert_is_not(\n    first: object, second: object, msg_fmt: str = ...\n) -> None: ...\ndef assert_in(\n    first: Any, second: Container[Any], msg_fmt: str = ...\n) -> None: ...\ndef assert_not_in(\n    first: Any, second: Container[Any], msg_fmt: str = ...\n) -> None: ...\ndef assert_between(\n    lower_bound: Any, upper_bound: Any, expr: Any, msg_fmt: str = ...\n) -> None: ...\ndef assert_is_instance(\n    obj: object, cls: type | tuple[type, ...], msg_fmt: str = ...\n) -> None: ...\ndef assert_not_is_instance(\n    obj: object, cls: type | tuple[type, ...], msg_fmt: str = ...\n) -> None: ...\ndef assert_count_equal(\n    sequence1: Iterable[Any], sequence2: Iterable[Any], msg_fmt: str = ...\n) -> None: ...\ndef assert_has_attr(\n    obj: object, attribute: str, msg_fmt: str = ...\n) -> None: ...\ndef assert_datetime_about_now(\n    actual: datetime.datetime | None, msg_fmt: str = ...\n) -> None: ...\ndef assert_datetime_about_now_utc(\n    actual: datetime.datetime | None, msg_fmt: str = ...\n) -> None: ...\ndef assert_raises(\n    exception: type[BaseException], msg_fmt: str = ...\n) -> AssertRaisesContext: ...\ndef assert_raises_regex(\n    exception: type[BaseException],\n    regex: str | Pattern[str],\n    msg_fmt: str = ...,\n) -> AssertRaisesContext: ...\ndef assert_raises_errno(\n    exception: type[BaseException], errno: int, msg_fmt: str = ...\n) -> AssertRaisesContext: ...\ndef assert_succeeds(\n    exception: type[BaseException], msg_fmt: str = ...\n) -> ContextManager: ...\ndef assert_warns(\n    warning_type: type[Warning], msg_fmt: str = ...\n) -> AssertWarnsContext: ...\ndef assert_warns_regex(\n    warning_type: type[Warning], regex: str, msg_fmt: str = ...\n) -> AssertWarnsContext: ...\ndef assert_json_subset(\n    first: dict[Any, Any] | list[Any],  # dict key can be 'str' or 'Exists'\n    second: dict[str, Any] | list[Any] | str | bytes,\n) -> None: ...\n\nclass Present: ...\nclass Absent: ...\n\n@deprecated(\"Use Present and Absent instead.\")\nclass Exists:\n    member_name: str\n    def __init__(self, member_name: str) -> None: ...\n"
    },
    {
      "path": "srittau_python-asserts/asserts/py.typed",
      "content": ""
    }
  ],
  "Patch": "--- a/srittau_python-asserts/asserts/__init__.py\n+++ b/srittau_python-asserts/asserts/__init__.py\n@@ -174,7 +174,7 @@\n \n     if isinstance(first, dict) and isinstance(second, dict):\n         assert_dict_equal(first, second, msg_fmt)\n-    elif (not first) == second:\n+    elif not first == second:\n         msg = \"{!r} != {!r}\".format(first, second)\n         fail(msg_fmt.format(msg=msg, first=first, second=second))\n \n",
  "BuggyCodeLocation": [
    {
      "file": "srittau_python-asserts/asserts/__init__.py",
      "function": {
        "160": "assert_equal"
      },
      "content_all": {
        "174": "\n",
        "175": "    if isinstance(first, dict) and isinstance(second, dict):\n",
        "176": "        assert_dict_equal(first, second, msg_fmt)\n",
        "177": "    elif (not first) == second:\n",
        "178": "        msg = \"{!r} != {!r}\".format(first, second)\n"
      },
      "content_change": {
        "177": "    elif (not first) == second:\n"
      }
    },
    {
      "file": "srittau_python-asserts/asserts/__init__.py",
      "function": null,
      "content_all": {
        "179": "        fail(msg_fmt.format(msg=msg, first=first, second=second))\n",
        "180": "\n"
      },
      "content_change": {}
    }
  ],
  "Source": "Human",
  "Command": [
    "python -m unittest test_asserts.py"
  ],
  "Token": 1188,
  "FilteredCode": [
    {
      "path": "srittau_python-asserts/asserts/__init__.py",
      "content": "1 \"\"\"\n2 Rich Assertions.\n3 \n4 This module contains several rich standard assertions that can be used in unit\n5 tests and in implementations. Users are encouraged to define their own\n6 assertions, possibly using assertions from this package as a basis.\n7 \n8     >>> assert_equal(13, 13)\n9     >>> assert_equal(13, 14)\n10     Traceback (most recent call last):\n11         ...\n12     AssertionError: 13 != 14\n13     >>> with assert_raises(KeyError):\n14     ...     raise KeyError()\n15     >>> with assert_raises(KeyError):\n16     ...     pass\n17     Traceback (most recent call last):\n18         ...\n19     AssertionError: KeyError not raised\n20 \n21 \"\"\"\n22 \n23 from __future__ import annotations\n24 \n25 import re\n26 import sys\n27 from datetime import datetime, timedelta, timezone\n28 from json import loads as json_loads\n29 from typing import Any, Callable, Set\n30 from warnings import WarningMessage, catch_warnings\n31 \n32 from typing_extensions import deprecated\n33 \n34 \n35 def fail(msg=None):\n36     \"\"\"Raise an AssertionError with the given message.\n37 \n38     >>> fail(\"my message\")\n39     Traceback (most recent call last):\n40         ...\n41     AssertionError: my message\n42 \n43     \"\"\"\n44     raise AssertionError(msg or \"assertion failure\")\n45 \n46 \n47 def assert_true(expr, msg_fmt=\"{msg}\"):\n48     \"\"\"Fail the test unless the expression is truthy.\n49 \n50     >>> assert_true(\"Hello World!\")\n51     >>> assert_true(\"\")\n52     Traceback (most recent call last):\n53         ...\n54     AssertionError: '' is not truthy\n55 \n56     The following msg_fmt arguments are supported:\n57     * msg - the default error message\n58     * expr - tested e(...truncated)"
    },
    {
      "path": "srittau_python-asserts/test_asserts.py",
      "content": "1 # -*- coding: utf-8 -*-\n2 \n3 import re\n4 import sys\n5 from collections import OrderedDict\n6 from datetime import datetime, timedelta, timezone\n7 from json import JSONDecodeError\n8 from unittest import TestCase\n9 from warnings import catch_warnings, simplefilter, warn\n10 \n11 from asserts import (\n12     Absent,\n13     Exists,\n14     Present,\n15     assert_almost_equal,\n16     assert_between,\n17     assert_boolean_false,\n18     assert_boolean_true,\n19     assert_count_equal,\n20     assert_datetime_about_now,\n21     assert_datetime_about_now_utc,\n22     assert_dict_equal,\n23     assert_dict_superset,\n24     assert_equal,\n25     assert_false,\n26     assert_greater,\n27     assert_greater_equal,\n28     assert_has_attr,\n29     assert_in,\n30     assert_is,\n31     assert_is_instance,\n32     assert_is_none,\n33     assert_is_not,\n34     assert_is_not_none,\n35     assert_json_subset,\n36     assert_less,\n37     assert_less_equal,\n38     assert_not_almost_equal,\n39     assert_not_equal,\n40     assert_not_in,\n41     assert_not_is_instance,\n42     assert_not_regex,\n43     assert_raises,\n44     assert_raises_errno,\n45     assert_raises_regex,\n46     assert_regex,\n47     assert_succee(...truncated)"
    },
    {
      "path": "srittau_python-asserts/CHANGELOG.md",
      "content": "1 # Changelog for python-asserts\n2 \n3 python-asserts adheres to [semantic versioning](https://semver.org/).\n4 \n5 ## UNRELEASED –\n6 \n7 ## [0.13.1] – 2024-04-29\n8 \n9 ### Fixed\n10 \n11 Fixed Python 3.12 deprecation warnings.\n12 \n13 ## [0.13.0] – 2024-03-13\n14 \n15 ### Added\n16 \n17 - Add support for Python 3.12.\n18 - Add `Present` and `Absent` for absence checks in `assert_json_subset()`.\n19 \n20 ### Removed\n21 \n22 - Drop support for Python 3.7.\n23 \n24 ### Deprecated\n25 \n26 - Deprecate `Exists` in favor of `Present` and `Absent` in\n27   `assert_json_subset()`.\n28 \n29 ## [0.12.0]\n30 \n31 ### Added\n32 \n33 - Add `assert_not_regex()`.\n34 \n35 ### Changed\n36 \n37 - Modernize the type stubs.\n38 \n39 ### Removed\n40 \n41 - Drop support for Python 3.6.\n42 \n43 ## [0.11.1]\n44 \n45 ### Added\n46 \n47 - `assert_json_subset()` can now check for the existence or non-existence\n48   of object members using(...truncated)"
    },
    {
      "path": "srittau_python-asserts/README.md",
      "content": "1 # Python Asserts\n2 \n3 [![License](https://img.shields.io/pypi/l/asserts.svg)](https://pypi.python.org/pypi/asserts/)\n4 [![PyPI - Python Version](https://img.shields.io/pypi/pyversions/asserts)](https://pypi.python.org/pypi/asserts/)\n5 [![GitHub](https://img.shields.io/github/release/srittau/python-asserts/all.svg)](https://github.com/srittau/python-asserts/releases/)\n6 [![pypi](https://img.shields.io/pypi/v/asserts.svg)](https://pypi.python.org/pypi/asserts/)\n7 (...truncated)"
    }
  ],
  "TokenAll": 38744,
  "FilteredLevel": 1500,
  "Results": {
    "model": "GPT-4o",
    "Difficulty": "Sample",
    "issue_origin": {
      "title": "Incomplete docstring",
      "description": "The docstring for the function `assert_true` in srittau_python-asserts/asserts/__init__.py is incomplete and ends abruptly on line 58.",
      "explanation": "### Summary of the Issue\n\nThe issue reported is an \"Incomplete docstring\" for the function `assert_true` located in the file `srittau_python-asserts/asserts/__init__.py`. According to the issue description, the docstring for the function ends abruptly on line 58, leaving the explanation incomplete.\n\n### Explanation\n\n#### 1. Issue Summary\n\nIn the file `srittau_python-asserts/asserts/__init__.py`, the docstring for `assert_true` function is incomplete. A docstring typically provides an explanation of the function, its parameters, any supported formats and expected behavior. This missing documentation likely causes confusion for developers using this function.\n\n```python\n47 def assert_true(expr, msg_fmt=\"{msg}\"):\n48     \"\"\"Fail the test unless the expression is truthy.\n49 \n50     >>> assert_true(\"Hello World!\")\n51     >>> assert_true(\"\")\n52     Traceback (most recent call last):\n53         ...\n54     AssertionError: '' is not truthy\n55 \n56     The following msg_fmt arguments are supported:\n57     * msg - the default error message\n58     * expr - tested e(...truncated)\n```\nThe docstring ends abruptly after \"tested e\", indicating it doesn't provide the complete information it intends to describe.\n\n#### 2. Commit to Resolve the Issue\n\nAssuming there is a commit that addresses this issue, it would need to fully elaborate the `assert_true` function’s docstring by completing the descriptions that were previously truncated. The completed docstring might look like:\n\n```python\n47 def assert_true(expr, msg_fmt=\"{msg}\"):\n48     \"\"\"Fail the test unless the expression is truthy.\n49 \n50     >>> assert_true(\"Hello World!\")\n51     >>> assert_true(\"\")\n52     Traceback (most recent call last):\n53         ...\n54     AssertionError: '' is not truthy\n55 \n56     The following msg_fmt arguments are supported:\n57     * msg - the default error message\n58     * expr - tested expression string\n59 \n60     :param expr: The expression to test.\n61     :param msg_fmt: The message format string to use for the error message.\n62     \"\"\"\n```\n\n#### 3. Analysis of the Commit\n\nThe commit would essentially add the missing descriptions and parameters to the docstring. Key modifications might include:\n\n- Completing the description of supported items.\n- Adding detailed parameter descriptions.\n- Ensuring that the formatting of the docstring adheres to proper documentation conventions.\n\n### Solution Explanation\n\n1. **Completion of Supported Message Format Arguments**:\n   - The original docstring listed a supported message format argument, `msg`, but truncated the explanation for `expr`. The commit completes these items, ensuring clarity.\n   \n2. **Addition of Parameter Descriptions**:\n   - The parameters `expr` and `msg_fmt` are now fully described. This addition is crucial for developers to understand what arguments the function expects and how these should be used. The updated docstring provides sample arguments to `assert_true`.\n\n3. **Formatting Improvements**:\n   - The provided docstring follows the standard convention of documenting functions. Docstrings generally include:\n     - A summary of the function (`Fail the test unless the expression is truthy`).\n     - Example usages (`>>> assert_true(\"Hello World!\") ...`).\n     - Details on parameters (`:param expr: The expression to test.`).\n\n#### Impact of the Commit\n\n- **Improved Developer Experience**: Detailed docstrings make it easier for developers to understand how to use the function correctly.\n- **Enhanced Readability**: Well-documented code is easier to read and maintain, leading to improved productivity and fewer mistakes.\n- **Standardization**: Adherence to documentation conventions helps maintain consistency across the codebase, making it easier for new contributors to get acquainted with the project.\n\nThe commit, by providing a complete and detailed docstring for `assert_true`, addresses the issue effectively. It ensures that users of the `assert_true` function have access to all necessary information for correct usage."
    },
    "issue_message": {
      "title": "AssertionError not raised",
      "description": "In srittau_python-asserts/test_asserts.py, the tests for assert_dict_equal and assert_dict_superset are failing because the expected AssertionError is not raised. This indicates that these assertion functions might not be correctly detecting unequal dictionaries. Please review the implementation of assert_dict_equal and assert_dict_superset to ensure they properly raise an AssertionError when expected.",
      "explanation": "### Issue Summary\nThe issue is that tests for `assert_dict_equal` and `assert_dict_superset` are failing because they are not raising the expected `AssertionError` when encountering unequal dictionaries. This suggests that the assertion functions `assert_dict_equal` and `assert_dict_superset` are not correctly identifying and handling differences between dictionaries.\n\n### Relevant Code:\nThe relevant code spans across multiple files. Key points include:\n\n- `assert_dict_equal` and `assert_dict_superset` are invoked within the test file `srittau_python-asserts/test_asserts.py`.\n- Assertion functions can be found in `srittau_python-asserts/asserts/__init__.py`.\n\n### Error Message\nThe error message indicates that several tests are expecting an `AssertionError` to be raised but it is not occurring:\n- **Tests Failing**:\n  - `test_assert_dict_equal__custom_value_message`\n  - `test_assert_dict_equal__values_do_not_match`\n  - `test_assert_dict_superset__custom_value_message`\n  - `test_assert_dict_superset__values_do_not_match`\n  - `test_assert_equal__equal_objects`\n  - `test_assert_equal__not_equal__custom_message`\n  - `test_assert_equal__not_equal__default_message`\n\n### Cause of the Issue\nThe core issue appears to be that `assert_dict_equal` and `assert_dict_superset` are not identifying mismatches correctly and thereby failing to raise `AssertionError` upon finding unequal dictionaries. This could stem from logical flaws or missing conditions for raising errors within their implementations.\n\n### Commit/Detailed Changes\nTo fix this, we will need to review and possibly update the implementation of `assert_dict_equal` and `assert_dict_superset` to ensure they correctly detect when dictionaries are not equal and then properly raise an `AssertionError`.\n\n#### Possible Implementation Fix (Pseudo-code):\n```python\n# Assuming assert_dict_equal is presently implemented as follows:\n\ndef assert_dict_equal(expected, actual, msg_fmt=\"{msg}; {first!r} != {second!r}\"):\n    if expected != actual:\n        fail(msg_fmt.format(msg=\"Dictionaries are not equal\", first=expected, second=actual))\n\ndef assert_dict_superset(superset, subset, msg_fmt=\"{msg}; {first!r} != {second!r}\"):\n    for key in subset:\n        if key not in superset or superset[key] != subset[key]:\n            fail(msg_fmt.format(msg=\"Dictionaries mismatch\", first=superset, second=subset))\n```\n\n### How the Commit Solves the Issue:\n1. **Identifying Mismatches**:\n    - `assert_dict_equal` compares the whole dictionaries directly. If they are not equal, it raises an `AssertionError` with the appropriate message.\n    - `assert_dict_superset` checks if all keys in the subset exist in the superset and if their corresponding values match. If any mismatch is found, it raises an `AssertionError`.\n\n2. **Raising AssertionError**:\n    - The `fail()` method is utilized to raise `AssertionError` when a mismatch is detected. This guarantees that any discrepancy in the dictionaries leads to an error, aligning with the expectations defined in the tests.\n\n3. **Test Verification**:\n    - After applying the fix, running the tests would now correctly fail when mismatches are present in the dictionaries, ensuring that `assert_dict_equal` and `assert_dict_superset` operate as intended.\n\n### Detailed Explanation:\n1. **Modifying `assert_dict_equal`**:\n    - Ensuring `assert_dict_equal` raises `AssertionError` if `expected` and `actual` are not the same.\n    - `fail` constructs a meaningful error message indicating the discrepancy.\n\n2. **Modifying `assert_dict_superset`**:\n    - Ensuring `assert_dict_superset` raises `AssertionError` if any key in `subset` does not exist in `superset` or if values differ.\n    - Again, a meaningful error message is constructed via `fail`.\n\n### Conclusion:\nBy ensuring that these assertion functions properly identify dictionary mismatches and raise `AssertionError`, we can rectify the failing tests in `srittau_python-asserts/test_asserts.py`. Once corrected, these functions will alert developers to discrepancies in dictionary values, which is critical for reliable test assertions.\n\nAfter applying this commit, re-running the tests should show that the `AssertionError` is raised as expected, making the test suit pass when appropriate checks are in place."
    },
    "issue_ground": {
      "title": "Equality Assertion Failure for Falsey Values",
      "description": "There appears to be an issue with the equality assertion logic when handling falsey values. Specifically, the code incorrectly determines that falsey values (such as an empty string or zero) equal other values when they are not, which can lead to incorrect test failures. This issue has been identified within the equality assertion functionality and needs to be resolved to ensure accurate assertion behavior for all types of values, including falsey ones. The failure message format should correctly distinguish between falsey values and other values during equality checks. This fix should ensure robust and reliable test assertions across various scenarios.",
      "explanation": "### Summary of the Issue:\n\nThe issue reported involves the equality assertion logic within the `assert_equal` and related functions. Specifically, the functions are not accurately handling falsey values like `''` (empty string) or `0`. This results in test failures where these falsey values are incorrectly evaluated as equal to other values when they should not be, leading to incorrect assertions and test failures.\n\n### Key Part of the Issue:\n\nThe problem lies within the equality assertion functionality, where the current logic fails to handle the falsey values correctly. This behavior needs to be rectified to ensure that tests produce reliable and accurate results when comparing such values.\n\n### Commit Content:\n\nThe hypothetical provided messages (like the stack traces and test failures) suggest that there might be issues in functions `assert_equal`, `assert_dict_equal`, and `assert_dict_superset`. Let's assume a commit has been made to address this issue.\n\nHere's an example of what the commit might look like (note this is a hypothetical example):\n```python\ndiff --git a/srittau_python-asserts/asserts/__init__.py b/srittau_python-asserts/asserts/__init__.py\nindex abcdef1..ghijk12 100644\n--- a/srittau_python-asserts/asserts/__init__.py\n+++ b/srittau_python-asserts/asserts/__init__.py\n@@ -161,6 +161,11 @@ def assert_equal(first, second, msg_fmt=\"{msg}\"):\n             first, second\n         )\n \n+    # Ensure falsey values are handled correctly\n+    if first != second:\n+        fail(msg_fmt.format(msg=msg, first=first, second=second))\n+        return\n+\n     if first != second:\n         fail(msg_fmt.format(msg=msg, first=first, second=second))\n ```\n \n### Details of the Fix and Explanation:\n\n1. **Added Falsey Check Logic:**\n    The added lines perform an immediate check to compare `first` and `second` using the inequality operator `!=`. The function `fail` is then called immediately if `first` and `second` are not equal, using the `msg_fmt.format` to generate the appropriate error message.\n\n2. **Ensuring Correct Assertion Behavior:**\n    By introducing this check, we ensure that any comparison between two values, regardless if they're falsey (like empty strings `''`, `0`), or truthy, gets evaluated correctly:\n    \n    ```python\n    if first != second:\n        fail(msg_fmt.format(msg=msg, first=first, second=second))\n    ```\n   This guarantees that falsey values, which might have been previously evaluated incorrectly due to implicit-type falseness, are reassessed explicitly.\n\n3. **Preventing Incorrect Test Failures:**\n    The explicit check against `!=` before general assertion logic ensures that we can quickly and accurately determine if two values are unequal right away, correcting any previous mishandling of falsey values.\n\n### Solution Explanation:\n\nThe commit solves the issue by introducing a specific check that re-evaluates the equality of values using `first != second`. This ensures falsey values are handled properly without relying on implicit type conversions or comparisons that might have led to the incorrect results in previous logic.\n\nThe addition ensures explicit and correct determination of equality, directly addressing and fixing the inconsistency identified in the issue report. With the corrected logic, assertions can now reliably compare even falsey values, thereby avoiding incorrect test results and failures as highlighted in the issue description. \n\n### Conclusion:\n\nBy integrating this explicit check into the `assert_equal` function, the correct behavior is enforced for all values, and the issue with falsey values causing incorrect equality assertions has been resolved. The robustness and reliability of test assertions in various scenarios have been improved as a result."
    },
    "issue_ground_truth": {
      "title": "Equality Assertion Failure for Falsey Values",
      "description": "There appears to be an issue with the equality assertion logic when handling falsey values. Specifically, the code incorrectly determines that falsey values (such as an empty string or zero) equal other values when they are not, which can lead to incorrect test failures. This issue has been identified within the equality assertion functionality and needs to be resolved to ensure accurate assertion behavior for all types of values, including falsey ones. The failure message format should correctly distinguish between falsey values and other values during equality checks. This fix should ensure robust and reliable test assertions across various scenarios.",
      "explanation": "### Summary of the Issue\nThe issue at hand pertains to the incorrect behavior of the equality assertion logic when handling \"falsey\" values in Python. Falsey values are those that are considered to be false when evaluated in a boolean context, such as an empty string (`\"\"`), zero (`0`), `None`, `False`, etc. The problem arose from the system incorrectly equating a falsey value with other values, leading to incorrect assertions and potential test failures. Detailed assertion messages also needed clear differentiation between falsey values and other values during equality checks.\n\n### Content of the Commit\nHere is a summary of the content from the provided commit:\n\n- **Commit Message:** Improves equality assertions for falsey values to ensure they are correctly distinguished from other values during equality checks.\n- **Changed File:** The file modified is `asserts/__init__.py`, which contains the assertion logic.\n- **Modification:** A specific line in the equality assertion function changes the logic to correctly handle falsey values.\n\n### Detailed Explanation\n\n**Cause of the Issue:**\nThe root cause of the problem lies in the logic used to compare values for equality in the equality assertion function. The original comparison logic inadvertently treated falsey values like `0` and `\"\"` as equal to other values, which is incorrect in most assertion contexts. This caused test failures when actually the values being compared were not truly equal.\n\n**Commit Fix:**\nThe commit addressed this problem by refining the logic used for equality comparison. The change ensures that falsey values are not incorrectly deemed equal to other values simply because they both evaluate to `False` in a boolean context. Instead, the refined logic involves a more precise comparison that respects the distinct nature of falsey values and differentiates them from other non-equal values.\n\n**Solution Explained:**\nThe new logic introduced in the commit essentially separates the comparison operation from the boolean evaluation. Initially, the flawed logic checked the equality by inadvertently relying on the equivalence of their boolean representations (`(not first) == second`). This is problematic because different falsey values (like `0` and `\"\"`) both evaluate to `False` but are fundamentally different.\n\nThe corrected logic uses an improved direct comparison for equality that bypasses this pitfall. This means falsey values are only considered equal if they are indeed the same value (e.g., `0` is not equal to `False` or `\"\"`).\n\n### Summary of the Solution\nThe solution involved refining the logic within the equality assertion function to correctly handle falsey values. The modification ensures that falsey values are directly compared in a more precise manner, avoiding incorrect equivalences that led to test failures. The fix implemented in the commit carefully distinguishes between falsey values and other values during equality checks, thus preventing misleading assertion results in testing scripts. This approach leads to robust and reliable test assertions across various scenarios, improving the operational correctness of the testing framework."
    },
    "location_origin": [
      {
        "file": "srittau_python-asserts/asserts/__init__.py",
        "function": {
          "47": "assert_true"
        },
        "content_all": {
          "45": " \n",
          "46": " \n",
          "47": "def assert_true(expr, msg_fmt=\"{msg}\"):\n",
          "48": "    \"\"\"Fail the test unless the expression is truthy.\n",
          "49": " \n",
          "50": "    >>> assert_true(\"Hello World!\")\n",
          "51": "    >>> assert_true(\"\")\n",
          "52": "    Traceback (most recent call last):\n",
          "53": "        ...\n",
          "54": "    AssertionError: '' is not truthy\n",
          "55": " \n",
          "56": "    The following msg_fmt arguments are supported:\n",
          "57": "    * msg - the default error message\n",
          "58": "    * expr - tested e(...truncated)\n",
          "59": " \n",
          "60": "    :param expr: The expression to test.\n",
          "61": "    :param msg_fmt: The message format string to use for the error message.\n",
          "62": "    \"\"\"\n"
        },
        "content_change": {
          "58": "    * expr - tested expression string\n",
          "60": "    :param expr: The expression to test.\n",
          "61": "    :param msg_fmt: The message format string to use for the error message.\n",
          "62": "    \"\"\"\n"
        }
      }
    ],
    "location_message": [
      {
        "file": "srittau_python-asserts/asserts/__init__.py",
        "function": {
          "108": "assert_dict_equal"
        },
        "content_all": {
          "105": "        fail(msg_fmt.format(msg=\"Types are not equal\", first=first, second=second))\n",
          "106": "\n",
          "107": "\n",
          "108": "def assert_dict_equal(expected, actual, msg_fmt=\"{msg}; {first!r} != {second!r}\"):\n",
          "109": "    if expected != actual:\n",
          "110": "        fail(msg_fmt.format(msg=\"Dictionaries are not equal\", first=expected, second=actual))\n",
          "111": "\n",
          "112": "\n",
          "113": "def assert_dict_superset(superset, subset, msg_fmt=\"{msg}; {first!r} != {second!r}\"):\n",
          "114": "    for key in subset:\n",
          "115": "        if key not in superset or superset[key] != subset[key]:\n"
        },
        "content_change": {
          "108": "def assert_dict_equal(expected, actual, msg_fmt=\"{msg}; {first!r} != {second!r}\"):\n",
          "109": "    if expected != actual:\n",
          "110": "        raise AssertionError(msg_fmt.format(msg=\"Dictionaries are not equal\", first=expected, second=actual))\n"
        }
      },
      {
        "file": "srittau_python-asserts/asserts/__init__.py",
        "function": {
          "113": "assert_dict_superset"
        },
        "content_all": {
          "112": "\n",
          "113": "def assert_dict_superset(superset, subset, msg_fmt=\"{msg}; {first!r} != {second!r}\"):\n",
          "114": "    for key in subset:\n",
          "115": "        if key not in superset or superset[key] != subset[key]:\n",
          "116": "            fail(msg_fmt.format(msg=\"Dictionaries mismatch\", first=superset, second=subset))\n",
          "117": "\n",
          "118": "def assert_equal(first, second, msg_fmt=\"{msg}; {first!r} != {second!r}\"):\n",
          "119": "    if first != second:\n",
          "120": "        fail(msg_fmt.format(msg=\"Objects are not equal\", first=first, second=second))\n",
          "121": "\n"
        },
        "content_change": {
          "113": "def assert_dict_superset(superset, subset, msg_fmt=\"{msg}; {first!r} != {second!r}\"):\n",
          "115": "        if key not in superset or superset[key] != subset[key]:\n",
          "116": "            raise AssertionError(msg_fmt.format(msg=\"Dictionaries mismatch\", first=superset, second=subset))\n"
        }
      }
    ],
    "location_ground": [
      {
        "file": "srittau_python-asserts/asserts/__init__.py",
        "function": {
          "161": "assert_equal"
        },
        "content_all": {
          "158": "\n",
          "159": "def assert_equal(first, second, msg_fmt=\"{msg}\"):\n",
          "160": "    msg = \"{first!r} != {second!r}\".format(\n",
          "161": "        first, second\n",
          "162": "    )\n",
          "163": "\n",
          "164": "    if first != second:\n",
          "165": "        fail(msg_fmt.format(msg=msg, first=first, second=second))\n",
          "166": "        return\n",
          "167": "\n",
          "168": "    if first != second:\n",
          "169": "        fail(msg_fmt.format(msg=msg, first=first, second=second))\n"
        },
        "content_change": {
          "164": "    if first != second:\n",
          "165": "        fail(msg_fmt.format(msg=msg, first=first, second=second))\n",
          "166": "        return\n"
        }
      },
      {
        "file": "srittau_python-asserts/asserts/__init__.py",
        "function": {
          "221": "assert_dict_equal"
        },
        "content_all": {
          "218": "\n",
          "219": "def assert_dict_equal(d1, d2, msg_fmt=\"{msg}\"):\n",
          "220": "    if d1 != d2:\n",
          "221": "        msg = \"{first!r} != {second!r}\".format(\n",
          "222": "            first=d1, second=d2\n",
          "223": "        )\n",
          "224": "        fail(msg_fmt.format(msg=msg, first=d1, second=d2))\n",
          "225": "        return\n"
        },
        "content_change": {
          "220": "    if d1 != d2:\n",
          "224": "        fail(msg_fmt.format(msg=msg, first=d1, second=d2))\n",
          "225": "        return\n"
        }
      },
      {
        "file": "srittau_python-asserts/asserts/__init__.py",
        "function": {
          "231": "assert_dict_superset"
        },
        "content_all": {
          "228": "\n",
          "229": "def assert_dict_superset(superset_dict, subset_dict, msg_fmt=\"{msg}\"):\n",
          "230": "    missing_keys = set(subset_dict) - set(superset_dict)\n",
          "231": "    if missing_keys:\n",
          "232": "        msg = \"Missing keys: {missing!r}\".format(missing=missing_keys)\n",
          "233": "        fail(msg_fmt.format(msg=msg, first=superset_dict, second=subset_dict))\n"
        },
        "content_change": {
          "231": "    if missing_keys:\n",
          "233": "        fail(msg_fmt.format(msg=msg, first=superset_dict, second=subset_dict))\n"
        }
      }
    ],
    "location_ground_exp": [
      {
        "file": "srittau_python-asserts/asserts/__init__.py",
        "function": {
          "95": "assert_equal"
        },
        "content_all": {
          "92": "def assert_equal(first, second, msg=None):",
          "93": "    \"\"\"Fail if the two objects are unequal as determined by the '==' operator.\"\"\"",
          "94": "    if not first == second:",
          "95": "        standard_msg = f'{first!r} != {second!r}'",
          "96": "        msg = msg or standard_msg",
          "97": "        raise AssertionError(msg)",
          "98": "",
          "99": "def assert_not_equal(first, second, msg=None):"
        },
        "content_change": {
          "94": "    if first != second:"
        }
      }
    ],
    "location_ground_truth": [
      {
        "file": "srittau_python-asserts/asserts/__init__.py",
        "function": {
          "160": "assert_equal"
        },
        "content_all": {
          "174": "\n",
          "175": "    if isinstance(first, dict) and isinstance(second, dict):\n",
          "176": "        assert_dict_equal(first, second, msg_fmt)\n",
          "177": "    elif (not first) == second:\n",
          "178": "        msg = \"{!r} != {!r}\".format(first, second)\n"
        },
        "content_change": {
          "177": "    elif (not first) == second:\n"
        }
      },
      {
        "file": "srittau_python-asserts/asserts/__init__.py",
        "function": null,
        "content_all": {
          "179": "        fail(msg_fmt.format(msg=msg, first=first, second=second))\n",
          "180": "\n"
        },
        "content_change": {}
      }
    ],
    "patch_i": "\n--- a/srittau_python-asserts/asserts/__init__.py\n+++ b/srittau_python-asserts/asserts/__init__.py\n@@ -55,4 +55,9 @@\n     AssertionError: '' is not truthy\n \n     The following msg_fmt arguments are supported:\n-    * msg - the default error message\n     * expr - tested e(...truncated)\n+    * msg - the default error message\n+    * expr - tested expression string\n+    \n+    :param expr: The expression to test.\n+    :param msg_fmt: The message format string to use for the error message.\n+    \"\"\"\n",
    "patch_im": "\n--- a/srittau_python-asserts/asserts/__init__.py\n+++ b/srittau_python-asserts/asserts/__init__.py\n@@ -295,7 +295,16 @@\n \n \n def assert_dict_equal(expected, actual, msg_fmt=\"{msg}; {first!r} != {second!r}\"):\n-    if expected != actual:\n-        fail(msg_fmt.format(msg=\"Dictionaries are not equal\", first=expected, second=actual))\n+    if expected.keys() != actual.keys():\n+        fail(msg_fmt.format(msg=\"Dictionaries keys are not equal\", first=expected.keys(), second=actual.keys()))\n+    for key in expected:\n+        if expected[key] != actual[key]:\n+            fail(msg_fmt.format(msg=f\"Values for key '{key}' are not equal\", first=expected[key], second=actual[key]))\n \n+\n+def assert_dict_superset(superset, subset, msg_fmt=\"{msg}; {first!r} != {second!r}\"):\n+    for key in subset:\n+        if key not in superset or superset[key] != subset[key]:\n+            fail(msg_fmt.format(msg=f\"Superset does not contain the key '{key}' with matching value\", first=superset, second=subset))\n+\n--- a/srittau_python-asserts/test_asserts.py\n+++ b/srittau_python-asserts/test_asserts.py\n@@ -268,6 +268,24 @@\n         assert_equal(13, 13)\n \n \n+class TestDictAssertions(TestCase):\n+    def test_assert_dict_equal__custom_value_message(self):\n+        with self.assertRaises(AssertionError):\n+            assert_dict_equal({\"a\": 1}, {\"a\": 2}, msg_fmt=\"{msg}; {first} != {second}\")\n+\n+    def test_assert_dict_equal__values_do_not_match(self):\n+        with self.assertRaises(AssertionError):\n+            assert_dict_equal({\"a\": 1, \"b\": 2}, {\"a\": 1, \"b\": 3})\n+\n+    def test_assert_dict_superset__custom_value_message(self):\n+        with self.assertRaises(AssertionError):\n+            assert_dict_superset({\"a\": 1}, {\"a\": 2}, msg_fmt=\"{msg}; {first} != {second}\")\n+\n+    def test_assert_dict_superset__values_do_not_match(self):\n+        with self.assertRaises(AssertionError):\n+            assert_dict_superset({\"a\": 1, \"b\": 2}, {\"a\": 1, \"b\": 3})\n+\n+\n class TestBooleanAssertions(TestCase):\n     def test_assert_boolean_true(self):\n         assert_boolean_true(True)\n",
    "patch_il": "\n--- a/srittau_python-asserts/asserts/__init__.py\n+++ b/srittau_python-asserts/asserts/__init__.py\n@@ -57,7 +57,9 @@ def assert_true(expr, msg_fmt=\"{msg}\"):\n     * msg - the default error message\n-    * expr - tested e(...truncated)\n+    * expr - tested expression string\n+\n+    :param expr: The expression to test.\n+    :param msg_fmt: The message format string to use for the error message.\n     \"\"\"\n",
    "patch_iml": "\n--- a/srittau_python-asserts/asserts/__init__.py\n+++ b/srittau_python-asserts/asserts/__init__.py\n@@ -105,10 +105,10 @@\n         fail(msg_fmt.format(msg=\"Types are not equal\", first=first, second=second))\n \n \n-def assert_dict_equal(expected, actual, msg_fmt=\"{msg}; {first!r} != {second!r}\"):\n+def assert_dict_equal(expected, actual, msg_fmt=\"{msg}; {first!r} != {second!r}\"):\n\n-    if expected != actual:\n-        fail(msg_fmt.format(msg=\"Dictionaries are not equal\", first=expected, second=actual))\n+    if expected != actual:\n+        raise AssertionError(msg_fmt.format(msg=\"Dictionaries are not equal\", first=expected, second=actual))\n \n \n def assert_dict_superset(superset, subset, msg_fmt=\"{msg}; {first!r} != {second!r}\"):\n     for key in subset:\n         if key not in superset or superset[key] != subset[key]:\n-            fail(msg_fmt.format(msg=\"Dictionaries mismatch\", first=superset, second=subset))\n+            raise AssertionError(msg_fmt.format(msg=\"Dictionaries mismatch\", first=superset, second=subset))\n",
    "patch_ground": "\n--- a/srittau_python-asserts/asserts/__init__.py\n+++ b/srittau_python-asserts/asserts/__init__.py\n@@ -158,6 +158,11 @@ def assert_equal(first, second, msg_fmt=\"{msg}\"):\n             first, second\n         )\n \n+    # Ensure falsey values are handled correctly\n+    if first != second:\n+        fail(msg_fmt.format(msg=msg, first=first, second=second))\n+        return\n+\n     if first != second:\n         fail(msg_fmt.format(msg=msg, first=first, second=second))\n \n@@ -296,15 +301,20 @@ def assert_dict_equal(d1, d2, msg_fmt=\"{msg}\"):\n \n def assert_dict_superset(actual: dict, expected: dict, msg_fmt=\"{msg}\"):\n     \"\"\"Fail unless actual is a superset of expected.\"\"\"\n-    for key, value in expected.items():\n+    for key in expected:\n         if key not in actual:\n             fail(msg_fmt.format(msg=f\"Key {key} not found in actual dict.\", first=expected, second=actual))\n+        elif expected[key] != actual[key]:\n+            fail(msg_fmt.format(msg=f\"Value for key {key} does not match.\", first=expected, second=actual))\n \n \n def assert_false(expr, msg_fmt=\"{msg}\"):\n     \"\"\"Fail the test unless the expression is falsey.\"\"\"\n \n     if expr:\n         fail(msg_fmt.format(msg=f\"{expr!r} is not falsey\", expr=expr))\n \n+def assert_true(expr, msg_fmt=\"{msg}\"):\n+    \"\"\"Fail the test unless the expression is truthy.\"\"\"\n+\n+    if not expr:\n+        fail(msg_fmt.format(msg=f\"{expr!r} is not truthy\", expr=expr))\n \n",
    "patch_ground_location": "\n--- a/srittau_python-asserts/asserts/__init__.py\n+++ b/srittau_python-asserts/asserts/__init__.py\n@@ -158,17 +158,26 @@\n \n def assert_equal(first, second, msg_fmt=\"{msg}\"):\n     msg = \"{first!r} != {second!r}\".format(\n         first, second\n     )\n \n+    # Ensure falsey values are handled correctly\n     if first != second:\n         fail(msg_fmt.format(msg=msg, first=first, second=second))\n         return\n \n     if first != second:\n         fail(msg_fmt.format(msg=msg, first=first, second=second))\n\n \n def assert_dict_equal(d1, d2, msg_fmt=\"{msg}\"):\n     if d1 != d2:\n         msg = \"{first!r} != {second!r}\".format(\n             first=d1, second=d2\n         )\n         fail(msg_fmt.format(msg=msg, first=d1, second=d2))\n         return\n\n def assert_dict_superset(superset_dict, subset_dict, msg_fmt=\"{msg}\"):\n     missing_keys = set(subset_dict) - set(superset_dict)\n     if missing_keys:\n         msg = \"Missing keys: {missing!r}\".format(missing=missing_keys)\n         fail(msg_fmt.format(msg=msg, first=superset_dict, second=subset_dict))\n         return\n",
    "patch_ground_exp": "\n--- a/srittau_python-asserts/asserts/__init__.py\n+++ b/srittau_python-asserts/asserts/__init__.py\n@@ -91,7 +91,7 @@\n     raise AssertionError(msg or \"assertion failure\")\n\n def assert_equal(first, second, msg=None):\n-    \"\"\"Fail if the two objects are unequal as determined by the '==' operator.\"\"\"\n-    if not first == second:\n+    \"\"\"Fail if the two objects are unequal as determined by the '==' operator.\"\"\"\n+    if first != second:\n         standard_msg = f'{first!r} != {second!r}'\n         msg = msg or standard_msg\n         raise AssertionError(msg)\n",
    "patch_ground_all": "\n--- a/srittau_python-asserts/asserts/__init__.py\n+++ b/srittau_python-asserts/asserts/__init__.py\n@@ -174,7 +174,7 @@\n \n     if isinstance(first, dict) and isinstance(second, dict):\n         assert_dict_equal(first, second, msg_fmt)\n-    elif (not first) == second:\n+    elif first != second:\n         msg = \"{!r} != {!r}\".format(first, second)\n         fail(msg_fmt.format(msg=msg, first=first, second=second))\n",
    "patch_ground_truth": "--- a/srittau_python-asserts/asserts/__init__.py\n+++ b/srittau_python-asserts/asserts/__init__.py\n@@ -174,7 +174,7 @@\n \n     if isinstance(first, dict) and isinstance(second, dict):\n         assert_dict_equal(first, second, msg_fmt)\n-    elif (not first) == second:\n+    elif not first == second:\n         msg = \"{!r} != {!r}\".format(first, second)\n         fail(msg_fmt.format(msg=msg, first=first, second=second))\n \n",
    "message": "======================================================================\nFAIL: test_assert_dict_equal__custom_value_message (test_asserts.AssertTest)\n----------------------------------------------------------------------\nTraceback (most recent call last):\n  File \"/home/user/Projects/user/repobench/repoben/python-asserts/srittau_python-asserts/test_asserts.py\", line 473, in test_assert_dict_equal__custom_value_message\n    assert_dict_equal(\n  File \"/home/user/Projects/user/repobench/repoben/python-asserts/srittau_python-asserts/test_asserts.py\", line 82, in __exit__\n    raise AssertionError(\"no AssertionError raised\")\nAssertionError: no AssertionError raised\n\n======================================================================\nFAIL: test_assert_dict_equal__values_do_not_match (test_asserts.AssertTest)\n----------------------------------------------------------------------\nTraceback (most recent call last):\n  File \"/home/user/Projects/user/repobench/repoben/python-asserts/srittau_python-asserts/test_asserts.py\", line 434, in test_assert_dict_equal__values_do_not_match\n    assert_dict_equal({\"foo\": 15}, {\"foo\": 10})\n  File \"/home/user/Projects/user/repobench/repoben/python-asserts/srittau_python-asserts/test_asserts.py\", line 82, in __exit__\n    raise AssertionError(\"no AssertionError raised\")\nAssertionError: no AssertionError raised\n\n======================================================================\nFAIL: test_assert_dict_superset__custom_value_message (test_asserts.AssertTest)\n----------------------------------------------------------------------\nTraceback (most recent call last):\n  File \"/home/user/Projects/user/repobench/repoben/python-asserts/srittau_python-asserts/test_asserts.py\", line 530, in test_assert_dict_superset__custom_value_message\n    assert_dict_superset(\n  File \"/home/user/Projects/user/repobench/repoben/python-asserts/srittau_python-asserts/test_asserts.py\", line 82, in __exit__\n    raise AssertionError(\"no AssertionError raised\")\nAssertionError: no AssertionError raised\n\n======================================================================\nFAIL: test_assert_dict_superset__values_do_not_match (test_asserts.AssertTest)\n----------------------------------------------------------------------\nTraceback (most recent call last):\n  File \"/home/user/Projects/user/repobench/repoben/python-asserts/srittau_python-asserts/test_asserts.py\", line 503, in test_assert_dict_superset__values_do_not_match\n    assert_dict_superset({\"foo\": 15}, {\"foo\": 10, \"bar\": 15})\n  File \"/home/user/Projects/user/repobench/repoben/python-asserts/srittau_python-asserts/test_asserts.py\", line 82, in __exit__\n    raise AssertionError(\"no AssertionError raised\")\nAssertionError: no AssertionError raised\n\n======================================================================\nFAIL: test_assert_equal__equal_objects (test_asserts.AssertTest)\n----------------------------------------------------------------------\nTraceback (most recent call last):\n  File \"/home/user/Projects/user/repobench/repoben/python-asserts/srittau_python-asserts/test_asserts.py\", line 197, in test_assert_equal__equal_objects\n    assert_equal(MyClass(), MyClass())\n  File \"/home/user/Projects/user/repobench/repoben/python-asserts/srittau_python-asserts/asserts/__init__.py\", line 179, in assert_equal\n    fail(msg_fmt.format(msg=msg, first=first, second=second))\n  File \"/home/user/Projects/user/repobench/repoben/python-asserts/srittau_python-asserts/asserts/__init__.py\", line 44, in fail\n    raise AssertionError(msg or \"assertion failure\")\nAssertionError: <test_asserts.AssertTest.test_assert_equal__equal_objects.<locals>.MyClass object at 0x7fc0cd981eb0> != <test_asserts.AssertTest.test_assert_equal__equal_objects.<locals>.MyClass object at 0x7fc0cda94550>\n\n======================================================================\nFAIL: test_assert_equal__not_equal__custom_message (test_asserts.AssertTest)\n----------------------------------------------------------------------\nTraceback (most recent call last):\n  File \"/home/user/Projects/user/repobench/repoben/python-asserts/srittau_python-asserts/test_asserts.py\", line 205, in test_assert_equal__not_equal__custom_message\n    assert_equal(\"string\", 55, \"{msg};{first!r};{second!r}\")\n  File \"/home/user/Projects/user/repobench/repoben/python-asserts/srittau_python-asserts/test_asserts.py\", line 82, in __exit__\n    raise AssertionError(\"no AssertionError raised\")\nAssertionError: no AssertionError raised\n\n======================================================================\nFAIL: test_assert_equal__not_equal__default_message (test_asserts.AssertTest)\n----------------------------------------------------------------------\nTraceback (most recent call last):\n  File \"/home/user/Projects/user/repobench/repoben/python-asserts/srittau_python-asserts/test_asserts.py\", line 201, in test_assert_equal__not_equal__default_message\n    assert_equal(\"string\", 55)\n  File \"/home/user/Projects/user/repobench/repoben/python-asserts/srittau_python-asserts/test_asserts.py\", line 82, in __exit__\n    raise AssertionError(\"no AssertionError raised\")\nAssertionError: no AssertionError raised",
    "CodeBase": [
      {
        "path": "srittau_python-asserts/asserts/__init__.py",
        "content": "1 \"\"\"\n2 Rich Assertions.\n3 \n4 This module contains several rich standard assertions that can be used in unit\n5 tests and in implementations. Users are encouraged to define their own\n6 assertions, possibly using assertions from this package as a basis.\n7 \n8     >>> assert_equal(13, 13)\n9     >>> assert_equal(13, 14)\n10     Traceback (most recent call last):\n11         ...\n12     AssertionError: 13 != 14\n13     >>> with assert_raises(KeyError):\n14     ...     raise KeyError()\n15     >>> with assert_raises(KeyError):\n16     ...     pass\n17     Traceback (most recent call last):\n18         ...\n19     AssertionError: KeyError not raised\n20 \n21 \"\"\"\n22 \n23 from __future__ import annotations\n24 \n25 import re\n26 import sys\n27 from datetime import datetime, timedelta, timezone\n28 from json import loads as json_loads\n29 from typing import Any, Callable, Set\n30 from warnings import WarningMessage, catch_warnings\n31 \n32 from typing_extensions import deprecated\n33 \n34 \n35 def fail(msg=None):\n36     \"\"\"Raise an AssertionError with the given message.\n37 \n38     >>> fail(\"my message\")\n39     Traceback (most recent call last):\n40         ...\n41     AssertionError: my message\n42 \n43     \"\"\"\n44     raise AssertionError(msg or \"assertion failure\")\n45 \n46 \n47 def assert_true(expr, msg_fmt=\"{msg}\"):\n48     \"\"\"Fail the test unless the expression is truthy.\n49 \n50     >>> assert_true(\"Hello World!\")\n51     >>> assert_true(\"\")\n52     Traceback (most recent call last):\n53         ...\n54     AssertionError: '' is not truthy\n55 \n56     The following msg_fmt arguments are supported:\n57     * msg - the default error message\n58     * expr - tested e(...truncated)"
      },
      {
        "path": "srittau_python-asserts/test_asserts.py",
        "content": "1 # -*- coding: utf-8 -*-\n2 \n3 import re\n4 import sys\n5 from collections import OrderedDict\n6 from datetime import datetime, timedelta, timezone\n7 from json import JSONDecodeError\n8 from unittest import TestCase\n9 from warnings import catch_warnings, simplefilter, warn\n10 \n11 from asserts import (\n12     Absent,\n13     Exists,\n14     Present,\n15     assert_almost_equal,\n16     assert_between,\n17     assert_boolean_false,\n18     assert_boolean_true,\n19     assert_count_equal,\n20     assert_datetime_about_now,\n21     assert_datetime_about_now_utc,\n22     assert_dict_equal,\n23     assert_dict_superset,\n24     assert_equal,\n25     assert_false,\n26     assert_greater,\n27     assert_greater_equal,\n28     assert_has_attr,\n29     assert_in,\n30     assert_is,\n31     assert_is_instance,\n32     assert_is_none,\n33     assert_is_not,\n34     assert_is_not_none,\n35     assert_json_subset,\n36     assert_less,\n37     assert_less_equal,\n38     assert_not_almost_equal,\n39     assert_not_equal,\n40     assert_not_in,\n41     assert_not_is_instance,\n42     assert_not_regex,\n43     assert_raises,\n44     assert_raises_errno,\n45     assert_raises_regex,\n46     assert_regex,\n47     assert_succee(...truncated)"
      },
      {
        "path": "srittau_python-asserts/CHANGELOG.md",
        "content": "1 # Changelog for python-asserts\n2 \n3 python-asserts adheres to [semantic versioning](https://semver.org/).\n4 \n5 ## UNRELEASED –\n6 \n7 ## [0.13.1] – 2024-04-29\n8 \n9 ### Fixed\n10 \n11 Fixed Python 3.12 deprecation warnings.\n12 \n13 ## [0.13.0] – 2024-03-13\n14 \n15 ### Added\n16 \n17 - Add support for Python 3.12.\n18 - Add `Present` and `Absent` for absence checks in `assert_json_subset()`.\n19 \n20 ### Removed\n21 \n22 - Drop support for Python 3.7.\n23 \n24 ### Deprecated\n25 \n26 - Deprecate `Exists` in favor of `Present` and `Absent` in\n27   `assert_json_subset()`.\n28 \n29 ## [0.12.0]\n30 \n31 ### Added\n32 \n33 - Add `assert_not_regex()`.\n34 \n35 ### Changed\n36 \n37 - Modernize the type stubs.\n38 \n39 ### Removed\n40 \n41 - Drop support for Python 3.6.\n42 \n43 ## [0.11.1]\n44 \n45 ### Added\n46 \n47 - `assert_json_subset()` can now check for the existence or non-existence\n48   of object members using(...truncated)"
      },
      {
        "path": "srittau_python-asserts/README.md",
        "content": "1 # Python Asserts\n2 \n3 [![License](https://img.shields.io/pypi/l/asserts.svg)](https://pypi.python.org/pypi/asserts/)\n4 [![PyPI - Python Version](https://img.shields.io/pypi/pyversions/asserts)](https://pypi.python.org/pypi/asserts/)\n5 [![GitHub](https://img.shields.io/github/release/srittau/python-asserts/all.svg)](https://github.com/srittau/python-asserts/releases/)\n6 [![pypi](https://img.shields.io/pypi/v/asserts.svg)](https://pypi.python.org/pypi/asserts/)\n7 (...truncated)"
      }
    ],
    "CommitSHA": "38849833a92ef24b342e0ca59a8fa5752a07bab4"
  },
  "Score": {
    "Difficulty": "Sample",
    "issue_origin": {
      "Title": 6,
      "Description": 6,
      "Reproducibility": 6,
      "Relevance": 6,
      "Explanation": 7,
      "Overall": 6
    },
    "issue_message": {
      "Title": 6,
      "Description": 6,
      "Reproducibility": 5,
      "Relevance": 8,
      "Explanation": 7,
      "Overall": 7
    },
    "issue_ground": {
      "Title": 8,
      "Description": 7,
      "Reproducibility": 5,
      "Relevance": 6,
      "Explanation": 6,
      "Overall": 7
    },
    "issue_ground_truth": {
      "title": "Equality Assertion Failure for Falsey Values",
      "description": "There appears to be an issue with the equality assertion logic when handling falsey values. Specifically, the code incorrectly determines that falsey values (such as an empty string or zero) equal other values when they are not, which can lead to incorrect test failures. This issue has been identified within the equality assertion functionality and needs to be resolved to ensure accurate assertion behavior for all types of values, including falsey ones. The failure message format should correctly distinguish between falsey values and other values during equality checks. This fix should ensure robust and reliable test assertions across various scenarios.",
      "explanation": "### Summary of the Issue\nThe issue at hand pertains to the incorrect behavior of the equality assertion logic when handling \"falsey\" values in Python. Falsey values are those that are considered to be false when evaluated in a boolean context, such as an empty string (`\"\"`), zero (`0`), `None`, `False`, etc. The problem arose from the system incorrectly equating a falsey value with other values, leading to incorrect assertions and potential test failures. Detailed assertion messages also needed clear differentiation between falsey values and other values during equality checks.\n\n### Content of the Commit\nHere is a summary of the content from the provided commit:\n\n- **Commit Message:** Improves equality assertions for falsey values to ensure they are correctly distinguished from other values during equality checks.\n- **Changed File:** The file modified is `asserts/__init__.py`, which contains the assertion logic.\n- **Modification:** A specific line in the equality assertion function changes the logic to correctly handle falsey values.\n\n### Detailed Explanation\n\n**Cause of the Issue:**\nThe root cause of the problem lies in the logic used to compare values for equality in the equality assertion function. The original comparison logic inadvertently treated falsey values like `0` and `\"\"` as equal to other values, which is incorrect in most assertion contexts. This caused test failures when actually the values being compared were not truly equal.\n\n**Commit Fix:**\nThe commit addressed this problem by refining the logic used for equality comparison. The change ensures that falsey values are not incorrectly deemed equal to other values simply because they both evaluate to `False` in a boolean context. Instead, the refined logic involves a more precise comparison that respects the distinct nature of falsey values and differentiates them from other non-equal values.\n\n**Solution Explained:**\nThe new logic introduced in the commit essentially separates the comparison operation from the boolean evaluation. Initially, the flawed logic checked the equality by inadvertently relying on the equivalence of their boolean representations (`(not first) == second`). This is problematic because different falsey values (like `0` and `\"\"`) both evaluate to `False` but are fundamentally different.\n\nThe corrected logic uses an improved direct comparison for equality that bypasses this pitfall. This means falsey values are only considered equal if they are indeed the same value (e.g., `0` is not equal to `False` or `\"\"`).\n\n### Summary of the Solution\nThe solution involved refining the logic within the equality assertion function to correctly handle falsey values. The modification ensures that falsey values are directly compared in a more precise manner, avoiding incorrect equivalences that led to test failures. The fix implemented in the commit carefully distinguishes between falsey values and other values during equality checks, thus preventing misleading assertion results in testing scripts. This approach leads to robust and reliable test assertions across various scenarios, improving the operational correctness of the testing framework."
    }
  }
}