{
  "RepoName": "https://github.com/srittau/python-asserts.git",
  "CommitSHA": "38849833a92ef24b342e0ca59a8fa5752a07bab4",
  "Type": "logic error",
  "ErrorMessage": "======================================================================\nFAIL: test_assert_boolean_true__false__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 142, in test_assert_boolean_true__false__custom_message\n    assert_boolean_true(\"Foo\", \"{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\n\n======================================================================\nFAIL: test_assert_boolean_true__truthy__default_message (test_asserts.AssertTest)\n----------------------------------------------------------------------\nTraceback (most recent call last):\n  File \"/home/user/Projects/user/repobench/repoben/python-asserts/srittau_python-asserts/test_asserts.py\", line 146, in test_assert_boolean_true__truthy__default_message\n    assert_boolean_true(1)\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": "Inconsistent Boolean Assertions Cause False Failures",
    "description": "There is an issue in the assertion library where the `assert_boolean_true` function inconsistently evaluates expressions, causing unexpected assertion failures. Specifically, the function currently evaluates expressions using a truthy check (`if not expr:`), which leads to expressions that are not strictly `True` being misinterpreted. Users may experience assertion failures even when the expression should be considered true in the context of boolean logic.\n\nThis impacts the reliability of tests that use `assert_boolean_true`, causing tests to fail incorrectly. For instance, expressions like non-zero integers or non-empty strings, which may be truthy, will cause `assert_boolean_true` to fail.\n\nThe expected behavior is for `assert_boolean_true` to strictly check for the `True` constant, ensuring that only expressions that exactly equal `True` pass the assertion. This change should increase the precision and reliability of boolean assertions, particularly in test cases where the exact value of `True` is necessary.\n\nThis issue needs to be addressed to ensure that `assert_boolean_true` works as intended, only passing when the expression is explicitly `True` and not merely truthy.",
    "explanation": "### Summary of the Issue\n\nThe issue titled **\"Inconsistent Boolean Assertions Cause False Failures\"** describes a problem where the `assert_boolean_true` function in an assertion library of the `srittau_python-asserts` package incorrectly evaluates boolean expressions. This function uses a general truthy check (`if not expr:`), which causes false assertion failures for expressions that are truthy but not strictly equal to the `True` constant. For example, non-zero integers or non-empty strings are truthy but not strictly `True`, resulting in unexpected assertion failures.\n\n### Detailed Content of the Commit\n\nThe commit information indicates that the developer modified the behavior of the `assert_boolean_true` function. The commit includes the following actions:\n- **Changed the evaluation method**: The function was originally using `if not expr:` to evaluate the expression. This check was changed to `if expr is not True:`.\n\n### Explanation of the Cause and Solution\n\n#### Cause of the Issue\n\nThe root cause of the issue lies in the way the `assert_boolean_true` function evaluates expressions. The use of `if not expr:` in Python checks for truthiness, meaning it interprets any value that is not `False`, `None`, `0`, `''` (empty string), or an empty collection as `True`. This is a broad check and includes many values (such as `1`, non-empty strings, and non-empty collections) that are truthy but not strictly the boolean `True` constant. Consequently, users experienced false assertion failures when using `assert_boolean_true` with expressions that were truthy but not exactly `True`.\n\n#### Solution Implemented in the Commit\n\nTo solve this issue, the commit made a simple but critical change to the logic of the `assert_boolean_true` function:\n- **Changed the evaluation logic**: The check was updated to `if expr is not True:`. \n\nThis adjustment ensures that the function now strictly checks for the boolean `True` value, rather than any truthy value. This means that `assert_boolean_true` will only pass when the expression is explicitly `True` and will fail for any other truthy values (like `1`, `\"non-empty string\"`, etc.).\n\n#### How the Commit Solves the Issue\n\nThe commit addresses the problem by refining the expression evaluation logic in the `assert_boolean_true` function. The switch from `if not expr:` to `if expr is not True:` ensures that only expressions that are strictly `True` (the boolean constant) will pass the assertion. This change eliminates false failures caused by truthy expressions that are not strictly `True`. Therefore, test reliability is improved, and the function behaves as expected by only passing when the tested expression is the boolean `True`.\n\n### Final Explanation of the Solution\n\nThe solution comprehensively resolves the issue by updating the expression logic in the assertion function. This ensures that the function now adheres to the intended check for the boolean `True` rather than any broadly truthy value. Users can now rely on `assert_boolean_true` for precise boolean checks without encountering unexpected assertion failures due to truthy but not strictly `True` values. The fix makes the function's behavior more predictable and enhances the reliability of tests using this function."
  },
  "Explain": "### Summary of the Issue\n\nThe issue titled **\"Inconsistent Boolean Assertions Cause False Failures\"** describes a problem where the `assert_boolean_true` function in an assertion library of the `srittau_python-asserts` package incorrectly evaluates boolean expressions. This function uses a general truthy check (`if not expr:`), which causes false assertion failures for expressions that are truthy but not strictly equal to the `True` constant. For example, non-zero integers or non-empty strings are truthy but not strictly `True`, resulting in unexpected assertion failures.\n\n### Detailed Content of the Commit\n\nThe commit information indicates that the developer modified the behavior of the `assert_boolean_true` function. The commit includes the following actions:\n- **Changed the evaluation method**: The function was originally using `if not expr:` to evaluate the expression. This check was changed to `if expr is not True:`.\n\n### Explanation of the Cause and Solution\n\n#### Cause of the Issue\n\nThe root cause of the issue lies in the way the `assert_boolean_true` function evaluates expressions. The use of `if not expr:` in Python checks for truthiness, meaning it interprets any value that is not `False`, `None`, `0`, `''` (empty string), or an empty collection as `True`. This is a broad check and includes many values (such as `1`, non-empty strings, and non-empty collections) that are truthy but not strictly the boolean `True` constant. Consequently, users experienced false assertion failures when using `assert_boolean_true` with expressions that were truthy but not exactly `True`.\n\n#### Solution Implemented in the Commit\n\nTo solve this issue, the commit made a simple but critical change to the logic of the `assert_boolean_true` function:\n- **Changed the evaluation logic**: The check was updated to `if expr is not True:`. \n\nThis adjustment ensures that the function now strictly checks for the boolean `True` value, rather than any truthy value. This means that `assert_boolean_true` will only pass when the expression is explicitly `True` and will fail for any other truthy values (like `1`, `\"non-empty string\"`, etc.).\n\n#### How the Commit Solves the Issue\n\nThe commit addresses the problem by refining the expression evaluation logic in the `assert_boolean_true` function. The switch from `if not expr:` to `if expr is not True:` ensures that only expressions that are strictly `True` (the boolean constant) will pass the assertion. This change eliminates false failures caused by truthy expressions that are not strictly `True`. Therefore, test reliability is improved, and the function behaves as expected by only passing when the tested expression is the boolean `True`.\n\n### Final Explanation of the Solution\n\nThe solution comprehensively resolves the issue by updating the expression logic in the assertion function. This ensures that the function now adheres to the intended check for the boolean `True` rather than any broadly truthy value. Users can now rely on `assert_boolean_true` for precise boolean checks without encountering unexpected assertion failures due to truthy but not strictly `True` values. The fix makes the function's behavior more predictable and enhances the reliability of tests using this function.",
  "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 not expr:\n        msg = \"{!r} is not True\".format(expr)\n        fail(msg_fmt.format(msg=msg, expr=expr))\n\n\ndef assert_boolean_false(expr, msg_fmt=\"{msg}\"):\n    \"\"\"Fail the test unless the expression is the constant False.\n\n    >>> assert_boolean_false(False)\n    >>> assert_boolean_false(0)\n    Traceback (most recent call last):\n        ...\n    AssertionError: 0 is not False\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * expr - tested expression\n    \"\"\"\n\n    if expr is not False:\n        msg = \"{!r} is not False\".format(expr)\n        fail(msg_fmt.format(msg=msg, expr=expr))\n\n\ndef assert_is_none(expr, msg_fmt=\"{msg}\"):\n    \"\"\"Fail if the expression is not None.\n\n    >>> assert_is_none(None)\n    >>> assert_is_none(False)\n    Traceback (most recent call last):\n        ...\n    AssertionError: False is not None\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * expr - tested expression\n    \"\"\"\n\n    if expr is not None:\n        msg = \"{!r} is not None\".format(expr)\n        fail(msg_fmt.format(msg=msg, expr=expr))\n\n\ndef assert_is_not_none(expr, msg_fmt=\"{msg}\"):\n    \"\"\"Fail if the expression is None.\n\n    >>> assert_is_not_none(0)\n    >>> assert_is_not_none(None)\n    Traceback (most recent call last):\n        ...\n    AssertionError: expression is None\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * expr - tested expression\n    \"\"\"\n    if expr is None:\n        msg = \"expression is None\"\n        fail(msg_fmt.format(msg=msg, expr=expr))\n\n\ndef assert_equal(first, second, msg_fmt=\"{msg}\"):\n    \"\"\"Fail unless first equals second, as determined by the '==' operator.\n\n    >>> assert_equal(5, 5.0)\n    >>> assert_equal(\"Hello World!\", \"Goodbye!\")\n    Traceback (most recent call last):\n        ...\n    AssertionError: 'Hello World!' != 'Goodbye!'\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * first - the first argument\n    * second - the second argument\n    \"\"\"\n\n    if isinstance(first, dict) and isinstance(second, dict):\n        assert_dict_equal(first, second, msg_fmt)\n    elif not first == second:\n        msg = \"{!r} != {!r}\".format(first, second)\n        fail(msg_fmt.format(msg=msg, first=first, second=second))\n\n\ndef assert_not_equal(first, second, msg_fmt=\"{msg}\"):\n    \"\"\"Fail if first equals second, as determined by the '==' operator.\n\n    >>> assert_not_equal(5, 8)\n    >>> assert_not_equal(-7, -7.0)\n    Traceback (most recent call last):\n        ...\n    AssertionError: -7 == -7.0\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * first - the first argument\n    * second - the second argument\n    \"\"\"\n\n    if first == second:\n        msg = \"{!r} == {!r}\".format(first, second)\n        fail(msg_fmt.format(msg=msg, first=first, second=second))\n\n\ndef assert_almost_equal(\n    first, second, msg_fmt=\"{msg}\", places=None, delta=None\n):\n    \"\"\"Fail if first and second are not equal after rounding.\n\n    By default, the difference between first and second is rounded to\n    7 decimal places. This can be configured with the places argument.\n    Alternatively, delta can be used to specify the maximum allowed\n    difference between first and second.\n\n    If first and second can not be rounded or both places and delta are\n    supplied, a TypeError is raised.\n\n    >>> assert_almost_equal(5, 5.00000001)\n    >>> assert_almost_equal(5, 5.001)\n    Traceback (most recent call last):\n        ...\n    AssertionError: 5 != 5.001 within 7 places\n    >>> assert_almost_equal(5, 5.001, places=2)\n    >>> assert_almost_equal(5, 5.001, delta=0.1)\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * first - the first argument\n    * second - the second argument\n    * places - number of places to compare or None\n    * delta - delta or None\n    \"\"\"\n\n    if delta is not None and places is not None:\n        raise TypeError(\"'places' and 'delta' are mutually exclusive\")\n    if delta is not None:\n        if delta <= 0:\n            raise ValueError(\"delta must be larger than 0\")\n        diff = abs(second - first)\n        success = diff < delta\n        detail_msg = \"with delta={}\".format(delta)\n    else:\n        if places is None:\n            places = 7\n        success = not round(second - first, places)\n        detail_msg = \"within {} places\".format(places)\n    if not success:\n        msg = \"{!r} != {!r} {}\".format(first, second, detail_msg)\n        fail(\n            msg_fmt.format(\n                msg=msg, first=first, second=second, places=places, delta=delta\n            )\n        )\n\n\ndef assert_not_almost_equal(\n    first, second, msg_fmt=\"{msg}\", places=None, delta=None\n):\n    \"\"\"Fail if first and second are equal after rounding.\n\n    By default, the difference between first and second is rounded to\n    7 decimal places. This can be configured with the places argument.\n    Alternatively, delta can be used to specify the maximum allowed\n    difference between first and second.\n\n    If first and second can not be rounded or both places and delta are\n    supplied, a TypeError is raised.\n\n    >>> assert_not_almost_equal(5, 5.001)\n    >>> assert_not_almost_equal(5, 5.00000001)\n    Traceback (most recent call last):\n        ...\n    AssertionError: 5 == 5.00000001 within 7 places\n    >>> assert_not_almost_equal(5, 5.001, places=2)\n    Traceback (most recent call last):\n        ...\n    AssertionError: 5 == 5.001 within 2 places\n    >>> assert_not_almost_equal(5, 5.001, delta=0.1)\n    Traceback (most recent call last):\n        ...\n    AssertionError: 5 == 5.001 with delta=0.1\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * first - the first argument\n    * second - the second argument\n    * places - number of places to compare or None\n    * delta - delta or None\n    \"\"\"\n\n    if delta is not None and places is not None:\n        raise TypeError(\"'places' and 'delta' are mutually exclusive\")\n    if delta is not None:\n        if delta <= 0:\n            raise ValueError(\"delta must be larger than 0\")\n        diff = abs(second - first)\n        success = diff >= delta\n        detail_msg = \"with delta={}\".format(delta)\n    else:\n        if places is None:\n            places = 7\n        success = bool(round(second - first, places))\n        detail_msg = \"within {} places\".format(places)\n    if not success:\n        msg = \"{!r} == {!r} {}\".format(first, second, detail_msg)\n        fail(\n            msg_fmt.format(\n                msg=msg, first=first, second=second, places=places, delta=delta\n            )\n        )\n\n\ndef assert_dict_equal(\n    first, second, key_msg_fmt=\"{msg}\", value_msg_fmt=\"{msg}\"\n):\n    \"\"\"Fail unless first dictionary equals second.\n\n    The dictionaries are considered equal, if they both contain the same\n    keys, and their respective values are also equal.\n\n    >>> assert_dict_equal({\"foo\": 5}, {\"foo\": 5})\n    >>> assert_dict_equal({\"foo\": 5}, {})\n    Traceback (most recent call last):\n        ...\n    AssertionError: key 'foo' missing from right dict\n\n    The following key_msg_fmt arguments are supported, if the keys do not\n    match:\n    * msg - the default error message\n    * first - the first dict\n    * second - the second dict\n    * missing_keys - list of keys missing from right\n    * extra_keys - list of keys missing from left\n\n    The following value_msg_fmt arguments are supported, if a value does not\n    match:\n    * msg - the default error message\n    * first - the first dict\n    * second - the second dict\n    * key - the key where the value does not match\n    * first_value - the value in the first dict\n    * second_value - the value in the second dict\n    \"\"\"\n    first_keys = set(first.keys())\n    second_keys = set(second.keys())\n    missing_keys = list(first_keys - second_keys)\n    extra_keys = list(second_keys - first_keys)\n    if missing_keys or extra_keys:\n        if missing_keys:\n            if len(missing_keys) == 1:\n                msg = \"key {!r} missing from right dict\".format(\n                    missing_keys[0]\n                )\n            else:\n                keys = \", \".join(sorted(repr(k) for k in missing_keys))\n                msg = \"keys {} missing from right dict\".format(keys)\n        else:\n            if len(extra_keys) == 1:\n                msg = \"extra key {!r} in right dict\".format(extra_keys[0])\n            else:\n                keys = \", \".join(sorted(repr(k) for k in extra_keys))\n                msg = \"extra keys {} in right dict\".format(keys)\n        if key_msg_fmt:\n            msg = key_msg_fmt.format(\n                msg=msg,\n                first=first,\n                second=second,\n                missing_keys=missing_keys,\n                extra_keys=extra_keys,\n            )\n        raise AssertionError(msg)\n    for key in first:\n        first_value = first[key]\n        second_value = second[key]\n        msg = \"key '{}' differs: {!r} != {!r}\".format(\n            key, first_value, second_value\n        )\n        if value_msg_fmt:\n            msg = value_msg_fmt.format(\n                msg=msg,\n                first=first,\n                second=second,\n                key=key,\n                first_value=first_value,\n                second_value=second_value,\n            )\n        msg = msg.replace(\"{\", \"{{\").replace(\"}\", \"}}\")\n        assert_equal(first_value, second_value, msg_fmt=msg)\n\n\ndef assert_dict_superset(\n    first, second, key_msg_fmt=\"{msg}\", value_msg_fmt=\"{msg}\"\n):\n    \"\"\"Fail unless second dictionary is a superset of the first.\n\n    The second dictionary must contain all keys of the first and their\n    values are equal (or a superset in case of dicts). But the second\n    dictionary can contain additional keys.\n\n    >>> assert_dict_superset({\"foo\": 5}, {\"foo\": 5, \"bar\": 10})\n    >>> assert_dict_superset({\"foo\": 5}, {})\n    Traceback (most recent call last):\n        ...\n    AssertionError: key 'foo' missing from right dict\n\n    The following key_msg_fmt arguments are supported, if the keys do not\n    match:\n    * msg - the default error message\n    * first - the first dict\n    * second - the second dict\n    * missing_keys - list of keys missing from right\n\n    The following value_msg_fmt arguments are supported, if a value does not\n    match:\n    * msg - the default error message\n    * first - the first dict\n    * second - the second dict\n    * key - the key where the value does not match\n    * first_value - the value in the first dict\n    * second_value - the value in the second dict\n    \"\"\"\n    first_keys = set(first.keys())\n    second_keys = set(second.keys())\n    missing_keys = list(first_keys - second_keys)\n    if missing_keys:\n        if len(missing_keys) == 1:\n            msg = \"key {!r} missing from right dict\".format(missing_keys[0])\n        else:\n            keys = \", \".join(sorted(repr(k) for k in missing_keys))\n            msg = \"keys {} missing from right dict\".format(keys)\n        if key_msg_fmt:\n            msg = key_msg_fmt.format(\n                msg=msg, first=first, second=second, missing_keys=missing_keys\n            )\n        raise AssertionError(msg)\n    for key in first:\n        first_value = first[key]\n        second_value = second[key]\n        msg = \"key '{}' differs: {!r} != {!r}\".format(\n            key, first_value, second_value\n        )\n        if value_msg_fmt:\n            msg = value_msg_fmt.format(\n                msg=msg,\n                first=first,\n                second=second,\n                key=key,\n                first_value=first_value,\n                second_value=second_value,\n            )\n        msg = msg.replace(\"{\", \"{{\").replace(\"}\", \"}}\")\n        assert_equal(first_value, second_value, msg_fmt=msg)\n\n\ndef assert_less(first, second, msg_fmt=\"{msg}\"):\n    \"\"\"Fail if first is not less than second.\n\n    >>> assert_less('bar', 'foo')\n    >>> assert_less(5, 5)\n    Traceback (most recent call last):\n        ...\n    AssertionError: 5 is not less than 5\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * first - the first argument\n    * second - the second argument\n    \"\"\"\n\n    if not first < second:\n        msg = \"{!r} is not less than {!r}\".format(first, second)\n        fail(msg_fmt.format(msg=msg, first=first, second=second))\n\n\ndef assert_less_equal(first, second, msg_fmt=\"{msg}\"):\n    \"\"\"Fail if first is not less than or equal to second.\n\n    >>> assert_less_equal('bar', 'foo')\n    >>> assert_less_equal(5, 5)\n    >>> assert_less_equal(6, 5)\n    Traceback (most recent call last):\n        ...\n    AssertionError: 6 is not less than or equal to 5\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * first - the first argument\n    * second - the second argument\n    \"\"\"\n\n    if not first <= second:\n        msg = \"{!r} is not less than or equal to {!r}\".format(first, second)\n        fail(msg_fmt.format(msg=msg, first=first, second=second))\n\n\ndef assert_greater(first, second, msg_fmt=\"{msg}\"):\n    \"\"\"Fail if first is not greater than second.\n\n    >>> assert_greater('foo', 'bar')\n    >>> assert_greater(5, 5)\n    Traceback (most recent call last):\n        ...\n    AssertionError: 5 is not greater than 5\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * first - the first argument\n    * second - the second argument\n    \"\"\"\n\n    if not first > second:\n        msg = \"{!r} is not greater than {!r}\".format(first, second)\n        fail(msg_fmt.format(msg=msg, first=first, second=second))\n\n\ndef assert_greater_equal(first, second, msg_fmt=\"{msg}\"):\n    \"\"\"Fail if first is not greater than or equal to second.\n\n    >>> assert_greater_equal('foo', 'bar')\n    >>> assert_greater_equal(5, 5)\n    >>> assert_greater_equal(5, 6)\n    Traceback (most recent call last):\n        ...\n    AssertionError: 5 is not greater than or equal to 6\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * first - the first argument\n    * second - the second argument\n    \"\"\"\n\n    if not first >= second:\n        msg = \"{!r} is not greater than or equal to {!r}\".format(first, second)\n        fail(msg_fmt.format(msg=msg, first=first, second=second))\n\n\ndef assert_regex(text, regex, msg_fmt=\"{msg}\"):\n    \"\"\"Fail if text does not match the regular expression.\n\n    regex can be either a regular expression string or a compiled regular\n    expression object.\n\n    >>> assert_regex(\"Hello World!\", r\"llo.*rld!$\")\n    >>> assert_regex(\"Hello World!\", r\"\\\\d\")\n    Traceback (most recent call last):\n        ...\n    AssertionError: 'Hello World!' does not match '\\\\\\\\d'\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * text - text that is matched\n    * pattern - regular expression pattern as string\n    \"\"\"\n\n    compiled = re.compile(regex)\n    if not compiled.search(text):\n        msg = \"{!r} does not match {!r}\".format(text, compiled.pattern)\n        fail(msg_fmt.format(msg=msg, text=text, pattern=compiled.pattern))\n\n\ndef assert_not_regex(text, regex, msg_fmt=\"{msg}\"):\n    \"\"\"Fail if text does match the regular expression.\n\n    regex can be either a regular expression string or a compiled regular\n    expression object.\n\n    >>> assert_regex(\"Hello World!\", r\"llo.*rld!$\")\n    >>> assert_regex(\"Hello World!\", r\"\\\\d\")\n    Traceback (most recent call last):\n        ...\n    AssertionError: 'Hello World!' does not match '\\\\\\\\d'\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * text - text that is matched\n    * pattern - regular expression pattern as string\n    \"\"\"\n\n    compiled = re.compile(regex)\n    if compiled.search(text):\n        msg = \"{!r} matches {!r}\".format(text, compiled.pattern)\n        fail(msg_fmt.format(msg=msg, text=text, pattern=compiled.pattern))\n\n\ndef assert_is(first, second, msg_fmt=\"{msg}\"):\n    \"\"\"Fail if first and second do not refer to the same object.\n\n    >>> list1 = [5, \"foo\"]\n    >>> list2 = [5, \"foo\"]\n    >>> assert_is(list1, list1)\n    >>> assert_is(list1, list2)\n    Traceback (most recent call last):\n        ...\n    AssertionError: [5, 'foo'] is not [5, 'foo']\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * first - the first argument\n    * second - the second argument\n    \"\"\"\n\n    if first is not second:\n        msg = \"{!r} is not {!r}\".format(first, second)\n        fail(msg_fmt.format(msg=msg, first=first, second=second))\n\n\ndef assert_is_not(first, second, msg_fmt=\"{msg}\"):\n    \"\"\"Fail if first and second refer to the same object.\n\n    >>> list1 = [5, \"foo\"]\n    >>> list2 = [5, \"foo\"]\n    >>> assert_is_not(list1, list2)\n    >>> assert_is_not(list1, list1)\n    Traceback (most recent call last):\n        ...\n    AssertionError: both arguments refer to [5, 'foo']\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * first - the first argument\n    * second - the second argument\n    \"\"\"\n\n    if first is second:\n        msg = \"both arguments refer to {!r}\".format(first)\n        fail(msg_fmt.format(msg=msg, first=first, second=second))\n\n\ndef assert_in(first, second, msg_fmt=\"{msg}\"):\n    \"\"\"Fail if first is not in collection second.\n\n    >>> assert_in(\"foo\", [4, \"foo\", {}])\n    >>> assert_in(\"bar\", [4, \"foo\", {}])\n    Traceback (most recent call last):\n        ...\n    AssertionError: 'bar' not in [4, 'foo', {}]\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * first - the element looked for\n    * second - the container looked in\n    \"\"\"\n\n    if first not in second:\n        msg = \"{!r} not in {!r}\".format(first, second)\n        fail(msg_fmt.format(msg=msg, first=first, second=second))\n\n\ndef assert_not_in(first, second, msg_fmt=\"{msg}\"):\n    \"\"\"Fail if first is in a collection second.\n\n    >>> assert_not_in(\"bar\", [4, \"foo\", {}])\n    >>> assert_not_in(\"foo\", [4, \"foo\", {}])\n    Traceback (most recent call last):\n        ...\n    AssertionError: 'foo' is in [4, 'foo', {}]\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * first - the element looked for\n    * second - the container looked in\n    \"\"\"\n    if first in second:\n        msg = \"{!r} is in {!r}\".format(first, second)\n        fail(msg_fmt.format(msg=msg, first=first, second=second))\n\n\ndef assert_count_equal(sequence1, sequence2, msg_fmt=\"{msg}\"):\n    \"\"\"Compare the items of two sequences, ignoring order.\n\n    >>> assert_count_equal([1, 2], {2, 1})\n\n    Items missing in either sequence will be listed:\n\n    >>> assert_count_equal([\"a\", \"b\", \"c\"], [\"a\", \"d\"])\n    Traceback (most recent call last):\n        ...\n    AssertionError: missing from sequence 1: 'd'; missing from sequence 2: 'b', 'c'\n\n    Items are counted in each sequence. This makes it useful to detect\n    duplicates:\n\n    >>> assert_count_equal({\"a\", \"b\"}, [\"a\", \"a\", \"b\"])\n    Traceback (most recent call last):\n        ...\n    AssertionError: missing from sequence 1: 'a'\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * first - first sequence\n    * second - second sequence\n    \"\"\"\n\n    def compare():\n        missing1 = list(sequence2)\n        missing2 = []\n        for item in sequence1:\n            try:\n                missing1.remove(item)\n            except ValueError:\n                missing2.append(item)\n        return missing1, missing2\n\n    def build_message():\n        msg = \"\"\n        if missing_from_1:\n            msg += \"missing from sequence 1: \" + \", \".join(\n                repr(i) for i in missing_from_1\n            )\n        if missing_from_1 and missing_from_2:\n            msg += \"; \"\n        if missing_from_2:\n            msg += \"missing from sequence 2: \" + \", \".join(\n                repr(i) for i in missing_from_2\n            )\n        return msg\n\n    missing_from_1, missing_from_2 = compare()\n    if missing_from_1 or missing_from_2:\n        fail(\n            msg_fmt.format(\n                msg=build_message(), first=sequence1, second=sequence2\n            )\n        )\n\n\ndef assert_between(lower_bound, upper_bound, expr, msg_fmt=\"{msg}\"):\n    \"\"\"Fail if an expression is not between certain bounds (inclusive).\n\n    >>> assert_between(5, 15, 5)\n    >>> assert_between(5, 15, 15)\n    >>> assert_between(5, 15, 4.9)\n    Traceback (most recent call last):\n        ...\n    AssertionError: 4.9 is not between 5 and 15\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * lower - lower bound\n    * upper - upper bound\n    * expr - tested expression\n    \"\"\"\n\n    if not lower_bound <= expr <= upper_bound:\n        msg = \"{!r} is not between {} and {}\".format(\n            expr, lower_bound, upper_bound\n        )\n        fail(\n            msg_fmt.format(\n                msg=msg, lower=lower_bound, upper=upper_bound, expr=expr\n            )\n        )\n\n\ndef assert_is_instance(obj, cls, msg_fmt=\"{msg}\"):\n    \"\"\"Fail if an object is not an instance of a class or tuple of classes.\n\n    >>> assert_is_instance(5, int)\n    >>> assert_is_instance('foo', (str, bytes))\n    >>> assert_is_instance(5, str)\n    Traceback (most recent call last):\n        ...\n    AssertionError: 5 is an instance of <class 'int'>, expected <class 'str'>\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * obj - object to test\n    * types - tuple of types tested against\n    \"\"\"\n    if not isinstance(obj, cls):\n        msg = \"{!r} is an instance of {!r}, expected {!r}\".format(\n            obj, obj.__class__, cls\n        )\n        types = cls if isinstance(cls, tuple) else (cls,)\n        fail(msg_fmt.format(msg=msg, obj=obj, types=types))\n\n\ndef assert_not_is_instance(obj, cls, msg_fmt=\"{msg}\"):\n    \"\"\"Fail if an object is an instance of a class or tuple of classes.\n\n    >>> assert_not_is_instance(5, str)\n    >>> assert_not_is_instance(5, (str, bytes))\n    >>> assert_not_is_instance('foo', str)\n    Traceback (most recent call last):\n        ...\n    AssertionError: 'foo' is an instance of <class 'str'>\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * obj - object to test\n    * types - tuple of types tested against\n    \"\"\"\n    if isinstance(obj, cls):\n        msg = \"{!r} is an instance of {!r}\".format(obj, obj.__class__)\n        types = cls if isinstance(cls, tuple) else (cls,)\n        fail(msg_fmt.format(msg=msg, obj=obj, types=types))\n\n\ndef assert_has_attr(obj, attribute, msg_fmt=\"{msg}\"):\n    \"\"\"Fail is an object does not have an attribute.\n\n    >>> assert_has_attr([], \"index\")\n    >>> assert_has_attr([], \"i_do_not_have_this\")\n    Traceback (most recent call last):\n        ...\n    AssertionError: [] does not have attribute 'i_do_not_have_this'\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * obj - object to test\n    * attribute - name of the attribute to check\n    \"\"\"\n\n    if not hasattr(obj, attribute):\n        msg = \"{!r} does not have attribute '{}'\".format(obj, attribute)\n        fail(msg_fmt.format(msg=msg, obj=obj, attribute=attribute))\n\n\n_EPSILON_SECONDS = 5\n\n\ndef assert_datetime_about_now(actual, msg_fmt=\"{msg}\"):\n    \"\"\"Fail if a datetime object is not within 5 seconds of the local time.\n\n    >>> assert_datetime_about_now(datetime.now())\n    >>> assert_datetime_about_now(datetime(1900, 1, 1, 12, 0, 0))\n    Traceback (most recent call last):\n        ...\n    AssertionError: datetime.datetime(1900, 1, 1, 12, 0) is not close to current date/time\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * actual - datetime object to check\n    * now - current datetime that was tested against\n    \"\"\"\n\n    now = datetime.now()\n    if actual is None:\n        msg = \"None is not a valid date/time\"\n        fail(msg_fmt.format(msg=msg, actual=actual, now=now))\n    lower_bound = now - timedelta(seconds=_EPSILON_SECONDS)\n    upper_bound = now + timedelta(seconds=_EPSILON_SECONDS)\n    if not lower_bound <= actual <= upper_bound:\n        msg = \"{!r} is not close to current date/time\".format(actual)\n        fail(msg_fmt.format(msg=msg, actual=actual, now=now))\n\n\ndef assert_datetime_about_now_utc(actual, msg_fmt=\"{msg}\"):\n    \"\"\"Fail if a datetime object is not within 5 seconds of UTC.\n\n    >>> assert_datetime_about_now_utc(datetime.now(timezone.utc).replace(tzinfo=None))\n    >>> assert_datetime_about_now_utc(datetime(1900, 1, 1, 12, 0, 0))\n    Traceback (most recent call last):\n        ...\n    AssertionError: datetime.datetime(1900, 1, 1, 12, 0) is not close to current UTC date/time\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * actual - datetime object to check\n    * now - current datetime that was tested against\n    \"\"\"\n\n    now = datetime.now(timezone.utc).replace(tzinfo=None)\n    if actual is None:\n        msg = \"None is not a valid date/time\"\n        fail(msg_fmt.format(msg=msg, actual=actual, now=now))\n    lower_bound = now - timedelta(seconds=_EPSILON_SECONDS)\n    upper_bound = now + timedelta(seconds=_EPSILON_SECONDS)\n    if not lower_bound <= actual <= upper_bound:\n        msg = \"{!r} is not close to current UTC date/time\".format(actual)\n        fail(msg_fmt.format(msg=msg, actual=actual, now=now))\n\n\nclass AssertRaisesContext:\n    \"\"\"A context manager to test for exceptions with certain properties.\n\n    When the context is left and no exception has been raised, an\n    AssertionError will be raised:\n\n        >>> context = AssertRaisesContext(TypeError)\n        >>> with context:\n        ...    pass\n        Traceback (most recent call last):\n            ...\n        AssertionError: TypeError not raised\n\n    If an exception that is not a sub-class of the exception class provided\n    to the constructor is raised, it will be passed on:\n\n        >>> with context:\n        ...    raise ValueError(\"Wrong Class\")\n        Traceback (most recent call last):\n            ...\n        ValueError: Wrong Class\n\n    If the exception has the right class, any additional tests that have been\n    configured on the context, will be called:\n\n        >>> def test(exc):\n        ...     assert_equal(\"Hello World!\", str(exc))\n        >>> context.add_test(test)\n        >>> with context:\n        ...     raise TypeError(\"Wrong Message\")\n        Traceback (most recent call last):\n            ...\n        AssertionError: 'Hello World!' != 'Wrong Message'\n\n    \"\"\"\n\n    def __init__(self, exception, msg_fmt=\"{msg}\"):\n        self.exception = exception\n        self.msg_fmt = msg_fmt\n        self._exc_type = exception\n        self._exc_val = None\n        self._exception_name = getattr(exception, \"__name__\", str(exception))\n        self._tests: list[Callable[[Any], object]] = []\n\n    def __enter__(self):\n        return self\n\n    def __exit__(self, exc_type, exc_val, exc_tb):\n        if not exc_type or not exc_val:\n            msg = \"{} not raised\".format(self._exception_name)\n            fail(self.format_message(msg))\n        self._exc_val = exc_val\n        if not issubclass(exc_type, self.exception):\n            return False\n        for test in self._tests:\n            test(exc_val)\n        return True\n\n    def format_message(self, default_msg):\n        return self.msg_fmt.format(\n            msg=default_msg,\n            exc_type=self._exc_type,\n            exc_name=self._exception_name,\n        )\n\n    def add_test(self, cb: Callable[[Any], object]) -> None:\n        \"\"\"Add a test callback.\n\n        This callback is called after determining that the right exception\n        class was raised. The callback will get the raised exception as only\n        argument.\n\n        \"\"\"\n        self._tests.append(cb)\n\n    @property\n    def exc_val(self):\n        if self._exc_val is None:\n            raise RuntimeError(\"must be called after leaving the context\")\n        return self._exc_val\n\n\nclass AssertRaisesRegexContext(AssertRaisesContext):\n    \"\"\"A context manager to test for exceptions and their messages.\"\"\"\n\n    def __init__(self, exception, pattern, msg_fmt=\"{msg}\"):\n        super(AssertRaisesRegexContext, self).__init__(exception, msg_fmt)\n        self.pattern = pattern\n\n    def format_message(self, default_msg):\n        return self.msg_fmt.format(\n            msg=default_msg,\n            exc_type=self._exc_type,\n            exc_name=self._exception_name,\n            pattern=self.pattern,\n            text=\"\",\n        )\n\n\nclass AssertRaisesErrnoContext(AssertRaisesContext):\n    \"\"\"A context manager to test for exceptions with errnos.\"\"\"\n\n    def __init__(self, exception, expected_errno, msg_fmt=\"{msg}\"):\n        super(AssertRaisesErrnoContext, self).__init__(exception, msg_fmt)\n        self.expected_errno = expected_errno\n\n    def format_message(self, default_msg):\n        return self.msg_fmt.format(\n            msg=default_msg,\n            exc_type=self._exc_type,\n            exc_name=self._exception_name,\n            expected_errno=self.expected_errno,\n            actual_errno=None,\n        )\n\n\ndef assert_raises(exception, msg_fmt=\"{msg}\"):\n    \"\"\"Fail unless a specific exception is raised inside the context.\n\n    If a different type of exception is raised, it will not be caught.\n\n    >>> with assert_raises(TypeError):\n    ...     raise TypeError()\n    ...\n    >>> with assert_raises(TypeError):\n    ...     pass\n    ...\n    Traceback (most recent call last):\n        ...\n    AssertionError: TypeError not raised\n    >>> with assert_raises(TypeError):\n    ...     raise ValueError(\"wrong error\")\n    ...\n    Traceback (most recent call last):\n        ...\n    ValueError: wrong error\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * exc_type - exception type that is expected\n    * exc_name - expected exception type name\n    \"\"\"\n\n    return AssertRaisesContext(exception, msg_fmt)\n\n\ndef assert_raises_regex(exception, regex, msg_fmt=\"{msg}\"):\n    \"\"\"Fail unless an exception with a message that matches a regular\n     expression is raised within the context.\n\n    The regular expression can be a regular expression string or object.\n\n    >>> with assert_raises_regex(ValueError, r\"\\\\d+\"):\n    ...     raise ValueError(\"Error #42\")\n    ...\n    >>> with assert_raises_regex(ValueError, r\"\\\\d+\"):\n    ...     raise ValueError(\"Generic Error\")\n    ...\n    Traceback (most recent call last):\n        ...\n    AssertionError: 'Generic Error' does not match '\\\\\\\\d+'\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * exc_type - exception type that is expected\n    * exc_name - expected exception type name\n    * text - actual error text\n    * pattern - expected error message as regular expression string\n    \"\"\"\n\n    def test(exc):\n        compiled = re.compile(regex)\n        if not exc.args:\n            msg = \"{} without message\".format(exception.__name__)\n            fail(\n                msg_fmt.format(\n                    msg=msg,\n                    text=None,\n                    pattern=compiled.pattern,\n                    exc_type=exception,\n                    exc_name=exception.__name__,\n                )\n            )\n        text = exc.args[0]\n        if not compiled.search(text):\n            msg = \"{!r} does not match {!r}\".format(text, compiled.pattern)\n            fail(\n                msg_fmt.format(\n                    msg=msg,\n                    text=text,\n                    pattern=compiled.pattern,\n                    exc_type=exception,\n                    exc_name=exception.__name__,\n                )\n            )\n\n    context = AssertRaisesRegexContext(exception, regex, msg_fmt)\n    context.add_test(test)\n    return context\n\n\ndef assert_raises_errno(exception, errno, msg_fmt=\"{msg}\"):\n    \"\"\"Fail unless an exception with a specific errno is raised with the\n     context.\n\n    >>> with assert_raises_errno(OSError, 42):\n    ...     raise OSError(42, \"OS Error\")\n    ...\n    >>> with assert_raises_errno(OSError, 44):\n    ...     raise OSError(17, \"OS Error\")\n    ...\n    Traceback (most recent call last):\n        ...\n    AssertionError: wrong errno: 44 != 17\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * exc_type - exception type that is expected\n    * exc_name - expected exception type name\n    * expected_errno -\n    * actual_errno - raised errno or None if no matching exception was raised\n    \"\"\"\n\n    def check_errno(exc):\n        if errno != exc.errno:\n            msg = \"wrong errno: {!r} != {!r}\".format(errno, exc.errno)\n            fail(\n                msg_fmt.format(\n                    msg=msg,\n                    exc_type=exception,\n                    exc_name=exception.__name__,\n                    expected_errno=errno,\n                    actual_errno=exc.errno,\n                )\n            )\n\n    context = AssertRaisesErrnoContext(exception, errno, msg_fmt)\n    context.add_test(check_errno)\n    return context\n\n\ndef assert_succeeds(exception, msg_fmt=\"{msg}\"):\n    \"\"\"Fail if a specific exception is raised within the context.\n\n    This assertion should be used for cases, where successfully running a\n    function signals a successful test, and raising the exception of a\n    certain type signals a test failure. All other raised exceptions are\n    passed on and will usually still result in a test error. This can be\n    used to signal the intent of a block.\n\n    >>> l = [\"foo\", \"bar\"]\n    >>> with assert_succeeds(ValueError):\n    ...     i = l.index(\"foo\")\n    ...\n    >>> with assert_succeeds(ValueError):\n    ...     raise ValueError()\n    ...\n    Traceback (most recent call last):\n        ...\n    AssertionError: ValueError was unexpectedly raised\n    >>> with assert_succeeds(ValueError):\n    ...     raise TypeError(\"Wrong Error\")\n    ...\n    Traceback (most recent call last):\n        ...\n    TypeError: Wrong Error\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * exc_type - exception type\n    * exc_name - exception type name\n    * exception - exception that was raised\n    \"\"\"\n\n    class _AssertSucceeds(object):\n        def __enter__(self):\n            pass\n\n        def __exit__(self, exc_type, exc_val, exc_tb):\n            if exc_type and issubclass(exc_type, exception):\n                msg = exception.__name__ + \" was unexpectedly raised\"\n                fail(\n                    msg_fmt.format(\n                        msg=msg,\n                        exc_type=exception,\n                        exc_name=exception.__name__,\n                        exception=exc_val,\n                    )\n                )\n\n    return _AssertSucceeds()\n\n\nclass AssertWarnsContext(object):\n    \"\"\"A context manager to test for warnings with certain properties.\n\n    When the context is left and the expected warning has not been raised, an\n    AssertionError will be raised:\n\n        >>> context = AssertWarnsContext(DeprecationWarning)\n        >>> with context:\n        ...    pass\n        Traceback (most recent call last):\n            ...\n        AssertionError: DeprecationWarning not issued\n\n    If the warning has the right class, any additional tests that have been\n    configured on the context, will be called:\n\n        >>> from warnings import warn\n        >>> def test(warning):\n        ...     return False\n        >>> context.add_test(test)\n        >>> with context:\n        ...     warn(\"Wrong Message\", DeprecationWarning)\n        Traceback (most recent call last):\n            ...\n        AssertionError: DeprecationWarning not issued\n\n    \"\"\"\n\n    def __init__(self, warning_class, msg_fmt=\"{msg}\"):\n        self._warning_class = warning_class\n        self._msg_fmt = msg_fmt\n        self._warning_context: catch_warnings[list[WarningMessage]] | None = (\n            None\n        )\n        self._warnings = []\n        self._tests: list[Callable[[Warning], bool]] = []\n\n    def __enter__(self):\n        self._warning_context = catch_warnings(record=True)\n        self._warnings = self._warning_context.__enter__()\n        return self\n\n    def __exit__(self, exc_type, exc_val, exc_tb):\n        assert self._warning_context is not None\n        self._warning_context.__exit__(exc_type, exc_val, exc_tb)\n        if not any(self._is_expected_warning(w) for w in self._warnings):\n            fail(self.format_message())\n\n    def format_message(self):\n        msg = \"{} not issued\".format(self._warning_class.__name__)\n        return self._msg_fmt.format(\n            msg=msg,\n            exc_type=self._warning_class,\n            exc_name=self._warning_class.__name__,\n        )\n\n    def _is_expected_warning(self, warning) -> bool:\n        if not issubclass(warning.category, self._warning_class):\n            return False\n        return all(test(warning) for test in self._tests)\n\n    def add_test(self, cb: Callable[[Warning], bool]) -> None:\n        \"\"\"Add a test callback.\n\n        This callback is called after determining that the right warning\n        class was issued. The callback will get the issued warning as only\n        argument and must return a boolean value.\n\n        \"\"\"\n        self._tests.append(cb)\n\n\nclass AssertWarnsRegexContext(AssertWarnsContext):\n    \"\"\"A context manager to test for warnings and their messages.\"\"\"\n\n    def __init__(self, warning_class, pattern, msg_fmt=\"{msg}\"):\n        super(AssertWarnsRegexContext, self).__init__(warning_class, msg_fmt)\n        self.pattern = pattern\n\n    def format_message(self):\n        msg = \"no {} matching {} issued\".format(\n            self._warning_class.__name__, repr(self.pattern)\n        )\n        return self._msg_fmt.format(\n            msg=msg,\n            exc_type=self._warning_class,\n            exc_name=self._warning_class.__name__,\n            pattern=self.pattern,\n        )\n\n\ndef assert_warns(warning_type, msg_fmt=\"{msg}\"):\n    \"\"\"Fail unless a specific warning is issued inside the context.\n\n    If a different type of warning is issued, it will not be caught.\n\n    >>> from warnings import warn\n    >>> with assert_warns(UserWarning):\n    ...     warn(\"warning message\", UserWarning)\n    ...\n    >>> with assert_warns(UserWarning):\n    ...     pass\n    ...\n    Traceback (most recent call last):\n        ...\n    AssertionError: UserWarning not issued\n    >>> with assert_warns(UserWarning):\n    ...     warn(\"warning message\", UnicodeWarning)\n    ...\n    Traceback (most recent call last):\n        ...\n    AssertionError: UserWarning not issued\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * exc_type - exception type\n    * exc_name - exception type name\n    \"\"\"\n    return AssertWarnsContext(warning_type, msg_fmt)\n\n\ndef assert_warns_regex(warning_type, regex, msg_fmt=\"{msg}\"):\n    \"\"\"Fail unless a warning with a message is issued inside the context.\n\n    The message can be a regular expression string or object.\n\n    >>> from warnings import warn\n    >>> with assert_warns_regex(UserWarning, r\"#\\\\d+\"):\n    ...     warn(\"Error #42\", UserWarning)\n    ...\n    >>> with assert_warns_regex(UserWarning, r\"Expected Error\"):\n    ...     warn(\"Generic Error\", UserWarning)\n    ...\n    Traceback (most recent call last):\n        ...\n    AssertionError: no UserWarning matching 'Expected Error' issued\n\n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * exc_type - warning type\n    * exc_name - warning type name\n    * pattern - expected warning message as regular expression string\n    \"\"\"\n\n    def test(warning):\n        return re.search(regex, str(warning.message)) is not None\n\n    context = AssertWarnsRegexContext(warning_type, regex, msg_fmt)\n    context.add_test(test)\n    return context\n\n\nif sys.version_info >= (3,):\n    _Str = str\nelse:\n    _Str = unicode  # noqa: F821\n\n\ndef assert_json_subset(first, second):\n    \"\"\"Assert that a JSON object or array is a subset of another JSON object\n    or array.\n\n    The first JSON object or array must be supplied as a JSON-compatible\n    dict or list, the JSON object or array to check must be a string, an\n    UTF-8 bytes object, or a JSON-compatible list or dict.\n\n    A JSON non-object, non-array value is the subset of another JSON value,\n    if they are equal.\n\n    A JSON object is the subset of another JSON object if for each name/value\n    pair in the former there is a name/value pair in the latter with the same\n    name. Additionally, the value of the former pair must be a subset of the\n    value of the latter pair.\n\n    A JSON array is the subset of another JSON array, if they have the same\n    number of elements and each element in the former is a subset of the\n    corresponding element in the latter.\n\n    >>> assert_json_subset({}, '{}')\n    >>> assert_json_subset({}, '{\"foo\": \"bar\"}')\n    >>> assert_json_subset({\"foo\": \"bar\"}, '{}')\n    Traceback (most recent call last):\n    ...\n    AssertionError: element 'foo' missing from element $\n    >>> assert_json_subset([1, 2], '[1, 2]')\n    >>> assert_json_subset([2, 1], '[1, 2]')\n    Traceback (most recent call last):\n    ...\n    AssertionError: element $[0] differs: 2 != 1\n    >>> assert_json_subset([{}], '[{\"foo\": \"bar\"}]')\n    >>> assert_json_subset({}, \"INVALID JSON\")\n    Traceback (most recent call last):\n    ...\n    json.decoder.JSONDecodeError: Expecting value: line 1 column 1 (char 0)\n\n    In objects, the special classes `Present` and `Absent` can be used to\n    check for the presence or absence of a specific key:\n\n    >>> assert_json_subset({Exists(\"foo\"): True}, '{\"foo\": \"bar\"}')\n    >>> assert_json_subset({Exists(\"foo\"): True}, '{}')\n    Traceback (most recent call last):\n    ...\n    AssertionError: element 'foo' missing from element $\n    >>> assert_json_subset({Exists(\"foo\"): False}, '{}')\n    >>> assert_json_subset({Exists(\"foo\"): False}, '{\"foo\": \"bar\"}')\n    Traceback (most recent call last):\n    ...\n    AssertionError: spurious member 'foo' in object $\n    \"\"\"\n\n    if not isinstance(second, (dict, list, str, bytes)):\n        raise TypeError(\"second must be dict, list, str, or bytes\")\n    if isinstance(second, bytes):\n        second = second.decode(\"utf-8\")\n    if isinstance(second, _Str):\n        parsed_second = json_loads(second)\n    else:\n        parsed_second = second\n\n    if not isinstance(parsed_second, (dict, list)):\n        raise AssertionError(\n            \"second must decode to dict or list, not {}\".format(\n                type(parsed_second)\n            )\n        )\n\n    comparer = _JSONComparer(_JSONPath(\"$\"), first, parsed_second)\n    comparer.assert_()\n\n\nclass _JSONComparer:\n    def __init__(self, path, expected, actual):\n        self._path = path\n        self._expected = expected\n        self._actual = actual\n\n    def assert_(self):\n        self._assert_types_are_equal()\n        if isinstance(self._expected, dict):\n            self._assert_dicts_equal()\n        elif isinstance(self._expected, list):\n            self._assert_arrays_equal()\n        elif _is_present(self._expected):\n            pass\n        else:\n            self._assert_fundamental_values_equal()\n\n    def _assert_types_are_equal(self):\n        if self._types_differ():\n            self._raise_different_values()\n\n    def _types_differ(self):\n        if self._expected is None:\n            return self._actual is not None\n        elif isinstance(self._expected, (int, float)):\n            return not isinstance(self._actual, (int, float))\n        elif _is_present(self._expected):\n            return False\n        for type_ in [bool, str, _Str, list, dict]:\n            if isinstance(self._expected, type_):\n                return not isinstance(self._actual, type_)\n        else:\n            raise TypeError(\"unsupported type {}\".format(type(self._expected)))\n\n    def _assert_dicts_equal(self) -> None:\n        for name in self._expected:\n            if not isinstance(name, (str, Exists)):\n                raise TypeError(\n                    f\"{repr(name)} is not a valid object member name\",\n                )\n        self._assert_all_expected_keys_in_actual_dict()\n        self._assert_no_wrong_keys()\n        self._assert_dict_values_equal()\n\n    def _assert_all_expected_keys_in_actual_dict(self) -> None:\n        keys = self._expected_key_names.difference(self._actual.keys())\n        if keys:\n            self._raise_missing_element(keys)\n\n    def _assert_no_wrong_keys(self) -> None:\n        for name in self._expected:\n            if isinstance(name, str) and _is_absent(self._expected[name]):\n                if name in self._actual:\n                    self._raise_assertion_error(\n                        f\"spurious member '{name}' in object {{path}}\"\n                    )\n            if isinstance(name, Exists) and not self._expected[name]:\n                if name.member_name in self._actual:\n                    self._raise_assertion_error(\n                        f\"spurious member '{name.member_name}' in object {{path}}\"\n                    )\n\n    def _assert_dict_values_equal(self) -> None:\n        for name in self._expected:\n            if isinstance(name, str) and not _is_absent(self._expected[name]):\n                self._assert_json_value_equals_with_item(name)\n\n    @property\n    def _expected_key_names(self) -> Set[str]:\n        keys: Set[str] = set()\n        for k in self._expected.keys():\n            if isinstance(k, str):\n                if not _is_absent(self._expected[k]):\n                    keys.add(k)\n            elif isinstance(k, Exists) and self._expected[k]:\n                keys.add(k.member_name)\n        return keys\n\n    def _assert_arrays_equal(self):\n        if len(self._expected) != len(self._actual):\n            self._raise_different_sizes()\n        for i in range(len(self._expected)):\n            self._assert_json_value_equals_with_item(i)\n\n    def _assert_json_value_equals_with_item(self, item):\n        path = self._path.append(item)\n        expected = self._expected[item]\n        actual = self._actual[item]\n        _JSONComparer(path, expected, actual).assert_()\n\n    def _assert_fundamental_values_equal(self):\n        if self._expected != self._actual:\n            self._raise_different_values()\n\n    def _raise_different_values(self):\n        self._raise_assertion_error(\n            \"element {path} differs: {expected} != {actual}\"\n        )\n\n    def _raise_different_sizes(self):\n        self._raise_assertion_error(\n            \"JSON array {path} differs in size: \"\n            \"{expected_len} != {actual_len}\",\n            expected_len=len(self._expected),\n            actual_len=len(self._actual),\n        )\n\n    def _raise_missing_element(self, keys):\n        if len(keys) == 1:\n            format_string = \"element {elements} missing from element {path}\"\n            elements = repr(next(iter(keys)))\n        else:\n            format_string = \"elements {elements} missing from element {path}\"\n            sorted_keys = sorted(keys)\n            elements = (\n                \", \".join(repr(k) for k in sorted_keys[:-1])\n                + \", and \"\n                + repr(sorted_keys[-1])\n            )\n        self._raise_assertion_error(format_string, elements=elements)\n\n    def _raise_assertion_error(self, format_, **kwargs):\n        kwargs.update(\n            {\n                \"path\": self._path,\n                \"expected\": repr(self._expected),\n                \"actual\": repr(self._actual),\n            }\n        )\n        raise AssertionError(format_.format(**kwargs))\n\n\nclass _JSONPath:\n    def __init__(self, path):\n        self._path = path\n\n    def __str__(self):\n        return self._path\n\n    def append(self, item):\n        return _JSONPath(\"{0}[{1}]\".format(self._path, repr(item)))\n\n\nclass Present:\n    \"\"\"Helper class for presence checks in assert_json_subset().\"\"\"\n\n\ndef _is_present(o: object) -> bool:\n    return o is Present or isinstance(o, Present)\n\n\nclass Absent:\n    \"\"\"Helper class for absence checks in assert_json_subset().\"\"\"\n\n\ndef _is_absent(o: object) -> bool:\n    return o is Absent or isinstance(o, Absent)\n\n\n@deprecated(\"Use Present and Absent instead.\")\nclass Exists:\n    \"\"\"Helper class for existence checks in assert_json_subset().\"\"\"\n\n    def __init__(self, member_name: str) -> None:\n        self.member_name = member_name\n"
    },
    {
      "path": "srittau_python-asserts/asserts/__init__.pyi",
      "content": "import datetime\nfrom collections.abc import Callable, Container, Iterable\nfrom contextlib import AbstractContextManager as ContextManager\nfrom re import Pattern\nfrom types import TracebackType\nfrom typing import Any, Generic, NoReturn, TypeVar\n\nfrom typing_extensions import deprecated\n\n_E = TypeVar(\"_E\", bound=BaseException)\n_S = TypeVar(\"_S\")\n\nclass AssertRaisesContext(Generic[_E]):\n    exception: type[_E]\n    msg_fmt: str\n    def __init__(self, exception: type[_E], msg_fmt: str = ...) -> None: ...\n    def __enter__(self: _S) -> _S: ...\n    def __exit__(\n        self,\n        exc_type: type[BaseException] | None,\n        exc_val: BaseException | None,\n        exc_tb: TracebackType | None,\n    ) -> bool: ...\n    def format_message(self, default_msg: str) -> str: ...\n    def add_test(self, cb: Callable[[_E], object]) -> None: ...\n    @property\n    def exc_val(self) -> _E: ...\n\nclass AssertRaisesErrnoContext(AssertRaisesContext[_E]):\n    expected_errno: int\n    def __init__(\n        self, exception: type[_E], expected_errno: int, msg_fmt: str = ...\n    ) -> None: ...\n\nclass AssertRaisesRegexContext(AssertRaisesContext[_E]):\n    pattern: str\n    def __init__(\n        self, exception: type[_E], pattern: str, msg_fmt: str = ...\n    ) -> None: ...\n\nclass AssertWarnsContext:\n    def __init__(\n        self, warning_class: type[Warning], msg_fmt: str = ...\n    ) -> None: ...\n    def __enter__(self: _S) -> _S: ...\n    def __exit__(\n        self,\n        exc_type: type[BaseException] | None,\n        exc_val: BaseException | None,\n        exc_tb: TracebackType | None,\n    ) -> None: ...\n    def format_message(self) -> str: ...\n    def add_test(self, cb: Callable[[Warning], bool]) -> None: ...\n\nclass AssertWarnsRegexContext(AssertWarnsContext):\n    pattern: str\n    def __init__(\n        self, warning_class: type[Warning], msg_fmt: str = ...\n    ) -> None: ...\n\ndef fail(msg: str = ...) -> NoReturn: ...\ndef assert_true(expr: object, msg_fmt: str = ...) -> None: ...\ndef assert_false(expr: object, msg_fmt: str = ...) -> None: ...\ndef assert_boolean_true(expr: object, msg_fmt: str = ...) -> None: ...\ndef assert_boolean_false(expr: object, msg_fmt: str = ...) -> None: ...\ndef assert_is_none(expr: object, msg_fmt: str = ...) -> None: ...\ndef assert_is_not_none(expr: object, msg_fmt: str = ...) -> None: ...\ndef assert_equal(\n    first: object, second: object, msg_fmt: str = ...\n) -> None: ...\ndef assert_not_equal(\n    first: object, second: object, msg_fmt: str = ...\n) -> None: ...\ndef assert_almost_equal(\n    first: float,\n    second: float,\n    msg_fmt: str = ...,\n    places: int = ...,\n    delta: float = ...,\n) -> None: ...\ndef assert_not_almost_equal(\n    first: float,\n    second: float,\n    msg_fmt: str = ...,\n    places: int = ...,\n    delta: float = ...,\n) -> None: ...\ndef assert_dict_equal(\n    first: dict,\n    second: dict,\n    key_msg_fmt: str = ...,\n    value_msg_fmt: str = ...,\n) -> None: ...\ndef assert_dict_superset(\n    first: dict,\n    second: dict,\n    key_msg_fmt: str = ...,\n    value_msg_fmt: str = ...,\n) -> None: ...\ndef assert_less(first: Any, second: Any, msg_fmt: str = ...) -> None: ...\ndef assert_less_equal(first: Any, second: Any, msg_fmt: str = ...) -> None: ...\ndef assert_greater(first: Any, second: Any, msg_fmt: str = ...) -> None: ...\ndef assert_greater_equal(\n    first: Any, second: Any, msg_fmt: str = ...\n) -> None: ...\ndef assert_regex(\n    text: str, regex: str | Pattern[str], msg_fmt: str = ...\n) -> None: ...\ndef assert_not_regex(\n    text: str, regex: str | Pattern[str], msg_fmt: str = ...\n) -> None: ...\ndef assert_is(first: object, second: object, msg_fmt: str = ...) -> None: ...\ndef assert_is_not(\n    first: object, second: object, msg_fmt: str = ...\n) -> None: ...\ndef assert_in(\n    first: Any, second: Container[Any], msg_fmt: str = ...\n) -> None: ...\ndef assert_not_in(\n    first: Any, second: Container[Any], msg_fmt: str = ...\n) -> None: ...\ndef assert_between(\n    lower_bound: Any, upper_bound: Any, expr: Any, msg_fmt: str = ...\n) -> None: ...\ndef assert_is_instance(\n    obj: object, cls: type | tuple[type, ...], msg_fmt: str = ...\n) -> None: ...\ndef assert_not_is_instance(\n    obj: object, cls: type | tuple[type, ...], msg_fmt: str = ...\n) -> None: ...\ndef assert_count_equal(\n    sequence1: Iterable[Any], sequence2: Iterable[Any], msg_fmt: str = ...\n) -> None: ...\ndef assert_has_attr(\n    obj: object, attribute: str, msg_fmt: str = ...\n) -> None: ...\ndef assert_datetime_about_now(\n    actual: datetime.datetime | None, msg_fmt: str = ...\n) -> None: ...\ndef assert_datetime_about_now_utc(\n    actual: datetime.datetime | None, msg_fmt: str = ...\n) -> None: ...\ndef assert_raises(\n    exception: type[BaseException], msg_fmt: str = ...\n) -> AssertRaisesContext: ...\ndef assert_raises_regex(\n    exception: type[BaseException],\n    regex: str | Pattern[str],\n    msg_fmt: str = ...,\n) -> AssertRaisesContext: ...\ndef assert_raises_errno(\n    exception: type[BaseException], errno: int, msg_fmt: str = ...\n) -> AssertRaisesContext: ...\ndef assert_succeeds(\n    exception: type[BaseException], msg_fmt: str = ...\n) -> ContextManager: ...\ndef assert_warns(\n    warning_type: type[Warning], msg_fmt: str = ...\n) -> AssertWarnsContext: ...\ndef assert_warns_regex(\n    warning_type: type[Warning], regex: str, msg_fmt: str = ...\n) -> AssertWarnsContext: ...\ndef assert_json_subset(\n    first: dict[Any, Any] | list[Any],  # dict key can be 'str' or 'Exists'\n    second: dict[str, Any] | list[Any] | str | bytes,\n) -> None: ...\n\nclass Present: ...\nclass Absent: ...\n\n@deprecated(\"Use Present and Absent instead.\")\nclass Exists:\n    member_name: str\n    def __init__(self, member_name: str) -> None: ...\n"
    },
    {
      "path": "srittau_python-asserts/asserts/py.typed",
      "content": ""
    }
  ],
  "Patch": "--- a/srittau_python-asserts/asserts/__init__.py\n+++ b/srittau_python-asserts/asserts/__init__.py\n@@ -96,7 +96,7 @@\n     * expr - tested expression\n     \"\"\"\n \n-    if not expr:\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",
  "BuggyCodeLocation": [
    {
      "file": "srittau_python-asserts/asserts/__init__.py",
      "function": {
        "85": "assert_boolean_true"
      },
      "content_all": {
        "96": "    * expr - tested expression\n",
        "97": "    \"\"\"\n",
        "98": "\n",
        "99": "    if not expr:\n",
        "100": "        msg = \"{!r} is not True\".format(expr)\n"
      },
      "content_change": {
        "99": "    if not expr:\n"
      }
    },
    {
      "file": "srittau_python-asserts/asserts/__init__.py",
      "function": null,
      "content_all": {
        "101": "        fail(msg_fmt.format(msg=msg, expr=expr))\n",
        "102": "\n"
      },
      "content_change": {}
    }
  ],
  "Source": "Human",
  "Command": [
    "python -m unittest test_asserts.py"
  ],
  "Token": 1426,
  "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/README.md",
      "content": "1 # Python Asserts\n2 \n3 [![License](https://img.shields.io/pypi/l/asserts.svg)](https://pypi.python.org/pypi/asserts/)\n4 [![PyPI - Python Version](https://img.shields.io/pypi/pyversions/asserts)](https://pypi.python.org/pypi/asserts/)\n5 [![GitHub](https://img.shields.io/github/release/srittau/python-asserts/all.svg)](https://github.com/srittau/python-asserts/releases/)\n6 [![pypi](https://img.shields.io/pypi/v/asserts.svg)](https://pypi.python.org/pypi/asserts/)\n7 [![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)\n8 \n9 Stand-alone Assertions for Python\n10 \n11 This package provides a few advantages over the assertions provided by\n12 unittest.TestCase:\n13 \n14 - Can be used stand-alone, for example:\n15   - In test cases, not derived from TestCase.\n16   - In fake and mock classes.\n17   - In implementations as rich alternative to the assert statement.\n18 - PEP 8 compliance.\n19 - Custom stand-alone assertions can be written easily.\n20 - Arguably a better separation of concerns, since TestCase is responsible\n21   for test running only, if assertion functions are used exclusively.\n22 \n23 There are a few regressions compared to assertions from TestCase:\n24 \n25 - The default assertion class (`AssertionError`) can not be overwritten. This\n26   is rarely a problem in practice.\n27 - asserts does not support the `addTypeEqualityFunc()` functionality.\n28 \n29 Usage:\n30 \n31 ```python\n32 >>> from asserts import assert_true, assert_equal, assert_raises\n33 >>> my_var = 13\n34 >>> assert_equal(13, my_var)\n35 >>> assert_true(True, msg=\"custom failure message\")\n36 >>> with assert_raises(KeyError):\n37 ...     raise KeyError()\n38 ```\n39 \n40 Failure messages can be customized:\n41 \n42 ```python\n43 >>> assert_equal(13, 14, msg_fmt=\"{got} is wrong, expected {expected}\")\n44 Traceback (most recent call last):\n45   ...\n46 AssertionError: 14 is wrong, expected 13\n47 ```"
    },
    {
      "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] (...truncated)"
    },
    {
      "path": "srittau_python-asserts/pyproject.toml",
      "content": "1 [tool.poetry]\n2 name = \"asserts\"\n3 version = \"0.13.2.dev0\"\n4 description = \"Stand-alone Assertions\"\n5 readm(...truncated)"
    }
  ],
  "TokenAll": 38740,
  "FilteredLevel": 1500,
  "Results": {
    "model": "GPT-4o",
    "Difficulty": "Medium",
    "issue_origin": {
      "title": "Incomplete README Example for assert_true",
      "description": "In the README.md file, the example for 'assert_true' lacks an explanation for the msg parameter's usage. Specifically, line 35 shows 'assert_true(True, msg=\"custom failure message\")', but this example does not indicate the circumstances under which the custom failure message will be displayed. Adding a failed assertion example to demonstrate the custom message will improve clarity.",
      "explanation": "### Issue Summary\nThe issue revolves around an incomplete example provided in the project's `README.md` file. Specifically, the example usage of the `assert_true` function on line 35 lacks an illustration of the `msg` parameter's usage under failure conditions. The current example:\n```python\n>>> assert_true(True, msg=\"custom failure message\")\n```\ndoes not clarify when the custom failure message will be displayed. The issue requests adding an example to demonstrate a failed assertion, thus showing how the custom message is used.\n\n### Content of the Commit\nTo resolve this issue, the following changes were made to the `README.md` file:\n\n```diff\n1 # Python Asserts\n2  \n...\n34 >>> assert_equal(13, my_var)\n35 >>> assert_true(True, msg=\"custom failure message\")\n+>>> assert_true(False, msg=\"custom failure message\")\n+Traceback (most recent call last):\n+  ...\n+AssertionError: custom failure message\n36 >>> with assert_raises(KeyError):\n37 ...     raise KeyError()\n...\n\n```\n\nThis addition introduces a failure scenario for `assert_true` where the boolean expression `False` is passed, causing the custom failure message `\"custom failure message\"` to be raised.\n\n### Explanation of the Commit\nThe commit primarily consists of adding the following lines:\n```python\n>>> assert_true(False, msg=\"custom failure message\")\nTraceback (most recent call last):\n    ...\nAssertionError: custom failure message\n```\n\n#### How the Commit Solves the Issue\n1. **Demonstrates Failure Behavior**: By adding an example where `assert_true` fails (`assert_true(False)`), it explicitly shows when the custom `msg` parameter would be utilized.\n2. **Clarifies Usage of `msg` Parameter**: Via this example, it becomes clear that when the expression passed to `assert_true` evaluates to `False`, the custom message specified in `msg` is included in the `AssertionError`. \n3. **Enhanced Documentation**: This improvement enhances the documentation by providing a comprehensive understanding of the `assert_true` function’s behavior, especially under failure conditions, making it more user-friendly.\n\n### Detailed Explanation for Developers\n\n**Original Scenario:**\nThe example in question:\n```python\n>>> assert_true(True, msg=\"custom failure message\")\n```\nThis example does not fail and, therefore, does not trigger the display of the custom message. Users might be confused about how and when this `msg` parameter affects the output.\n\n**Improved Documentation:**\nBy adding a failing example:\n```python\n>>> assert_true(False, msg=\"custom failure message\")\nTraceback (most recent call last):\n    ...\nAssertionError: custom failure message\n```\n1. **Importance of Example Failure**: An example that fails is crucial for understanding how error messages are composed and displayed. It validates that the `msg` parameter is indeed integrated into the failure message.\n2. **Message Customization**: Developers now understand that the `msg` argument can be customized to provide meaningful error messages, thus enhancing debugging and clarity during unit testing.\n3. **Educational Benefit**: The improved example educates users on the full capabilities of the `assert_true` function beyond just asserting truthiness, including customizing failure messages, which is a common need in testing to provide context-specific error information.\n\nBy making these documentation changes, the commit effectively resolves the issue, offering a clearer, more instructive example for potential users of the library."
    },
    "issue_message": {
      "title": "AssertionError not raised in assert_boolean_true",
      "description": "In the tests for `assert_boolean_true` in `test_asserts.py`, the function does not raise an AssertionError as expected. Specifically, in test cases `test_assert_boolean_true__false__custom_message` and `test_assert_boolean_true__truthy__default_message`, the code raises an AssertionError stating 'no AssertionError raised'. Review the implementation of `assert_boolean_true` to ensure it correctly raises an AssertionError for false or non-truthy expressions.",
      "explanation": "### Issue Summary\n\nThe issue reported involves the `assert_boolean_true` function within the `test_asserts.py` file. Specifically, this function is failing to raise an `AssertionError` when it encounters a false or non-truthy expression. The failing test cases are `test_assert_boolean_true__false__custom_message` and `test_assert_boolean_true__truthy__default_message`. When these tests are executed, they end up raising an `AssertionError` with the message \"no AssertionError raised,\" indicating that the expected `AssertionError` was not triggered by the `assert_boolean_true` function.\n\n### Commit Analysis\n\nTo resolve this issue, the commit needs to ensure that the `assert_boolean_true` function properly raises an `AssertionError` when the expression being tested is false or non-truthy.\n\n#### Content of the Commit\nThe specific code relevant to the issue in `srittau_python-asserts/asserts/__init__.py` is:\n```python\n47 def assert_true(expr, msg_fmt=\"{msg}\"):\n48     \"\"\"Fail the test unless the expression is truthy.\n49 \n50     >>> assert_true(\"Hello World!\")\n51     >>> assert_true(\"\")\n52     Traceback (most recent call last):\n53         ...\n54     AssertionError: '' is not truthy\n55 \n56     The following msg_fmt arguments are supported:\n57     * msg - the default error message\n58     * expr - tested expression\n59 \n60     \"\"\"\n61     default_msg = f\"'{expr}' is not truthy\"\n62     if not expr:\n63         fail(msg_fmt.format(msg=default_msg, expr=expr))\n```\n\n### Explanation\n\n1. **Function Definition `assert_true` at Line 47**: \n   This function is designed to test the \"truthiness\" of an expression. If the expression is not truthy, it should raise an `AssertionError`.\n\n2. **Docstring for `assert_true`**:\n   - The docstring explains that the purpose of the function is to fail the test unless the expression is truthy.\n   - It provides examples where `assert_true(\"Hello World!\")` works without error, while `assert_true(\"\")` should raise an `AssertionError`.\n\n3. **Implementation Details**:\n   - **Line 47 - Function Definition**: `def assert_true(expr, msg_fmt=\"{msg}\"):`\n   - **Line 61**: It sets up a default message explaining that the expression is not truthy.\n   - **Line 62**: It checks whether the expression is truthy using a simple `if not expr:` condition.\n   - **Line 63**: If the expression is not truthy, it calls the `fail` function with a formatted message that raises an `AssertionError`.\n\n### How the Commit Solves the Issue\n\n1. **Cause of the Issue**:\n   - The issue stemmed from `assert_boolean_true` not functioning as expected. The problem was likely due to a logical error or a missing implementation line that did not raise the `AssertionError` properly when a false or non-truthy value was encountered.\n\n2. **Fix implemented**:\n   - The crucial part of the fix is the addition or correction in the `assert_true` (or an equivalent function) where it properly performs a truthiness check with `if not expr:`.\n   - If the expression is false or non-truthy, it calls the `fail` function with the appropriate message, thereby raising an `AssertionError`.\n\n3. **Effectiveness of the Commit**:\n   - By ensuring that `assert_true` properly evaluates the truthiness and raises an appropriate error when necessary, the test cases `test_assert_boolean_true__false__custom_message` and `test_assert_boolean_true__truthy__default_message` should now pass successfully. These test cases were failing previously because the `AssertionError` was not raised as expected.\n\n4. **Validation of the Fix**:\n   - The updated implementation can be tested by executing the test cases again. If the problem is fixed, the `AssertionError` should be raised correctly when the conditions for truthiness are not met.\n   - The expected output asserts that the false or non-truthy condition triggers the `AssertionError`.\n\n### Conclusion\nThe commit resolves the issue by guaranteeing that the `assert_boolean_true` function raises an `AssertionError` when given a false or non-truthy expression. This is achieved by ensuring that the logical check within the function rigorously tests the expression and calls the `fail` function with the appropriate error message when the expression is false. This commitment to raising the correct errors ensures that the test cases no longer fail for the expected scenarios, thus preserving the integrity of the assertion function."
    },
    "issue_ground": {
      "title": "Inconsistent Boolean Assertions Cause False Failures",
      "description": "There is an issue in the assertion library where the `assert_boolean_true` function inconsistently evaluates expressions, causing unexpected assertion failures. Specifically, the function currently evaluates expressions using a truthy check (`if not expr:`), which leads to expressions that are not strictly `True` being misinterpreted. Users may experience assertion failures even when the expression should be considered true in the context of boolean logic.\n\nThis impacts the reliability of tests that use `assert_boolean_true`, causing tests to fail incorrectly. For instance, expressions like non-zero integers or non-empty strings, which may be truthy, will cause `assert_boolean_true` to fail.\n\nThe expected behavior is for `assert_boolean_true` to strictly check for the `True` constant, ensuring that only expressions that exactly equal `True` pass the assertion. This change should increase the precision and reliability of boolean assertions, particularly in test cases where the exact value of `True` is necessary.\n\nThis issue needs to be addressed to ensure that `assert_boolean_true` works as intended, only passing when the expression is explicitly `True` and not merely truthy.",
      "explanation": "### Summary of the Issue\n\nThe issue at hand is related to the `assert_boolean_true` function in the assertion library. This function is currently evaluating expressions using a truthy check (e.g., `if not expr:`), which causes it to fail assertions for values that are truthy but not strictly `True` in a boolean sense (such as non-zero integers or non-empty strings). Consequently, this leads to unexpected assertion failures, impacting the reliability of tests.\n\n### Explanation of the Commit Content\n\nThe corrective commit should ideally modify the `assert_boolean_true` function to ensure that it checks if the given expression strictly equals the boolean constant `True`. This involves changing the evaluation from a general truthiness check to a strict equality check against `True`.\n\n### Detailed Explanation\n\n#### Problematic Code\n\nIn `srittau_python-asserts/asserts/__init__.py`, the existing code for `assert_boolean_true` might look something like this (based on the understanding of the issue description):\n\n```python\ndef assert_boolean_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 converted into string\n    \"\"\"\n    if not expr:\n        fail(msg_fmt.format(msg=f\"'{expr}' is not truthy\", expr=expr))\n```\n\n#### Identified Issue\n\nThe issue here is in the line `if not expr:`. This condition checks if `expr` is \"falsy\", meaning it captures a wide range of values beyond just the boolean `False` (e.g., zero, empty strings, empty lists, `None`, etc.).\n\n#### Commit to Fix the Issue\n\nA commit to fix this issue would modify this line to ensure that only expressions which are exactly `True` pass the assertion. The revised code might look like this:\n\n```python\ndef assert_boolean_true(expr, msg_fmt=\"{msg}\"):\n    \"\"\"Fail the test unless the expression is exactly True.\n    \n    >>> assert_boolean_true(True)\n    >>> assert_boolean_true(False)\n    Traceback (most recent call last):\n        ...\n    AssertionError: 'False' is not True\n    \n    The following msg_fmt arguments are supported:\n    * msg - the default error message\n    * expr - tested expression converted into string\n    \"\"\"\n    if expr is not True:\n        fail(msg_fmt.format(msg=f\"'{expr}' is not True\", expr=expr))\n```\n\n#### Explanation of the Fix\n\n- **Strict Comparison**: The condition `if expr is not True:` ensures that the expression `expr` will only pass the assertion if it strictly equals the boolean `True`.\n- **Enhanced Debug Information**: The failure message has been updated to clearly indicate that the expression was expected to be `True`.\n\n#### Testing Changes\n\nIn `srittau_python-asserts/test_asserts.py`, we should ensure that tests reflect the new functionality:\n\n```python\ndef test_assert_boolean_true__false__custom_message(self):\n    with self.assertRaises(AssertionError):\n        assert_boolean_true(False, \"{msg};{expr}\")\n\ndef test_assert_boolean_true__truthy__default_message(self):\n    with self.assertRaises(AssertionError):\n        assert_boolean_true(1)  # Should now fail since 1 is truthy but not exactly True\n\ndef test_assert_boolean_true__true(self):\n    assert_boolean_true(True)  # Should pass\n```\n\nThis set of tests verifies:\n1. That `False` fails the assertion as expected.\n2. That a truthy value (`1`) which is not strictly `True` fails the assertion.\n3. That the boolean value `True` passes the assertion.\n\n### Conclusion\n\nThe issue with `assert_boolean_true` stemmed from its use of a generalized truthy check, which caused it to incorrectly fail for a range of truthy values. The commit changing this to a strict `True` check ensures only values that are exactly `True` pass, thereby increasing the assertion's precision and reliability. The changes directly address and resolve the inconsistency causing false assertion failures, ensuring more reliable test outcomes."
    },
    "issue_ground_truth": {
      "title": "Inconsistent Boolean Assertions Cause False Failures",
      "description": "There is an issue in the assertion library where the `assert_boolean_true` function inconsistently evaluates expressions, causing unexpected assertion failures. Specifically, the function currently evaluates expressions using a truthy check (`if not expr:`), which leads to expressions that are not strictly `True` being misinterpreted. Users may experience assertion failures even when the expression should be considered true in the context of boolean logic.\n\nThis impacts the reliability of tests that use `assert_boolean_true`, causing tests to fail incorrectly. For instance, expressions like non-zero integers or non-empty strings, which may be truthy, will cause `assert_boolean_true` to fail.\n\nThe expected behavior is for `assert_boolean_true` to strictly check for the `True` constant, ensuring that only expressions that exactly equal `True` pass the assertion. This change should increase the precision and reliability of boolean assertions, particularly in test cases where the exact value of `True` is necessary.\n\nThis issue needs to be addressed to ensure that `assert_boolean_true` works as intended, only passing when the expression is explicitly `True` and not merely truthy.",
      "explanation": "### Summary of the Issue\n\nThe issue titled **\"Inconsistent Boolean Assertions Cause False Failures\"** describes a problem where the `assert_boolean_true` function in an assertion library of the `srittau_python-asserts` package incorrectly evaluates boolean expressions. This function uses a general truthy check (`if not expr:`), which causes false assertion failures for expressions that are truthy but not strictly equal to the `True` constant. For example, non-zero integers or non-empty strings are truthy but not strictly `True`, resulting in unexpected assertion failures.\n\n### Detailed Content of the Commit\n\nThe commit information indicates that the developer modified the behavior of the `assert_boolean_true` function. The commit includes the following actions:\n- **Changed the evaluation method**: The function was originally using `if not expr:` to evaluate the expression. This check was changed to `if expr is not True:`.\n\n### Explanation of the Cause and Solution\n\n#### Cause of the Issue\n\nThe root cause of the issue lies in the way the `assert_boolean_true` function evaluates expressions. The use of `if not expr:` in Python checks for truthiness, meaning it interprets any value that is not `False`, `None`, `0`, `''` (empty string), or an empty collection as `True`. This is a broad check and includes many values (such as `1`, non-empty strings, and non-empty collections) that are truthy but not strictly the boolean `True` constant. Consequently, users experienced false assertion failures when using `assert_boolean_true` with expressions that were truthy but not exactly `True`.\n\n#### Solution Implemented in the Commit\n\nTo solve this issue, the commit made a simple but critical change to the logic of the `assert_boolean_true` function:\n- **Changed the evaluation logic**: The check was updated to `if expr is not True:`. \n\nThis adjustment ensures that the function now strictly checks for the boolean `True` value, rather than any truthy value. This means that `assert_boolean_true` will only pass when the expression is explicitly `True` and will fail for any other truthy values (like `1`, `\"non-empty string\"`, etc.).\n\n#### How the Commit Solves the Issue\n\nThe commit addresses the problem by refining the expression evaluation logic in the `assert_boolean_true` function. The switch from `if not expr:` to `if expr is not True:` ensures that only expressions that are strictly `True` (the boolean constant) will pass the assertion. This change eliminates false failures caused by truthy expressions that are not strictly `True`. Therefore, test reliability is improved, and the function behaves as expected by only passing when the tested expression is the boolean `True`.\n\n### Final Explanation of the Solution\n\nThe solution comprehensively resolves the issue by updating the expression logic in the assertion function. This ensures that the function now adheres to the intended check for the boolean `True` rather than any broadly truthy value. Users can now rely on `assert_boolean_true` for precise boolean checks without encountering unexpected assertion failures due to truthy but not strictly `True` values. The fix makes the function's behavior more predictable and enhances the reliability of tests using this function."
    },
    "location_origin": [
      {
        "location": "\n[\n    {\n        \"file\": \"srittau_python-asserts/README.md\",\n        \"function\": {\n            \"31\": \"Usage\"\n        },\n        \"content_all\": {\n            \"32\": \">>> from asserts import assert_true, assert_equal, assert_raises\\n\",\n            \"33\": \">>> my_var = 13\\n\",\n            \"34\": \">>> assert_equal(13, my_var)\\n\",\n            \"35\": \">>> assert_true(True, msg=\\\"custom failure message\\\")\\n\",\n            \"36\": \">>> with assert_raises(KeyError):\\n\",\n            \"37\": \"...     raise KeyError()\\n\",\n            \"38\": \""
      }
    ],
    "location_message": [
      {
        "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 expression\n",
          "59": " \n",
          "60": "    \"\"\"\n",
          "61": "    default_msg = f\"'{expr}' is not truthy\"\n",
          "62": "    if not expr:\n",
          "63": "        fail(msg_fmt.format(msg=default_msg, expr=expr))\n"
        },
        "content_change": {
          "62": "    if not expr:\n",
          "63": "        fail(msg_fmt.format(msg=default_msg, expr=expr))\n"
        }
      },
      {
        "file": "srittau_python-asserts/test_asserts.py",
        "function": {
          "18": "assert_boolean_true",
          "142": "test_assert_boolean_true__false__custom_message",
          "146": "test_assert_boolean_true__truthy__default_message"
        },
        "content_all": {
          "15": "    assert_almost_equal,\n",
          "16": "    assert_between,\n",
          "17": "    assert_boolean_false,\n",
          "18": "    assert_boolean_true,\n",
          "19": "    assert_count_equal,\n",
          "20": "    assert_datetime_about_now,\n",
          "21": "    assert_datetime_about_now_utc,\n",
          "22": "    assert_dict_equal,\n",
          "23": "    assert_dict_superset,\n",
          "24": "    assert_equal,\n",
          "25": "    assert_false,\n",
          "26": "    assert_greater,\n",
          "27": "    assert_greater_equal,\n",
          "28": "    assert_has_attr,\n",
          "29": "    assert_in,\n",
          "30": "    assert_is,\n",
          "31": "    assert_is_instance,\n",
          "32": "    assert_is_none,\n",
          "33": "    assert_is_not,\n",
          "34": "    assert_is_not_none,\n",
          "35": "    assert_json_subset,\n",
          "36": "    assert_less,\n",
          "37": "    assert_less_equal,\n",
          "38": "    assert_not_almost_equal,\n",
          "39": "    assert_not_equal,\n",
          "40": "    assert_not_in,\n",
          "41": "    assert_not_is_instance,\n",
          "42": "    assert_not_regex,\n",
          "43": "    assert_raises,\n",
          "44": "    assert_raises_errno,\n",
          "45": "    assert_raises_regex,\n",
          "46": "    assert_regex,\n",
          "82": "    raise AssertionError(\"no AssertionError raised\")\n",
          "142": "    assert_boolean_true(\"Foo\", \"{msg};{expr}\")\n",
          "146": "    assert_boolean_true(1)\n"
        },
        "content_change": {
          "142": "    assert_boolean_true(\"Foo\", \"{msg};{expr}\")\n",
          "146": "    assert_boolean_true(1)\n"
        }
      }
    ],
    "location_ground": [
      {
        "file": "srittau_python-asserts/asserts/__init__.py",
        "function": {
          "47": "assert_boolean_true"
        },
        "content_all": {
          "44": "    raise AssertionError(msg or \"assertion failure\")\n",
          "45": "\n",
          "46": "\n",
          "47": "def assert_boolean_true(expr, msg_fmt=\"{msg}\"):\n",
          "48": "    \"\"\"Fail the test unless the expression is truthy.\n",
          "49": "    \n",
          "50": "    >>> assert_boolean_true(\"Hello World!\")\n",
          "51": "    >>> assert_boolean_true(\"\")\n",
          "52": "    Traceback (most recent call last):\n"
        },
        "content_change": {
          "47": "def assert_boolean_true(expr, msg_fmt=\"{msg}\"):\n",
          "48": "    \"\"\"Fail the test unless the expression is exactly True.\n",
          "56": "    if expr is not True:\n",
          "57": "        fail(msg_fmt.format(msg=f\"'{expr}' is not True\", expr=expr))\n"
        }
      },
      {
        "file": "srittau_python-asserts/test_asserts.py",
        "function": {
          "18": "test_assert_boolean_true__false__custom_message"
        },
        "content_all": {
          "16": "    def test_assert_boolean_true__false__custom_message(self):\n",
          "17": "        with self.assertRaises(AssertionError):\n",
          "18": "            assert_boolean_true(False, \"{msg};{expr}\")\n",
          "19": "\n",
          "20": "    def test_assert_boolean_true__truthy__default_message(self):\n",
          "21": "        with self.assertRaises(AssertionError):\n",
          "22": "            assert_boolean_true(1)  # Should now fail since 1 is truthy but not exactly True\n"
        },
        "content_change": {
          "16": "    def test_assert_boolean_true__false__custom_message(self):\n",
          "17": "        with self.assertRaises(AssertionError):\n",
          "18": "            assert_boolean_true(False, \"{msg};{expr}\")\n",
          "19": "\n",
          "20": "    def test_assert_boolean_true__truthy__default_message(self):\n",
          "21": "        with self.assertRaises(AssertionError):\n",
          "22": "            assert_boolean_true(1)  # Should now fail since 1 is truthy but not exactly True\n"
        }
      },
      {
        "file": "srittau_python-asserts/test_asserts.py",
        "function": {
          "24": "test_assert_boolean_true__true"
        },
        "content_all": {
          "23": "    \n",
          "24": "    def test_assert_boolean_true__true(self):\n",
          "25": "        assert_boolean_true(True)  # Should pass\n",
          "26": "\n",
          "27": "\n",
          "28": "\n",
          "29": "\n"
        },
        "content_change": {
          "24": "    def test_assert_boolean_true__true(self):\n",
          "25": "        assert_boolean_true(True)  # Should pass\n",
          "26": "\n"
        }
      }
    ],
    "location_ground_exp": [
      {
        "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 expr\n",
          "59": "    * fmt - format arguments for the message,\n",
          "60": "      e.g. {'msg_fmt': \"Expected {expected}, got {got}\"}\n",
          "61": "    \"\"\"\n",
          "62": "    if not expr:\n",
          "63": "        fail(msg_fmt.format(msg=\"{!r} is not truthy\".format(expr)))\n",
          "64": "\n"
        },
        "content_change": {
          "62": "    if expr is not True:\n"
        }
      },
      {
        "file": "srittau_python-asserts/asserts/__init__.py",
        "function": {
          "18": "assert_boolean_true"
        },
        "content_all": {
          "17": "def assert_boolean_false(expr: Any, msg: str = \"\") -> None:\n",
          "18": "def assert_boolean_true(expr: Any, msg: str = \"\") -> None:\n",
          "19": "    \"\"\"Assert that `expr` is exactly `True`.\n",
          "20": "\n",
          "21": "    >>> assert_boolean_true(True)\n",
          "22": "    >>> assert_boolean_true(1)\n",
          "23": "    Traceback (most recent call last):\n",
          "24": "        ...\n",
          "25": "    AssertionError: '' is not True\n",
          "26": "\n",
          "27": "    \"\"\"\n",
          "28": "    if expr is not True:\n",
          "29": "        fail(msg or \"{!r} is not True\".format(expr))\n"
        },
        "content_change": {
          "28": "    if expr is not True:\n"
        }
      }
    ],
    "location_ground_truth": [
      {
        "file": "srittau_python-asserts/asserts/__init__.py",
        "function": {
          "85": "assert_boolean_true"
        },
        "content_all": {
          "96": "    * expr - tested expression\n",
          "97": "    \"\"\"\n",
          "98": "\n",
          "99": "    if not expr:\n",
          "100": "        msg = \"{!r} is not True\".format(expr)\n"
        },
        "content_change": {
          "99": "    if not expr:\n"
        }
      },
      {
        "file": "srittau_python-asserts/asserts/__init__.py",
        "function": null,
        "content_all": {
          "101": "        fail(msg_fmt.format(msg=msg, expr=expr))\n",
          "102": "\n"
        },
        "content_change": {}
      }
    ],
    "patch_i": "\n--- a/srittau_python-asserts/README.md\n+++ b/srittau_python-asserts/README.md\n@@ -34,6 +34,10 @@\n >>> assert_equal(13, my_var)\n >>> assert_true(True, msg=\"custom failure message\")\n >>> assert_true(False, msg=\"custom failure message\")\n+Traceback (most recent call last):\n+  ...\n+AssertionError: custom failure message\n >>> with assert_raises(KeyError):\n ...     raise KeyError()\n ",
    "patch_im": "Sure, based on the issue statement and the code information provided, here is the required patch to resolve the problem in your repository:\n\n```diff\n--- a/srittau_python-asserts/asserts/__init__.py\n+++ b/srittau_python-asserts/asserts/__init__.py\n@@ -1,10 +1,11 @@\n-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 expression\\n59 \\n60     \\\"\\\"\\\"\\n61     default_msg = f\\\"'{expr}' is not truthy\\\"\\n62     if not expr:\\n63         fail(msg_fmt.format(msg=default_msg, expr=expr))\\n\n+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     Assertion",
    "patch_il": "\n--- a/srittau_python-asserts/README.md\n+++ b/srittau_python-asserts/README.md\n@@ -34,6 +34,10 @@\n >>> assert_equal(13, my_var)\n >>> assert_true(True, msg=\"custom failure message\")\n >>> assert_true(False, msg=\"custom failure message\")\n+Traceback (most recent call last):\n+  ...\n+AssertionError: custom failure message\n >>> with assert_raises(KeyError):\n ...     raise KeyError()\n",
    "patch_iml": "\n--- a/srittau_python-asserts/asserts/__init__.py\n+++ b/srittau_python-asserts/asserts/__init__.py\n@@ -1,4 +1,4 @@\n-    raise AssertionError(msg or \\\"assertion failure\\\")\n+    raise AssertionError(msg or \"assertion failure\")\n\n\n def assert_true(expr, msg_fmt=\"{msg}\"):\n@@ -1,27 +1,31 @@\n def euclidean(a, b):\n-    while b:\n-        a, b = b, a % b\n-    return a\n+    if b == 0:\n+        return a\n+    return euclidean(b, a % b)\n+\n-\n def bresenham(x0, y0, x1, y1):\n     points = []\n     dx = abs(x1 - x0)\n     dy = abs(y1 - y0)\n-    sx = 1 if x0 < x1 else -1\n-    sy = 1 if y0 < y1 else -1\n-    err = dx - dy\n+    x, y = x0, y0\n+    sx = -1 if x0 > x1 else 1\n+    sy = -1 if y0 > y1 else 1\n\n-    while True:\n-        points.append((x0, y0))\n-        if x0 == x1 and y0 == y1:\n-            break\n-        e2 = 2 * err\n-        if e2 > -dy:\n+    if dx > dy:\n+        err = dx / 2.0\n+        while x != x1:\n+            points.append((x, y))\n             err -= dy\n-            x0 += sx\n-        if e2 < dx:\n-            err += dx\n-            y0 += sy\n+            if err < 0:\n+                y += sy\n+                err += dx\n+            x += sx\n+    else:\n+        err = dy / 2.0\n+        while y != y1:\n+            points.append((x, y))\n+            err -= dx\n+            if err < 0:\n+                x += sx\n+                err += dy\n+            y += sy\n\n+    points.append((x, y))\n     return points\n",
    "patch_ground": "\n--- a/srittau_python-asserts/asserts/__init__.py\n+++ b/srittau_python-asserts/asserts/__init__.py\n@@ -47,15 +47,17 @@\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 converted into string\n     \\\"\\\"\\\"\n-    if not expr:\n-        fail(msg_fmt.format(msg=f\\\"'{expr}' is not truthy\\\", expr=expr))\n+    if expr is not True:\n+        fail(msg_fmt.format(msg=f\\\"'{expr}' is not True\\\", expr=expr))\n \n",
    "patch_ground_location": "\n--- a/srittau_python-asserts/asserts/__init__.py\n+++ b/srittau_python-asserts/asserts/__init__.py\n@@ -44,12 +44,12 @@\n     raise AssertionError(msg or \"assertion failure\")\n \n \n-def assert_boolean_true(expr, msg_fmt=\"{msg}\"):\n-    \"\"\"Fail the test unless the expression is truthy.\n+def assert_boolean_true(expr, msg_fmt=\"{msg}\"):\n+    \"\"\"Fail the test unless the expression is exactly True.\n \n-    >>> assert_true(\"Hello World!\")\n-    >>> assert_true(\"\")\n-    Traceback (most recent call last):\n+    >>> assert_boolean_true(True)\n+    >>> assert_boolean_true(False)\n+    Traceback (most recent call last):\n         ...\n-    AssertionError: '' is not truthy\n+    AssertionError: 'False' is not True\n \n     The following msg_fmt arguments are supported:\n     * msg - the default error message\n@@ -56,7 +56,7 @@\n     * expr - tested expression converted into string\n     \"\"\"\n-    if not expr:\n-        fail(msg_fmt.format(msg=f\"'{expr}' is not truthy\", expr=expr))\n+    if expr is not True:\n+        fail(msg_fmt.format(msg=f\"'{expr}' is not True\", expr=expr))\n",
    "patch_ground_exp": "\n--- a/srittau_python-asserts/asserts/__init__.py\n+++ b/srittau_python-asserts/asserts/__init__.py\n@@ -62,7 +62,7 @@\n     * expr - tested expr\n     * fmt - format arguments for the message,\n       e.g. {'msg_fmt': \"Expected {expected}, got {got}\"}\n     \"\"\"\n-    if not expr:\n+    if expr is not True:\n         fail(msg_fmt.format(msg=\"{!r} is not truthy\".format(expr)))\n \n def assert_boolean_false(expr: Any, msg: str = \"\") -> None:\n@@ -28,7 +28,7 @@\n     \"\"\"\n-    if not expr:\n+    if expr is not True:\n         fail(msg or \"{!r} is not True\".format(expr))\n",
    "patch_ground_all": "\n--- a/srittau_python-asserts/asserts/__init__.py\n+++ b/srittau_python-asserts/asserts/__init__.py\n@@ -95,8 +95,8 @@\n     * expr - tested expression\n     \"\"\"\n \n-    if not expr:\n-        msg = \"{!r} is not True\".format(expr)\n+    if expr is not True:\n+        msg = \"{!r} is not the boolean True\".format(expr)\n         fail(msg_fmt.format(msg=msg, expr=expr))\n \n \n",
    "patch_ground_truth": "--- a/srittau_python-asserts/asserts/__init__.py\n+++ b/srittau_python-asserts/asserts/__init__.py\n@@ -96,7 +96,7 @@\n     * expr - tested expression\n     \"\"\"\n \n-    if not expr:\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",
    "message": "======================================================================\nFAIL: test_assert_boolean_true__false__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 142, in test_assert_boolean_true__false__custom_message\n    assert_boolean_true(\"Foo\", \"{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\n\n======================================================================\nFAIL: test_assert_boolean_true__truthy__default_message (test_asserts.AssertTest)\n----------------------------------------------------------------------\nTraceback (most recent call last):\n  File \"/home/user/Projects/user/repobench/repoben/python-asserts/srittau_python-asserts/test_asserts.py\", line 146, in test_assert_boolean_true__truthy__default_message\n    assert_boolean_true(1)\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/README.md",
        "content": "1 # Python Asserts\n2 \n3 [![License](https://img.shields.io/pypi/l/asserts.svg)](https://pypi.python.org/pypi/asserts/)\n4 [![PyPI - Python Version](https://img.shields.io/pypi/pyversions/asserts)](https://pypi.python.org/pypi/asserts/)\n5 [![GitHub](https://img.shields.io/github/release/srittau/python-asserts/all.svg)](https://github.com/srittau/python-asserts/releases/)\n6 [![pypi](https://img.shields.io/pypi/v/asserts.svg)](https://pypi.python.org/pypi/asserts/)\n7 [![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)\n8 \n9 Stand-alone Assertions for Python\n10 \n11 This package provides a few advantages over the assertions provided by\n12 unittest.TestCase:\n13 \n14 - Can be used stand-alone, for example:\n15   - In test cases, not derived from TestCase.\n16   - In fake and mock classes.\n17   - In implementations as rich alternative to the assert statement.\n18 - PEP 8 compliance.\n19 - Custom stand-alone assertions can be written easily.\n20 - Arguably a better separation of concerns, since TestCase is responsible\n21   for test running only, if assertion functions are used exclusively.\n22 \n23 There are a few regressions compared to assertions from TestCase:\n24 \n25 - The default assertion class (`AssertionError`) can not be overwritten. This\n26   is rarely a problem in practice.\n27 - asserts does not support the `addTypeEqualityFunc()` functionality.\n28 \n29 Usage:\n30 \n31 ```python\n32 >>> from asserts import assert_true, assert_equal, assert_raises\n33 >>> my_var = 13\n34 >>> assert_equal(13, my_var)\n35 >>> assert_true(True, msg=\"custom failure message\")\n36 >>> with assert_raises(KeyError):\n37 ...     raise KeyError()\n38 ```\n39 \n40 Failure messages can be customized:\n41 \n42 ```python\n43 >>> assert_equal(13, 14, msg_fmt=\"{got} is wrong, expected {expected}\")\n44 Traceback (most recent call last):\n45   ...\n46 AssertionError: 14 is wrong, expected 13\n47 ```"
      },
      {
        "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] (...truncated)"
      },
      {
        "path": "srittau_python-asserts/pyproject.toml",
        "content": "1 [tool.poetry]\n2 name = \"asserts\"\n3 version = \"0.13.2.dev0\"\n4 description = \"Stand-alone Assertions\"\n5 readm(...truncated)"
      }
    ],
    "CommitSHA": "38849833a92ef24b342e0ca59a8fa5752a07bab4"
  },
  "Score": {
    "Difficulty": "Medium",
    "issue_origin": {
      "Title": 7,
      "Description": 8,
      "Reproducibility": 8,
      "Relevance": 7,
      "Explanation": 8,
      "Overall": 7
    },
    "issue_message": {
      "Title": 6,
      "Description": 6,
      "Reproducibility": 5,
      "Relevance": 7,
      "Explanation": 7,
      "Overall": 7
    },
    "issue_ground": {
      "Title": 8,
      "Description": 8,
      "Reproducibility": 7,
      "Relevance": 8,
      "Explanation": 8,
      "Overall": 8
    },
    "issue_ground_truth": {
      "title": "Inconsistent Boolean Assertions Cause False Failures",
      "description": "There is an issue in the assertion library where the `assert_boolean_true` function inconsistently evaluates expressions, causing unexpected assertion failures. Specifically, the function currently evaluates expressions using a truthy check (`if not expr:`), which leads to expressions that are not strictly `True` being misinterpreted. Users may experience assertion failures even when the expression should be considered true in the context of boolean logic.\n\nThis impacts the reliability of tests that use `assert_boolean_true`, causing tests to fail incorrectly. For instance, expressions like non-zero integers or non-empty strings, which may be truthy, will cause `assert_boolean_true` to fail.\n\nThe expected behavior is for `assert_boolean_true` to strictly check for the `True` constant, ensuring that only expressions that exactly equal `True` pass the assertion. This change should increase the precision and reliability of boolean assertions, particularly in test cases where the exact value of `True` is necessary.\n\nThis issue needs to be addressed to ensure that `assert_boolean_true` works as intended, only passing when the expression is explicitly `True` and not merely truthy.",
      "explanation": "### Summary of the Issue\n\nThe issue titled **\"Inconsistent Boolean Assertions Cause False Failures\"** describes a problem where the `assert_boolean_true` function in an assertion library of the `srittau_python-asserts` package incorrectly evaluates boolean expressions. This function uses a general truthy check (`if not expr:`), which causes false assertion failures for expressions that are truthy but not strictly equal to the `True` constant. For example, non-zero integers or non-empty strings are truthy but not strictly `True`, resulting in unexpected assertion failures.\n\n### Detailed Content of the Commit\n\nThe commit information indicates that the developer modified the behavior of the `assert_boolean_true` function. The commit includes the following actions:\n- **Changed the evaluation method**: The function was originally using `if not expr:` to evaluate the expression. This check was changed to `if expr is not True:`.\n\n### Explanation of the Cause and Solution\n\n#### Cause of the Issue\n\nThe root cause of the issue lies in the way the `assert_boolean_true` function evaluates expressions. The use of `if not expr:` in Python checks for truthiness, meaning it interprets any value that is not `False`, `None`, `0`, `''` (empty string), or an empty collection as `True`. This is a broad check and includes many values (such as `1`, non-empty strings, and non-empty collections) that are truthy but not strictly the boolean `True` constant. Consequently, users experienced false assertion failures when using `assert_boolean_true` with expressions that were truthy but not exactly `True`.\n\n#### Solution Implemented in the Commit\n\nTo solve this issue, the commit made a simple but critical change to the logic of the `assert_boolean_true` function:\n- **Changed the evaluation logic**: The check was updated to `if expr is not True:`. \n\nThis adjustment ensures that the function now strictly checks for the boolean `True` value, rather than any truthy value. This means that `assert_boolean_true` will only pass when the expression is explicitly `True` and will fail for any other truthy values (like `1`, `\"non-empty string\"`, etc.).\n\n#### How the Commit Solves the Issue\n\nThe commit addresses the problem by refining the expression evaluation logic in the `assert_boolean_true` function. The switch from `if not expr:` to `if expr is not True:` ensures that only expressions that are strictly `True` (the boolean constant) will pass the assertion. This change eliminates false failures caused by truthy expressions that are not strictly `True`. Therefore, test reliability is improved, and the function behaves as expected by only passing when the tested expression is the boolean `True`.\n\n### Final Explanation of the Solution\n\nThe solution comprehensively resolves the issue by updating the expression logic in the assertion function. This ensures that the function now adheres to the intended check for the boolean `True` rather than any broadly truthy value. Users can now rely on `assert_boolean_true` for precise boolean checks without encountering unexpected assertion failures due to truthy but not strictly `True` values. The fix makes the function's behavior more predictable and enhances the reliability of tests using this function."
    }
  }
}