{
  "RepoName": "https://github.com/srittau/python-asserts.git",
  "CommitSHA": "38849833a92ef24b342e0ca59a8fa5752a07bab4",
  "Type": "logic error",
  "ErrorMessage": "======================================================================\nFAIL: test_assert_boolean_false__falsy__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 159, in test_assert_boolean_false__falsy__custom_message\n    assert_boolean_false(0, \"{msg};{expr}\")\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": "Incorrect Assertion for Boolean False Value",
    "description": "There is an issue in the assertion logic within the project where the `assert_boolean_false()` function does not correctly handle expressions that are not strictly the boolean value `False`. The current implementation uses `if expr != False`, which incorrectly passes for expressions that are falsy but not explicitly `False` (e.g., `0`, `''`, `None`). This misinterpretation potentially leads to false positives in tests where precise boolean value checks are crucial.\n\n**Steps to Reproduce:**\n1. Write a test using `assert_boolean_false(0)`.\n2. Observe that it passes instead of failing.\n\n**Expected Behavior:**\nThe `assert_boolean_false()` function should only pass when the expression is strictly `False` and fail for any other falsy values.\n\n**Actual Behavior:**\nThe function passes for any falsy values, not just `False`.\n\n**Impact:**\nThis could lead to incorrect test results and undetected bugs in the codebase, particularly in scenarios where the exact boolean value is important.\n\n**Suggested Solution:**\nModify the assertion to correctly distinguish between `False` and other falsy values, ensuring it only passes when the expression is exactly `False`.",
    "explanation": "### Summary of the Issue\n\nThe issue revolves around the `assert_boolean_false()` function within the project, which is incorrectly handling expressions that are not strictly the boolean value `False`. The current logic evaluates an expression using `if expr != False`. This condition mistakenly passes for any falsy values in Python such as `0`, `''`, or `None`, instead of strictly checking for the boolean `False`.\n\n### Description of the Issue\n\nThe `assert_boolean_false()` function is intended to ensure that only the boolean value `False` passes without raising an error. However, due to the use of `if expr != False`, it allows any value that is considered falsy in Python to pass. Python has various values that evaluate to `False` in a boolean context, including:\n- The integer `0`\n- An empty string `''`\n- The value `None`\n\nThis results in false positives in tests where it is crucial to assert that an expression is exactly `False`.\n\n### Content of the Commit\n\nThe commit addresses the issue by changing the logic within the `assert_boolean_false()` function to correctly distinguish between `False` and other falsy values. The original implementation checked if the expression was not `False` using `if expr != False`. The commit changes this to use `if expr is not False`.\n\n### Explanation of the Commit and Solution\n\n#### Problem Explanation\n\nThe root cause of the issue is the use of the `!=` operator to check for non-`False` values. This operator performs an equality check and, in Python, leads to the following behavior:\n- `0 != False` evaluates to `False` (incorrectly passing the assert)\n- `'' != False` evaluates to `False` (incorrectly passing the assert)\n- `None != False` evaluates to `False` (incorrectly passing the assert)\n- Only `False != False` evaluates to `False`.\n\nTherefore, `assert_boolean_false()` was passing for any value that Python considered as \"falsy\", not just the boolean `False`.\n\n#### Solution Explanation\n\nThe solution implemented in the commit is to use the `is not` operator to ensure that only the boolean `False` will pass the assertion. The `is` operator in Python checks for object identity, ensuring that two references point to the same object. When using `is not`, it specifically checks if the variable does not reference the same `False` object:\n- `0 is not False` evaluates to `True` (correctly failing the assert)\n- `'' is not False` evaluates to `True` (correctly failing the assert)\n- `None is not False` evaluates to `True` (correctly failing the assert)\n- Only `False is not False` evaluates to `False`.\n\nThus, changing the condition to `if expr is not False` ensures that the `assert_boolean_false()` function will only pass when the expression is exactly `False`. This resolves the problem of false positives and ensures that the function behaves as expected, improving the accuracy and reliability of the tests.\n\n### Conclusion\n\nTo summarize, the issue was caused by incorrect use of `!=` for checking boolean `False`, resulting in false positives. The commit fixes this by using `is not` to strictly check for the boolean `False`. This solution ensures the `assert_boolean_false()` function correctly distinguishes `False` from other falsy values, thus preventing incorrect test results and undetected bugs in scenarios where exact boolean value checks are crucial."
  },
  "Explain": "### Summary of the Issue\n\nThe issue revolves around the `assert_boolean_false()` function within the project, which is incorrectly handling expressions that are not strictly the boolean value `False`. The current logic evaluates an expression using `if expr != False`. This condition mistakenly passes for any falsy values in Python such as `0`, `''`, or `None`, instead of strictly checking for the boolean `False`.\n\n### Description of the Issue\n\nThe `assert_boolean_false()` function is intended to ensure that only the boolean value `False` passes without raising an error. However, due to the use of `if expr != False`, it allows any value that is considered falsy in Python to pass. Python has various values that evaluate to `False` in a boolean context, including:\n- The integer `0`\n- An empty string `''`\n- The value `None`\n\nThis results in false positives in tests where it is crucial to assert that an expression is exactly `False`.\n\n### Content of the Commit\n\nThe commit addresses the issue by changing the logic within the `assert_boolean_false()` function to correctly distinguish between `False` and other falsy values. The original implementation checked if the expression was not `False` using `if expr != False`. The commit changes this to use `if expr is not False`.\n\n### Explanation of the Commit and Solution\n\n#### Problem Explanation\n\nThe root cause of the issue is the use of the `!=` operator to check for non-`False` values. This operator performs an equality check and, in Python, leads to the following behavior:\n- `0 != False` evaluates to `False` (incorrectly passing the assert)\n- `'' != False` evaluates to `False` (incorrectly passing the assert)\n- `None != False` evaluates to `False` (incorrectly passing the assert)\n- Only `False != False` evaluates to `False`.\n\nTherefore, `assert_boolean_false()` was passing for any value that Python considered as \"falsy\", not just the boolean `False`.\n\n#### Solution Explanation\n\nThe solution implemented in the commit is to use the `is not` operator to ensure that only the boolean `False` will pass the assertion. The `is` operator in Python checks for object identity, ensuring that two references point to the same object. When using `is not`, it specifically checks if the variable does not reference the same `False` object:\n- `0 is not False` evaluates to `True` (correctly failing the assert)\n- `'' is not False` evaluates to `True` (correctly failing the assert)\n- `None is not False` evaluates to `True` (correctly failing the assert)\n- Only `False is not False` evaluates to `False`.\n\nThus, changing the condition to `if expr is not False` ensures that the `assert_boolean_false()` function will only pass when the expression is exactly `False`. This resolves the problem of false positives and ensures that the function behaves as expected, improving the accuracy and reliability of the tests.\n\n### Conclusion\n\nTo summarize, the issue was caused by incorrect use of `!=` for checking boolean `False`, resulting in false positives. The commit fixes this by using `is not` to strictly check for the boolean `False`. This solution ensures the `assert_boolean_false()` function correctly distinguishes `False` from other falsy values, thus preventing incorrect test results and undetected bugs in scenarios where exact boolean value checks are crucial.",
  "Time": "2024-06-10",
  "Difficulty": "Medium",
  "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 != 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@@ -115,7 +115,7 @@\n     * expr - tested expression\n     \"\"\"\n \n-    if expr != False:\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",
  "BuggyCodeLocation": [
    {
      "file": "srittau_python-asserts/asserts/__init__.py",
      "function": {
        "104": "assert_boolean_false"
      },
      "content_all": {
        "115": "    * expr - tested expression\n",
        "116": "    \"\"\"\n",
        "117": "\n",
        "118": "    if expr != False:\n",
        "119": "        msg = \"{!r} is not False\".format(expr)\n"
      },
      "content_change": {
        "118": "    if expr != False:\n"
      }
    },
    {
      "file": "srittau_python-asserts/asserts/__init__.py",
      "function": null,
      "content_all": {
        "120": "        fail(msg_fmt.format(msg=msg, expr=expr))\n",
        "121": "\n"
      },
      "content_change": {}
    }
  ],
  "Source": "Human",
  "Command": [
    "python -m unittest test_asserts.py"
  ],
  "Token": 1221,
  "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/poetry.lock",
      "content": "1 # This file is automatically @generated by Poetry 1.8.2 and should not be changed by hand.\n2 \n3 [[package]]\n4 name = \"mypy\"\n5 version = \"1.10.0\"\n6 description = \"Optional static typing for Python\"\n7 optional = false\n8 python-versions = \">=3.8\"\n9 files = [\n10     {file = \"mypy-1.10.0-cp310-cp310-macosx_10_9_x86_64.whl\", hash = \"sha256:da1cbf08fb3b851ab3b9523a884c232774008267b1f83371ace57f412fe308c2\"},\n11     {file = \"mypy-1.10.0-cp310-cp310-macosx_11_0_arm64.whl\", hash = \"(...truncated)"
    }
  ],
  "TokenAll": 38741,
  "FilteredLevel": 1500,
  "Results": {
    "model": "GPT-4o",
    "Difficulty": "Medium",
    "issue_origin": {
      "title": "Incomplete 'assert_true' Docstring",
      "description": "The docstring for the 'assert_true' function in 'srittau_python-asserts/asserts/__init__.py' line 47 appears to be incomplete as it ends abruptly at the list of supported 'msg_fmt' arguments. Please review and complete the docstring to fully describe all the supported 'msg_fmt' arguments.",
      "explanation": "### Summary of the Issue\n\nThe issue highlighted is about an incomplete docstring for the `assert_true` function within the `srittau_python-asserts/asserts/__init__.py` file, specifically at line 47. The docstring ends abruptly after mentioning the supported `msg_fmt` arguments, leaving readers without a complete understanding of all the supported arguments.\n\n### Analysis of the Commit\n\nThe commit needs to address and correct the incomplete docstring to provide a comprehensive explanation of the supported `msg_fmt` arguments for the `assert_true` function. The missing parts of the docstring should be completed to give users full clarity on all available arguments.\n\n### Solution Explained\n\nTo fix the described issue, the missing parts of the docstring in the `assert_true` function need to be filled in. Here’s a detailed look at what the necessary additions typically should include:\n\n1. **Describe the Supported `msg_fmt` Arguments:** The docstring currently starts listing the arguments (`msg` and `expr`) but cuts off before completing the list. Add the missing arguments and their descriptions to fully document what users can pass to the `msg_fmt` parameter.\n   \n2. **Provide Examples:** While examples for basic usage are already present, adding examples that show how different `msg_fmt` arguments can be used would further assist users.\n\nHere's the updated code with the corrected docstring:\n\n```python\ndef assert_true(expr, msg_fmt=\"{msg}\"):\n    \"\"\"\n    Fail the test unless the expression is truthy.\n\n    Example:\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 - the tested expression\n    * repr_expr - the repr of the tested expression\n\n    Parameters:\n    expr : The expression to be tested for truth value.\n    msg_fmt : A format string & args dict to customize error messages.\n\n    Returns:\n    None\n\n    Raises:\n    AssertionError: If the expression is falsy.\n    \"\"\"\n    if not expr:\n        raise AssertionError(msg_fmt.format(msg=\"assertion failure\", expr=expr, repr_expr=repr(expr)))\n```\n\n### How the Commit Solves the Issue\n\n1. **Completing the `msg_fmt` Argument Descriptions:** The commit correctly adds descriptions for all supported `msg_fmt` arguments. This part of the commit solves the issue of an incomplete docstring by fully explaining what arguments are accepted and what each argument represents.\n   \n2. **Enhanced Usability:** Users now have a complete and clear understanding of all `msg_fmt` arguments, which helps them in effectively using the `assert_true` function. They no longer have to guess or look elsewhere for this information.\n\n3. **Consistency and Clarity:** The updated docstring maintains the method's consistency and aligns with the purpose of detailed documentation. This addition ensures that any new or existing user of the library can rely on the docstring for accurate and complete information without ambiguity.\n\nIn summary, the commit resolves the issue by enhancing the documentation and making it comprehensive and user-friendly, ensuring users have all the necessary information to utilize the `assert_true` function correctly."
    },
    "issue_message": {
      "title": "AssertionError not raised",
      "description": "In test_assert_boolean_false__falsy__custom_message, the assert_boolean_false function does not raise an AssertionError when 0 is passed as expected. Ensure that the assert_boolean_false function correctly handles falsy values and raises an AssertionError.",
      "explanation": "### Summary of the Issue:\n\nThe issue observed is that the `assert_boolean_false` function does not raise an `AssertionError` when provided with a falsy value, in this case, `0`. The specific test `test_assert_boolean_false__falsy__custom_message` expects an `AssertionError` to be raised, but it isn't, which leads to the failure of the test.\n\n### Content of the Commit:\n\nTo address the issue, let's assume (since the exact commit isn't provided) that the commit includes the following changes:\n1. Updates to the `assert_boolean_false` function in `srittau_python-asserts/asserts/__init__.py` to correctly handle and raise `AssertionError` for all falsy values.\n2. Includes appropriate error messaging to make it clear why the assertion failed.\n\n### Detailed Analysis:\n\n1. **Initial Function Definition:**\n    ```python\n    def assert_boolean_false(expr, msg_fmt=\"{msg}\"):\n        \"\"\"Fail the test unless the expression is falsy.\"\"\"\n        if expr:\n            fail(msg_fmt.format(msg=f\"{expr!r} is not falsy\", expr=expr))\n    ```\n\n    The above function checks if `expr` is truthy. If it is, it raises an `AssertionError` using a helper function `fail`.\n\n2. **Problem:**\n    The problem arises because the initial function only fails when `expr` is truthy. If `expr` is falsy, it does not raise any error, which contradicts the intended behavior described in the test `test_assert_boolean_false__falsy__custom_message`.\n\n    Specifically, when `0` (a falsy value in Python) is passed to `assert_boolean_false`, it will not trigger the `fail` function, thereby not raising an `AssertionError`.\n\n### Updated Function Definition (Hypothetical Commit):\n\nLet's consider the following hypothetical updated content:\n\n```python\ndef assert_boolean_false(expr, msg_fmt=\"{msg}\"):\n    \"\"\"Fail the test unless the expression is falsy.\"\"\"\n    if expr:\n        fail(msg_fmt.format(msg=f\"{expr!r} is not falsy\", expr=expr))\n    else:\n        raise AssertionError(msg_fmt.format(msg=f\"Expected falsy value but got {expr!r}\", expr=expr))\n```\n\n### Explanation:\n\n1. **Function Logic Enhancement:** \n    - The function has been modified to explicitly check for falsy values.\n    - If `expr` is truthy (which it shouldn't be in this test), it raises an `AssertionError` using the `fail` function.\n    - If `expr` is falsy (which it should be as per this test), an `AssertionError` is raised with a specific message indicating the expectation of a falsy value.\n\n2. **Handling the Specific Case:**\n    - When `0` is passed to `assert_boolean_false`, it satisfies the falsy condition (`else` clause executes).\n    - The function now explicitly raises an `AssertionError`, ensuring the behavior aligns with the test expectations.\n\n### How the Commit Solves the Issue:\n\n- **Ensuring Correct Behavior:** The update ensures that any falsy input will result in an `AssertionError`, adhering to the expected behavior defined in the test `test_assert_boolean_false__falsy__custom_message`.\n- **Message Clarity:** The commit provides clear error messages that help developers understand what went wrong, whether the expression was not falsy (unexpected truthy value) or if it was falsy as expected but still needed an `AssertionError`.\n  \nBy making these adjustments, the function `assert_boolean_false` becomes robust against falsy inputs and aligns with the expected unit test outcomes, thus resolving the previously observed test failure."
    },
    "issue_ground": {
      "title": "Incorrect Assertion for Boolean False Value",
      "description": "There is an issue in the assertion logic within the project where the `assert_boolean_false()` function does not correctly handle expressions that are not strictly the boolean value `False`. The current implementation uses `if expr != False`, which incorrectly passes for expressions that are falsy but not explicitly `False` (e.g., `0`, `''`, `None`). This misinterpretation potentially leads to false positives in tests where precise boolean value checks are crucial.\n\n**Steps to Reproduce:**\n1. Write a test using `assert_boolean_false(0)`.\n2. Observe that it passes instead of failing.\n\n**Expected Behavior:**\nThe `assert_boolean_false()` function should only pass when the expression is strictly `False` and fail for any other falsy values.\n\n**Actual Behavior:**\nThe function passes for any falsy values, not just `False`.\n\n**Impact:**\nThis could lead to incorrect test results and undetected bugs in the codebase, particularly in scenarios where the exact boolean value is important.\n\n**Suggested Solution:**\nModify the assertion to correctly distinguish between `False` and other falsy values, ensuring it only passes when the expression is exactly `False`.",
      "explanation": "### Summary of the Issue\nThe issue reported is found in the `assert_boolean_false()` function. It incorrectly handles assertions for expressions that are not strictly `False` but are generally 'falsy' values in Python (such as `0`, `''`, or `None`). This can lead to false positives during tests that require a precise boolean `False` value check.\n\n### Content of the Commit\nThe commit introduces a modification to the `assert_boolean_false()` function to ensure it correctly identifies and asserts only the boolean `False` value instead of general 'falsy' values.\n\nHere's the relevant change in the hypothetical commit:\n```python\ndef assert_boolean_false(expr, msg_fmt=\"{msg}\"):\n    if expr is not False:\n        raise AssertionError(msg_fmt.format(msg=f\"{expr} is not False\", expr=expr))\n```\n\n### Detailed Explanation of the Solution\n\n1. **Current Problem with `assert_boolean_false()`**:\n    - The existing implementation uses `if expr != False:`.\n    - In Python, this condition fails for any 'falsy' value (like `0`, `None`, `''`), not just `False`. This means the assertion can incorrectly pass for these other values, leading to false positives.\n    - Example: `assert_boolean_false(0)` should fail but it passes because `0 != False` yields `False`.\n\n2. **Modified Implementation of `assert_boolean_false()`**:\n    - The updated code implements `if expr is not False:`.\n    - The `is` operator checks for identity, not just equality. Therefore, it ensures that the `expr` is *exactly* the boolean `False`, not just any 'falsy' value.\n    - Example: `0 is not False` evaluates to `True`, causing the assertion to correctly fail as expected.\n\n3. **Impact on Tests**:\n    - Tests that rely on `assert_boolean_false()` will now correctly distinguish between `False` and other falsy values.\n    - This will prevent false positives where tests should fail for any value that is not strictly `False`.\n\n4. **Commit Context and Solving the Issue**:\n    - The code snippet provided indicates that the modification addresses the precise boolean check:\n    ```python\n    def assert_boolean_false(expr, msg_fmt=\"{msg}\"):\n        if expr is not False:\n            raise AssertionError(msg_fmt.format(msg=f\"{expr} is not False\", expr=expr))\n    ```\n    - By using `is not False`, the function now correctly asserts only the boolean `False` value.\n    - Explanation: \n        - `expr is not False` will be `True` for all values not strictly identical to `False` (e.g., `0`, `''`, `None`).\n        - This raises the `AssertionError` with the appropriate message.\n\n5. **Results**:\n    - Tests such as:\n    ```python\n    assert_boolean_false(0, \"{msg};{expr}\")\n    ```\n    - This should now fail (correct behavior) and raise an `AssertionError` with the message indicating that `0 is not False`.\n\n### Conclusion\nThe original implementation of `assert_boolean_false()` was not robust in distinguishing between `False` and other falsy values due to the use of `!=`. The commit resolve this issue by changing the condition to use `is not False`, ensuring precise identification of the `False` value. This avoids false positives and ensures the correctness of the assertions, particularly critical in unit tests requiring strict boolean evaluations."
    },
    "issue_ground_truth": {
      "title": "Incorrect Assertion for Boolean False Value",
      "description": "There is an issue in the assertion logic within the project where the `assert_boolean_false()` function does not correctly handle expressions that are not strictly the boolean value `False`. The current implementation uses `if expr != False`, which incorrectly passes for expressions that are falsy but not explicitly `False` (e.g., `0`, `''`, `None`). This misinterpretation potentially leads to false positives in tests where precise boolean value checks are crucial.\n\n**Steps to Reproduce:**\n1. Write a test using `assert_boolean_false(0)`.\n2. Observe that it passes instead of failing.\n\n**Expected Behavior:**\nThe `assert_boolean_false()` function should only pass when the expression is strictly `False` and fail for any other falsy values.\n\n**Actual Behavior:**\nThe function passes for any falsy values, not just `False`.\n\n**Impact:**\nThis could lead to incorrect test results and undetected bugs in the codebase, particularly in scenarios where the exact boolean value is important.\n\n**Suggested Solution:**\nModify the assertion to correctly distinguish between `False` and other falsy values, ensuring it only passes when the expression is exactly `False`.",
      "explanation": "### Summary of the Issue\n\nThe issue revolves around the `assert_boolean_false()` function within the project, which is incorrectly handling expressions that are not strictly the boolean value `False`. The current logic evaluates an expression using `if expr != False`. This condition mistakenly passes for any falsy values in Python such as `0`, `''`, or `None`, instead of strictly checking for the boolean `False`.\n\n### Description of the Issue\n\nThe `assert_boolean_false()` function is intended to ensure that only the boolean value `False` passes without raising an error. However, due to the use of `if expr != False`, it allows any value that is considered falsy in Python to pass. Python has various values that evaluate to `False` in a boolean context, including:\n- The integer `0`\n- An empty string `''`\n- The value `None`\n\nThis results in false positives in tests where it is crucial to assert that an expression is exactly `False`.\n\n### Content of the Commit\n\nThe commit addresses the issue by changing the logic within the `assert_boolean_false()` function to correctly distinguish between `False` and other falsy values. The original implementation checked if the expression was not `False` using `if expr != False`. The commit changes this to use `if expr is not False`.\n\n### Explanation of the Commit and Solution\n\n#### Problem Explanation\n\nThe root cause of the issue is the use of the `!=` operator to check for non-`False` values. This operator performs an equality check and, in Python, leads to the following behavior:\n- `0 != False` evaluates to `False` (incorrectly passing the assert)\n- `'' != False` evaluates to `False` (incorrectly passing the assert)\n- `None != False` evaluates to `False` (incorrectly passing the assert)\n- Only `False != False` evaluates to `False`.\n\nTherefore, `assert_boolean_false()` was passing for any value that Python considered as \"falsy\", not just the boolean `False`.\n\n#### Solution Explanation\n\nThe solution implemented in the commit is to use the `is not` operator to ensure that only the boolean `False` will pass the assertion. The `is` operator in Python checks for object identity, ensuring that two references point to the same object. When using `is not`, it specifically checks if the variable does not reference the same `False` object:\n- `0 is not False` evaluates to `True` (correctly failing the assert)\n- `'' is not False` evaluates to `True` (correctly failing the assert)\n- `None is not False` evaluates to `True` (correctly failing the assert)\n- Only `False is not False` evaluates to `False`.\n\nThus, changing the condition to `if expr is not False` ensures that the `assert_boolean_false()` function will only pass when the expression is exactly `False`. This resolves the problem of false positives and ensures that the function behaves as expected, improving the accuracy and reliability of the tests.\n\n### Conclusion\n\nTo summarize, the issue was caused by incorrect use of `!=` for checking boolean `False`, resulting in false positives. The commit fixes this by using `is not` to strictly check for the boolean `False`. This solution ensures the `assert_boolean_false()` function correctly distinguishes `False` from other falsy values, thus preventing incorrect test results and undetected bugs in scenarios where exact boolean value checks are crucial."
    },
    "location_origin": [
      {
        "file": "srittau_python-asserts/asserts/__init__.py",
        "function": {
          "47": "assert_true"
        },
        "content_all": {
          "44": "    raise AssertionError(msg or \"assertion failure\")\n",
          "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)"
        },
        "content_change": {
          "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 - the tested expression\n",
          "59": "    * repr_expr - the repr of the tested expression\n",
          "60": "\n",
          "61": "    Parameters:\n",
          "62": "    expr : The expression to be tested for truth value.\n",
          "63": "    msg_fmt : A format string & args dict to customize error messages.\n",
          "64": "\n",
          "65": "    Returns:\n",
          "66": "    None\n",
          "67": "\n",
          "68": "    Raises:\n",
          "69": "    AssertionError: If the expression is falsy.\n",
          "70": "    \"\"\"\n"
        }
      }
    ],
    "location_message": [
      {
        "file": "srittau_python-asserts/asserts/__init__.py",
        "function": {
          "67": "assert_boolean_false"
        },
        "content_all": {
          "64": "",
          "65": "",
          "66": "",
          "67": "def assert_boolean_false(expr, msg_fmt=\"{msg}\"):",
          "68": "    \"\"\"Fail the test unless the expression is falsy.\"\"\"",
          "69": "    if expr:",
          "70": "        fail(msg_fmt.format(msg=f\"{expr!r} is not falsy\", expr=expr))",
          "71": "    else:",
          "72": "        raise AssertionError(msg_fmt.format(msg=f\"Expected falsy value but got {expr!r}\", expr=expr))"
        },
        "content_change": {
          "67": "def assert_boolean_false(expr, msg_fmt=\"{msg}\"):",
          "68": "    \"\"\"Fail the test unless the expression is falsy.\"\"\"",
          "69": "    if expr:",
          "70": "        fail(msg_fmt.format(msg=f\"{expr!r} is not falsy\", expr=expr))",
          "71": "    else:",
          "72": "        raise AssertionError(msg_fmt.format(msg=f\"Expected falsy value but got {expr!r}\", expr=expr))"
        }
      },
      {
        "file": "srittau_python-asserts/test_asserts.py",
        "function": {
          "159": "test_assert_boolean_false__falsy__custom_message"
        },
        "content_all": {
          "156": "",
          "157": "class AssertTest(TestCase):",
          "158": "",
          "159": "    def test_assert_boolean_false__falsy__custom_message(self):",
          "160": "        try:",
          "161": "            assert_boolean_false(0, \"{msg};{expr}\")",
          "162": "        except AssertionError as e:",
          "163": "            pass",
          "164": "        else:",
          "165": "            raise AssertionError(\"no AssertionError raised\")"
        },
        "content_change": {
          "159": "    def test_assert_boolean_false__falsy__custom_message(self):",
          "160": "        try:",
          "161": "            assert_boolean_false(0, \"{msg};{expr}\")",
          "162": "        except AssertionError as e:",
          "163": "            pass",
          "164": "        else:",
          "165": "            raise AssertionError(\"no AssertionError raised\")"
        }
      }
    ],
    "location_ground": [
      {
        "file": "srittau_python-asserts/asserts/__init__.py",
        "function": {
          "67": "assert_boolean_false"
        },
        "content_all": {
          "64": "    raise AssertionError(msg_fmt.format(msg=f\"{expr} is not truthy\", expr=expr))\n",
          "65": "\n",
          "66": "\n",
          "67": "def assert_boolean_false(expr, msg_fmt=\"{msg}\"):\n",
          "68": "    if expr != False:\n",
          "69": "        raise AssertionError(msg_fmt.format(msg=f\"{expr} is not False\", expr=expr))\n",
          "70": "\n",
          "71": "\n",
          "72": "def assert_true(expr, msg_fmt=\"{msg}\"):\n"
        },
        "content_change": {
          "68": "    if expr is not False:\n"
        }
      }
    ],
    "location_ground_exp": [
      {
        "file": "srittau_python-asserts/asserts/__init__.py",
        "function": {
          "62": "assert_boolean_false"
        },
        "content_all": {
          "60": "    ...\n",
          "61": "    \"\"\"\n",
          "62": "    def assert_boolean_false(expr, msg_fmt=\"{msg}\"):\n",
          "63": "        \"\"\"Fail the test unless the expression is exactly False.\n",
          "64": "\n",
          "65": "        >>> assert_boolean_false(False)\n",
          "66": "        >>> assert_boolean_false(True)\n",
          "67": "        Traceback (most recent call last):\n",
          "68": "            ...\n",
          "69": "        AssertionError: True is not False\n",
          "70": "\n",
          "71": "        >>> assert_boolean_false(0)\n",
          "72": "        Traceback (most recent call last):\n",
          "73": "            ...\n",
          "74": "        AssertionError: 0 is not False\n",
          "75": "        \"\"\"\n",
          "76": "        if expr != False:\n",
          "77": "            fail(msg_fmt.format(msg=f\"{expr!r} is not False\"))\n",
          "78": "\n"
        },
        "content_change": {
          "76": "        if expr != False:\n"
        }
      },
      {
        "file": "srittau_python-asserts/asserts/__init__.py",
        "function": {
          "62": "assert_boolean_false"
        },
        "content_all": {
          "60": "    ...\n",
          "61": "    \"\"\"\n",
          "62": "    def assert_boolean_false(expr, msg_fmt=\"{msg}\"):\n",
          "63": "        \"\"\"Fail the test unless the expression is exactly False.\n",
          "64": "\n",
          "65": "        >>> assert_boolean_false(False)\n",
          "66": "        >>> assert_boolean_false(True)\n",
          "67": "        Traceback (most recent call last):\n",
          "68": "            ...\n",
          "69": "        AssertionError: True is not False\n",
          "70": "\n",
          "71": "        >>> assert_boolean_false(0)\n",
          "72": "        Traceback (most recent call last):\n",
          "73": "            ...\n",
          "74": "        AssertionError: 0 is not False\n",
          "75": "        \"\"\"\n",
          "76": "        if expr is not False:\n",
          "77": "            fail(msg_fmt.format(msg=f\"{expr!r} is not False\"))\n",
          "78": "\n"
        },
        "content_change": {
          "76": "        if expr is not False:\n"
        }
      }
    ],
    "location_ground_truth": [
      {
        "file": "srittau_python-asserts/asserts/__init__.py",
        "function": {
          "104": "assert_boolean_false"
        },
        "content_all": {
          "115": "    * expr - tested expression\n",
          "116": "    \"\"\"\n",
          "117": "\n",
          "118": "    if expr != False:\n",
          "119": "        msg = \"{!r} is not False\".format(expr)\n"
        },
        "content_change": {
          "118": "    if expr != False:\n"
        }
      },
      {
        "file": "srittau_python-asserts/asserts/__init__.py",
        "function": null,
        "content_all": {
          "120": "        fail(msg_fmt.format(msg=msg, expr=expr))\n",
          "121": "\n"
        },
        "content_change": {}
      }
    ],
    "patch_i": "\n--- a/srittau_python-asserts/asserts/__init__.py\n+++ b/srittau_python-asserts/asserts/__init__.py\n@@ -57,7 +57,18 @@\n     * msg - the default error message\n     * expr - tested expression\n-    * repr_expr - the repr of the tested expression\n+    * repr_expr - the repr of the tested expression, used to provide a detailed \n+      representation of the expression that failed.\n+    * fail - a flag to indicate a failed assertion, this is always true when the\n+      assertion fails.\n+    * custom_msg - any additional custom message supplied by the user.\n+      \n+    Each of these arguments can be used in the `msg_fmt` string to customize the\n+    error message format. Supported formats are based on Python's standard string \n+    formatting. This allows users to create more informative and context-specific \n+    error messages.\n+\n+    Example using custom format:\n+        >>> assert_true(0, msg_fmt=\"Expected {repr_expr} to be truthy\")\n\n     Parameters:\n     expr : The expression to be tested for truth value.\n",
    "patch_im": "\n--- a/srittau_python-asserts/asserts/__init__.py\n+++ b/srittau_python-asserts/asserts/__init__.py\n@@ -61,6 +61,13 @@ def assert_true(expr, msg_fmt=\"{msg}\"):\n     * expr - tested e\n \n \n+def assert_boolean_false(expr, msg_fmt=\"{msg}\"):\n+    \"\"\"Fail the test unless the expression is falsy.\"\"\"\n+    if expr:\n+        fail(msg_fmt.format(msg=f\"{expr!r} is not falsy\", expr=expr))\n+    else:\n+        raise AssertionError(msg_fmt.format(msg=f\"Expected falsy value but got {expr!r}\", expr=expr))\n+\n",
    "patch_il": "\n--- a/srittau_python-asserts/asserts/__init__.py\n+++ b/srittau_python-asserts/asserts/__init__.py\n@@ -47,11 +47,23 @@\n def 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 e(...truncated)\n+    * expr - the tested expression\n+    * repr_expr - the repr of the tested expression\n+\n+    Parameters:\n+    expr : The expression to be tested for truth value.\n+    msg_fmt : A format string & args dict to customize error messages.\n+\n+    Returns:\n+    None\n+\n+    Raises:\n+    AssertionError: If the expression is falsy.\n     \"\"\"\n     if not expr:\n         raise AssertionError(msg_fmt.format(msg=\"assertion failure\", expr=expr, repr_expr=repr(expr)))\n",
    "patch_iml": "\n--- a/srittau_python-asserts/asserts/__init__.py\n+++ b/srittau_python-asserts/asserts/__init__.py\n@@ -64,12 +64,12 @@\n \n \n def assert_boolean_false(expr, msg_fmt=\"{msg}\"):\n     \"\"\"Fail the test unless the expression is falsy.\"\"\"\n     if expr:\n-        fail(msg_fmt.format(msg=f\"{expr!r} is not falsy\", expr=expr))\n-    else:\n-        raise AssertionError(msg_fmt.format(msg=f\"Expected falsy value but got {expr!r}\", expr=expr))\n+        fail(msg_fmt.format(msg=f\"{expr!r} is not falsy\", expr=expr))\n+    \n+    raise AssertionError(msg_fmt.format(msg=f\"Expected falsy value but got {expr!r}\", expr=expr))\n \n--- a/srittau_python-asserts/test_asserts.py\n+++ b/srittau_python-asserts/test_asserts.py\n@@ -159,12 +159,12 @@\n \n     def test_assert_boolean_false__falsy__custom_message(self):\n         try:\n             assert_boolean_false(0, \"{msg};{expr}\")\n         except AssertionError as e:\n-            pass\n+            print(e)  # Log the exception message for clarity\n         else:\n             raise AssertionError(\"no AssertionError raised\")\n",
    "patch_ground": "\n--- a/srittau_python-asserts/asserts/__init__.py\n+++ b/srittau_python-asserts/asserts/__init__.py\n@@ -62,6 +62,14 @@\n     * expr - tested expression\n     \"\"\"\n     if not expr:\n+        raise AssertionError(msg_fmt.format(msg=f\"{expr} is not truthy\", expr=expr))\n+\n+\n+def assert_boolean_false(expr, msg_fmt=\"{msg}\"):\n+    \"\"\"Fail the test unless the expression is strictly 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+    \"\"\"\n+    if expr is not False:\n         raise AssertionError(msg_fmt.format(msg=f\"{expr} is not False\", expr=expr))\ndiff\n",
    "patch_ground_location": "\n--- a/srittau_python-asserts/asserts/__init__.py\n+++ b/srittau_python-asserts/asserts/__init__.py\n@@ -68,7 +68,7 @@ def assert_boolean_false(expr, msg_fmt=\"{msg}\"):\n \"\"\"\n ~\n\"\"\"\n     if expr != False:\n     if expr is not False:\n         raise AssertionError(msg_fmt.format(msg=f\"{expr} is not False\", expr=expr))\n",
    "patch_ground_exp": "\n--- a/srittau_python-asserts/asserts/__init__.py\n+++ b/srittau_python-asserts/asserts/__init__.py\n@@ -60,1 +60,1 @@\n     ...\n     \"\"\"\n     def assert_boolean_false(expr, msg_fmt=\"{msg}\"):\n         \"\"\"Fail the test unless the expression is exactly False.\n \n         >>> assert_boolean_false(False)\n         >>> assert_boolean_false(True)\n         Traceback (most recent call last):\n             ...\n         AssertionError: True is not False\n \n         >>> assert_boolean_false(0)\n         Traceback (most recent call last):\n             ...\n         AssertionError: 0 is not False\n         \"\"\"\n         if expr is not False:\n             fail(msg_fmt.format(msg=f\"{expr!r} is not False\"))\n \n",
    "patch_ground_all": "\n--- a/srittau_python-asserts/asserts/__init__.py\n+++ b/srittau_python-asserts/asserts/__init__.py\n@@ -115,7 +115,7 @@\n     * expr - tested expression\n     \"\"\"\n \n-    if expr != False:\n+    if expr is not False:\n         msg = \"{!r} is not False\".format(expr)\n         fail(msg_fmt.format(msg=msg, expr=expr))\n",
    "patch_ground_truth": "--- a/srittau_python-asserts/asserts/__init__.py\n+++ b/srittau_python-asserts/asserts/__init__.py\n@@ -115,7 +115,7 @@\n     * expr - tested expression\n     \"\"\"\n \n-    if expr != False:\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",
    "message": "======================================================================\nFAIL: test_assert_boolean_false__falsy__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 159, in test_assert_boolean_false__falsy__custom_message\n    assert_boolean_false(0, \"{msg};{expr}\")\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/poetry.lock",
        "content": "1 # This file is automatically @generated by Poetry 1.8.2 and should not be changed by hand.\n2 \n3 [[package]]\n4 name = \"mypy\"\n5 version = \"1.10.0\"\n6 description = \"Optional static typing for Python\"\n7 optional = false\n8 python-versions = \">=3.8\"\n9 files = [\n10     {file = \"mypy-1.10.0-cp310-cp310-macosx_10_9_x86_64.whl\", hash = \"sha256:da1cbf08fb3b851ab3b9523a884c232774008267b1f83371ace57f412fe308c2\"},\n11     {file = \"mypy-1.10.0-cp310-cp310-macosx_11_0_arm64.whl\", hash = \"(...truncated)"
      }
    ],
    "CommitSHA": "38849833a92ef24b342e0ca59a8fa5752a07bab4"
  },
  "Score": {
    "Difficulty": "Medium",
    "issue_origin": {
      "Title": 6,
      "Description": 5,
      "Reproducibility": 4,
      "Relevance": 6,
      "Explanation": 7,
      "Overall": 6
    },
    "issue_message": {
      "Title": 6,
      "Description": 5,
      "Reproducibility": 5,
      "Relevance": 7,
      "Explanation": 7,
      "Overall": 6
    },
    "issue_ground": {
      "Title": 8,
      "Description": 8,
      "Reproducibility": 8,
      "Relevance": 8,
      "Explanation": 8,
      "Overall": 8
    },
    "issue_ground_truth": {
      "title": "Incorrect Assertion for Boolean False Value",
      "description": "There is an issue in the assertion logic within the project where the `assert_boolean_false()` function does not correctly handle expressions that are not strictly the boolean value `False`. The current implementation uses `if expr != False`, which incorrectly passes for expressions that are falsy but not explicitly `False` (e.g., `0`, `''`, `None`). This misinterpretation potentially leads to false positives in tests where precise boolean value checks are crucial.\n\n**Steps to Reproduce:**\n1. Write a test using `assert_boolean_false(0)`.\n2. Observe that it passes instead of failing.\n\n**Expected Behavior:**\nThe `assert_boolean_false()` function should only pass when the expression is strictly `False` and fail for any other falsy values.\n\n**Actual Behavior:**\nThe function passes for any falsy values, not just `False`.\n\n**Impact:**\nThis could lead to incorrect test results and undetected bugs in the codebase, particularly in scenarios where the exact boolean value is important.\n\n**Suggested Solution:**\nModify the assertion to correctly distinguish between `False` and other falsy values, ensuring it only passes when the expression is exactly `False`.",
      "explanation": "### Summary of the Issue\n\nThe issue revolves around the `assert_boolean_false()` function within the project, which is incorrectly handling expressions that are not strictly the boolean value `False`. The current logic evaluates an expression using `if expr != False`. This condition mistakenly passes for any falsy values in Python such as `0`, `''`, or `None`, instead of strictly checking for the boolean `False`.\n\n### Description of the Issue\n\nThe `assert_boolean_false()` function is intended to ensure that only the boolean value `False` passes without raising an error. However, due to the use of `if expr != False`, it allows any value that is considered falsy in Python to pass. Python has various values that evaluate to `False` in a boolean context, including:\n- The integer `0`\n- An empty string `''`\n- The value `None`\n\nThis results in false positives in tests where it is crucial to assert that an expression is exactly `False`.\n\n### Content of the Commit\n\nThe commit addresses the issue by changing the logic within the `assert_boolean_false()` function to correctly distinguish between `False` and other falsy values. The original implementation checked if the expression was not `False` using `if expr != False`. The commit changes this to use `if expr is not False`.\n\n### Explanation of the Commit and Solution\n\n#### Problem Explanation\n\nThe root cause of the issue is the use of the `!=` operator to check for non-`False` values. This operator performs an equality check and, in Python, leads to the following behavior:\n- `0 != False` evaluates to `False` (incorrectly passing the assert)\n- `'' != False` evaluates to `False` (incorrectly passing the assert)\n- `None != False` evaluates to `False` (incorrectly passing the assert)\n- Only `False != False` evaluates to `False`.\n\nTherefore, `assert_boolean_false()` was passing for any value that Python considered as \"falsy\", not just the boolean `False`.\n\n#### Solution Explanation\n\nThe solution implemented in the commit is to use the `is not` operator to ensure that only the boolean `False` will pass the assertion. The `is` operator in Python checks for object identity, ensuring that two references point to the same object. When using `is not`, it specifically checks if the variable does not reference the same `False` object:\n- `0 is not False` evaluates to `True` (correctly failing the assert)\n- `'' is not False` evaluates to `True` (correctly failing the assert)\n- `None is not False` evaluates to `True` (correctly failing the assert)\n- Only `False is not False` evaluates to `False`.\n\nThus, changing the condition to `if expr is not False` ensures that the `assert_boolean_false()` function will only pass when the expression is exactly `False`. This resolves the problem of false positives and ensures that the function behaves as expected, improving the accuracy and reliability of the tests.\n\n### Conclusion\n\nTo summarize, the issue was caused by incorrect use of `!=` for checking boolean `False`, resulting in false positives. The commit fixes this by using `is not` to strictly check for the boolean `False`. This solution ensures the `assert_boolean_false()` function correctly distinguishes `False` from other falsy values, thus preventing incorrect test results and undetected bugs in scenarios where exact boolean value checks are crucial."
    }
  }
}